From 791fae359862fadd342ffcd4eb41c52df0ae5bbc Mon Sep 17 00:00:00 2001 From: Levi Bostian Date: Mon, 23 Nov 2020 18:20:30 -0600 Subject: [PATCH] feat: build module for validating receipts --- .eslintrc.json | 3 +- .github/ISSUE_TEMPLATE/BUG_REPORT.md | 16 + .github/ISSUE_TEMPLATE/FEATURE_REQUEST.md | 16 + .github/ISSUE_TEMPLATE/HELP.md | 16 + .github/ISSUE_TEMPLATE/config.yml | 1 + .gitignore | 2 + @types/tiny-json-http/index.d.ts | 15 + README.md | 115 +- app/constants.ts | 12 + app/index.test.ts | 7 - app/index.ts | 8 - app/underscore.test.ts | 96 + app/underscore.ts | 52 + app/verify_receipt.test.ts | 74 + app/verify_receipt.ts | 92 + .../apple_response/error.test.ts | 26 + app/verify_receipt/apple_response/error.ts | 53 + app/verify_receipt/apple_response/index.ts | 5 + .../apple_response/latest_receipt_info.ts | 18 + .../apple_response/pending_renewal_info.ts | 89 + app/verify_receipt/apple_response/receipt.ts | 127 + app/verify_receipt/apple_response/success.ts | 56 + .../auto_renewable_subscription/index.test.ts | 28 + .../auto_renewable_subscription/index.ts | 38 + .../subscription.test.ts | 397 +++ .../subscription.ts | 175 ++ .../transaction.test.ts | 312 ++ .../transactions.ts | 105 + app/verify_receipt/parse/error.test.ts | 141 + app/verify_receipt/parse/error.ts | 97 + .../parse/purchase/index.test.ts | 14 + app/verify_receipt/parse/purchase/index.ts | 20 + .../parse/purchase/transaction.test.ts | 131 + .../parse/purchase/transaction.ts | 54 + .../result/auto_renewable_subscription.ts | 184 ++ app/verify_receipt/parse/result/index.ts | 3 + .../parse/result/parsed_receipt.ts | 54 + app/verify_receipt/parse/result/purchase.ts | 28 + app/verify_receipt/parse/success.test.ts | 10 + app/verify_receipt/parse/success.ts | 22 + docs/api/assets/css/main.css | 2535 +---------------- docs/api/assets/js/main.js | 2331 +-------------- docs/api/assets/js/search.json | 36 +- docs/api/classes/appleerror.html | 364 +++ docs/api/classes/internalerror.html | 376 +++ .../enums/appleverifyreceipterrorcode.html | 381 +++ docs/api/globals.html | 748 ++++- docs/api/index.html | 748 ++++- .../appleinapppurchasetransaction.html | 616 ++++ .../interfaces/applependingrenewalinfo.html | 500 ++++ docs/api/interfaces/applereceipt.html | 625 ++++ .../appleverifyreceipterrorresponse.html | 262 ++ .../appleverifyreceiptsuccessfulresponse.html | 384 +++ .../interfaces/autorenewablesubscription.html | 567 ++++ docs/api/interfaces/parsedreceipt.html | 350 +++ docs/api/interfaces/productpurchase.html | 274 ++ .../productpurchasetransaction.html | 298 ++ .../verifyreceiptoptions.html} | 165 +- example/.nvmrc | 1 + example/README.md | 20 + example/app/index.ts | 29 + example/package.json | 18 + example/secrets.json.example | 4 + example/tsconfig.json | 18 + junit.xml | 7 - package-lock.json | 7 +- package.json | 7 +- samples/autorenewable_subscription.json | 18 + samples/autorenewable_subscription2.json | 18 + samples/bought_consumable.json | 306 ++ samples/bought_non_consumable.json | 319 +++ samples/consumable.json | 13 + samples/consumable2.json | 13 + ...auto_renewable_subscription_transaction.ts | 45 + samples/simple.json | 24 + samples/subscription_first_time_purchase.json | 106 + ...ption_multiple_subscription_purchases.json | 291 ++ samples/subscription_renewed.json | 354 +++ .../subscription_second_time_purchase.json | 125 + .../will_autorenew_pending_renewal_info.json | 6 + tsconfig.json | 2 + typedoc.json | 5 +- 82 files changed, 11076 insertions(+), 4952 deletions(-) create mode 100644 .github/ISSUE_TEMPLATE/BUG_REPORT.md create mode 100644 .github/ISSUE_TEMPLATE/FEATURE_REQUEST.md create mode 100644 .github/ISSUE_TEMPLATE/HELP.md create mode 100644 .github/ISSUE_TEMPLATE/config.yml create mode 100644 @types/tiny-json-http/index.d.ts create mode 100644 app/constants.ts delete mode 100644 app/index.test.ts delete mode 100644 app/index.ts create mode 100644 app/underscore.test.ts create mode 100644 app/underscore.ts create mode 100644 app/verify_receipt.test.ts create mode 100644 app/verify_receipt.ts create mode 100644 app/verify_receipt/apple_response/error.test.ts create mode 100644 app/verify_receipt/apple_response/error.ts create mode 100644 app/verify_receipt/apple_response/index.ts create mode 100644 app/verify_receipt/apple_response/latest_receipt_info.ts create mode 100644 app/verify_receipt/apple_response/pending_renewal_info.ts create mode 100644 app/verify_receipt/apple_response/receipt.ts create mode 100644 app/verify_receipt/apple_response/success.ts create mode 100644 app/verify_receipt/parse/auto_renewable_subscription/index.test.ts create mode 100644 app/verify_receipt/parse/auto_renewable_subscription/index.ts create mode 100644 app/verify_receipt/parse/auto_renewable_subscription/subscription.test.ts create mode 100644 app/verify_receipt/parse/auto_renewable_subscription/subscription.ts create mode 100644 app/verify_receipt/parse/auto_renewable_subscription/transaction.test.ts create mode 100644 app/verify_receipt/parse/auto_renewable_subscription/transactions.ts create mode 100644 app/verify_receipt/parse/error.test.ts create mode 100644 app/verify_receipt/parse/error.ts create mode 100644 app/verify_receipt/parse/purchase/index.test.ts create mode 100644 app/verify_receipt/parse/purchase/index.ts create mode 100644 app/verify_receipt/parse/purchase/transaction.test.ts create mode 100644 app/verify_receipt/parse/purchase/transaction.ts create mode 100644 app/verify_receipt/parse/result/auto_renewable_subscription.ts create mode 100644 app/verify_receipt/parse/result/index.ts create mode 100644 app/verify_receipt/parse/result/parsed_receipt.ts create mode 100644 app/verify_receipt/parse/result/purchase.ts create mode 100644 app/verify_receipt/parse/success.test.ts create mode 100644 app/verify_receipt/parse/success.ts create mode 100644 docs/api/classes/appleerror.html create mode 100644 docs/api/classes/internalerror.html create mode 100644 docs/api/enums/appleverifyreceipterrorcode.html create mode 100644 docs/api/interfaces/appleinapppurchasetransaction.html create mode 100644 docs/api/interfaces/applependingrenewalinfo.html create mode 100644 docs/api/interfaces/applereceipt.html create mode 100644 docs/api/interfaces/appleverifyreceipterrorresponse.html create mode 100644 docs/api/interfaces/appleverifyreceiptsuccessfulresponse.html create mode 100644 docs/api/interfaces/autorenewablesubscription.html create mode 100644 docs/api/interfaces/parsedreceipt.html create mode 100644 docs/api/interfaces/productpurchase.html create mode 100644 docs/api/interfaces/productpurchasetransaction.html rename docs/api/{modules/_app_index_.html => interfaces/verifyreceiptoptions.html} (96%) create mode 100644 example/.nvmrc create mode 100644 example/README.md create mode 100644 example/app/index.ts create mode 100644 example/package.json create mode 100644 example/secrets.json.example create mode 100644 example/tsconfig.json delete mode 100644 junit.xml create mode 100644 samples/autorenewable_subscription.json create mode 100644 samples/autorenewable_subscription2.json create mode 100644 samples/bought_consumable.json create mode 100644 samples/bought_non_consumable.json create mode 100644 samples/consumable.json create mode 100644 samples/consumable2.json create mode 100644 samples/parsed/auto_renewable_subscription_transaction.ts create mode 100644 samples/simple.json create mode 100644 samples/subscription_first_time_purchase.json create mode 100644 samples/subscription_multiple_subscription_purchases.json create mode 100644 samples/subscription_renewed.json create mode 100644 samples/subscription_second_time_purchase.json create mode 100644 samples/will_autorenew_pending_renewal_info.json diff --git a/.eslintrc.json b/.eslintrc.json index a611c5e..80e2f4f 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -15,7 +15,8 @@ "files": ["*.test.ts"], "rules": { "@typescript-eslint/naming-convention": "off", - "@typescript-eslint/no-explicit-any": "off" + "@typescript-eslint/no-explicit-any": "off", + "@typescript-eslint/no-var-requires": "off" } } ] diff --git a/.github/ISSUE_TEMPLATE/BUG_REPORT.md b/.github/ISSUE_TEMPLATE/BUG_REPORT.md new file mode 100644 index 0000000..717d6af --- /dev/null +++ b/.github/ISSUE_TEMPLATE/BUG_REPORT.md @@ -0,0 +1,16 @@ +--- +name: 🐛 Bug Report +about: If something isn't working as expected +--- + + + +### New Issue Checklist + +- [ ] What version are you using? (Run `npm ls dollabill-apple` to get it): +- [ ] I read [the documentation](https://github.com/levibostian/dollabill-apple/) +- [ ] I searched for [existing GitHub issues](https://github.com/levibostian/dollabill-apple/issues) + +### Issue Description + + diff --git a/.github/ISSUE_TEMPLATE/FEATURE_REQUEST.md b/.github/ISSUE_TEMPLATE/FEATURE_REQUEST.md new file mode 100644 index 0000000..5d171ca --- /dev/null +++ b/.github/ISSUE_TEMPLATE/FEATURE_REQUEST.md @@ -0,0 +1,16 @@ +--- +name: 💡 Feature request +about: If you have an idea or something to improve +--- + + + +### New Issue Checklist + +- [ ] What version are you using? (Run `npm ls dollabill-apple` to get it): +- [ ] I read [the documentation](https://github.com/levibostian/dollabill-apple/) to see if the project can already do what I am suggesting. +- [ ] I searched for [existing GitHub issues](https://github.com/levibostian/dollabill-apple/issues) to see if anyone else had the same idea. + +### What is the feature request you had in mind? + + diff --git a/.github/ISSUE_TEMPLATE/HELP.md b/.github/ISSUE_TEMPLATE/HELP.md new file mode 100644 index 0000000..20e1a17 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/HELP.md @@ -0,0 +1,16 @@ +--- +name: 😄 Help +about: You have a question about the project +--- + + + +### New Issue Checklist + +- [ ] What version are you using? (Run `npm ls dollabill-apple` to get it): +- [ ] I read [the documentation](https://github.com/levibostian/dollabill-apple/) to see if it could answer my question. +- [ ] I searched for [existing GitHub issues](https://github.com/levibostian/dollabill-apple/issues) to see if anyone else has asked the same question. + +### What's your question 😄 + + diff --git a/.github/ISSUE_TEMPLATE/config.yml b/.github/ISSUE_TEMPLATE/config.yml new file mode 100644 index 0000000..ec4bb38 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/config.yml @@ -0,0 +1 @@ +blank_issues_enabled: false \ No newline at end of file diff --git a/.gitignore b/.gitignore index 8b46845..72c1ed6 100644 --- a/.gitignore +++ b/.gitignore @@ -1,5 +1,7 @@ coverage/ dist/ +secrets.json +junit.xml # Logs logs diff --git a/@types/tiny-json-http/index.d.ts b/@types/tiny-json-http/index.d.ts new file mode 100644 index 0000000..49986d9 --- /dev/null +++ b/@types/tiny-json-http/index.d.ts @@ -0,0 +1,15 @@ +declare type Options = { + url: string + data?: any // form vars for tiny.post, tiny.put, tiny.patch, and tiny.delete otherwise querystring vars for tiny.get + headers?: any // key/value map used for headers (including support for uploading files with multipart/form-data) + buffer?: boolean // if set to true the response body is returned as a buffer + timeout?: number +} + +declare type Callback = (err?: Error, res: { headers: any; body: any }) => void + +export declare function post(options: Options): Promise +export declare function post(options: Options, callback: Callback): void + +export declare function get(options: Options): Promise +export declare function get(options: Options, callback: Callback): void diff --git a/README.md b/README.md index aa40cf2..5f88f8d 100644 --- a/README.md +++ b/README.md @@ -1,9 +1,112 @@ -# npm-module-blanky +# dolla bill -Opinionated boilerplate used to make and deploy npm modules. +Easily work with Apple in-app purchases. So you can more easily collect your "Dolla dolla bills, ya'll" -# Goals of this project +# Why use dolla bill? -* Contain configuration files to setup all tools I tend to use in my development flow. -* Clone, rename some files, and get developing! -* Start with zero dependencies. I try my best to keep all npm modules as slim as possible. +When you are accepting in-app purchases in an iOS mobile app, you need to verify the transaction and continuously update your customer's subscription status. This verification and parsing of the responses back from Apple is boilerplate. This small module exists to help you with these tasks. + +- [x] **No dependencies**. This module tries to be as small as possible. +- [x] **Fully tested**. Automated testing (with high test coverage) + QA testing gives you the confidence that the module will work well for you. +- [x] **Complete API documentation** to more easily use it in your code. +- [x] **Up-to-date** If and when Apple changes the way you perform verification or Apple adds more features to verification, update this module and your projects can take advantage of the changes quickly and easily. +- [x] **Typescript typings** This project is written in Typescript for that strict typing goodness. + +> Note: Technically there is [one dependency](https://github.com/brianleroux/tiny-json-http) but it's a _tiny wrapper around a built-in nodejs feature_. This module still remains super small with this. + +# Getting started + +```ts +import dollaBill from "dollabill" + +const receiptFromStoreKit = // The base64 encoded receipt data that StoreKit gave your app + +const appleResponse = await dollaBill.verifyReceipt({ + appPassword: process.env.APPLE_IN_APP_PURCHASE_PASSWORD, + receipt: receiptFromStoreKit +}) + +// If an error happens, it's fatal. It means that there is a bug with this library (create a GitHub issue with stacktrace and other info, please) or you the developer made a mistake when using this library. + +// It's recommended that you, the developer, views the error message and stacktrace to fix the issue. Note: The error here is not meant to be shown to your users. + +if (!appleResponse.isValid) { + // There was a problem. The request was valid and can be retried. + + // look at the error and act on it however you wish. + // It's recommended that you log the error and then return back a message to your users saying there was a problem. The error here is meant for the developer to see, not a good error to return back to the user. + appleResponse.error +} else { + // Time for you to update your database with the status of your customer and their subscription. + // This is easy because dolla bill parses the response from Apple to be easily readable. + // Check out the API documentation to learn about what `appleResponse` properties there are. +} +``` + +```ts +type Response = AppleResponse | AppleError + +enum AppleErrorType {} +// TODO + +type AppleError = { + code: number + type: AppleErrorType +} + +type AppleResponse = { + isValid: boolean + autoRenewableSubscription?: AutoRenewableSubscription // If your in-app purchase is for auto renewable subscriptions, go here. + decodedReceipt: Receipt + raw: any // just in case you need it, here is the raw JSON response from Apple. https://developer.apple.com/documentation/appstorereceipts/responsebody +} + +type AutoRenewableSubscription = { + transactions: Array + pendingRenewals: Array + isGracePeriod: bool + gracePeriod: GracePeriodInfo +} + +type GracePeriodInfo = { + // TODO + // make this up. it comes from: https://developer.apple.com/documentation/appstorereceipts/responsebody/pending_renewal_info +} + +type PendingRenewal = { + // TODO + // https://developer.apple.com/documentation/appstorereceipts/responsebody/pending_renewal_info +} + +type Transaction = { + // TODO + // https://developer.apple.com/documentation/appstorereceipts/responsebody/latest_receipt_info +} + +type Receipt = { + // TODO + // https://developer.apple.com/documentation/appstorereceipts/responsebody/receipt +} +``` + +# Documentation + +// TODO + +## Contributors + +Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)) + + + + + + + + +

Levi Bostian

💻 📖 🚧
+ + + + + diff --git a/app/constants.ts b/app/constants.ts new file mode 100644 index 0000000..45b017a --- /dev/null +++ b/app/constants.ts @@ -0,0 +1,12 @@ +/** + * @internal + */ +export const bugReportLink = `https://github.com/levibostian/dollabill-apple/issues/new?template=BUG_REPORT.md` +/** + * @internal + */ +export const appleProductionVerifyReceiptEndpoint = "https://buy.itunes.apple.com/verifyReceipt" +/** + * @internal + */ +export const appleSandboxVerifyReceiptEndpoint = "https://sandbox.itunes.apple.com/verifyReceipt" diff --git a/app/index.test.ts b/app/index.test.ts deleted file mode 100644 index d1e7cdf..0000000 --- a/app/index.test.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { hello } from "." - -describe(`hello`, () => { - it(`given text expect result`, () => { - expect(hello("world")).toEqual("hello world") - }) -}) diff --git a/app/index.ts b/app/index.ts deleted file mode 100644 index 0a961eb..0000000 --- a/app/index.ts +++ /dev/null @@ -1,8 +0,0 @@ -/** - * Simple function that returns back "hello X" - * @param text The text to combine with "hello" - * @return combined text "hello {text}" - */ -export const hello = (text: string): string => { - return `hello ${text}` -} diff --git a/app/underscore.test.ts b/app/underscore.test.ts new file mode 100644 index 0000000..3670ddb --- /dev/null +++ b/app/underscore.test.ts @@ -0,0 +1,96 @@ +import _ from "./underscore" + +describe("array", () => { + describe("contains", () => { + it(`given empty array, expect false`, async () => { + const given: string[] = [] + + const actual = _.array.contains(given, (item) => item === "foo") + + expect(actual).toEqual(false) + }) + it(`given array without item, expect false`, async () => { + const given: string[] = ["bar"] + + const actual = _.array.contains(given, (item) => item === "foo") + + expect(actual).toEqual(false) + }) + it(`given array with item, expect true`, async () => { + const given: string[] = ["bar", "foo"] + + const actual = _.array.contains(given, (item) => item === "foo") + + expect(actual).toEqual(true) + }) + }) +}) + +describe("date", () => { + describe("sortOldToNew", () => { + it(`given empty array, expect empty array`, async () => { + const given: Date[] = [] + + given.sort(_.date.sortOldToNew) + + expect(given).toEqual([]) + }) + it(`given items already in order, expect unchanged`, async () => { + const given: Date[] = [new Date('2018-04-04T16:00:00.000Z'), new Date('2020-04-04T16:00:00.000Z')] + const expected = given + + given.sort(_.date.sortOldToNew) + + expect(given).toEqual(expected) + }) + it(`given items not in order, expect to sort`, async () => { + const given: Date[] = [new Date('2020-04-04T16:00:00.000Z'), new Date('2018-04-04T16:00:00.000Z')] + const expected = [given[1], given[0]] + + given.sort(_.date.sortOldToNew) + + expect(given).toEqual(expected) + }) + it(`given identical dates, expect unchanged`, async () => { + const given: Date[] = [new Date('2018-04-04T16:00:00.000Z'), new Date('2018-04-04T16:00:00.000Z')] + const expected = given + + given.sort(_.date.sortOldToNew) + + expect(given).toEqual(expected) + }) + }) + describe("sortNewToOld", () => { + it(`given empty array, expect empty array`, async () => { + const given: Date[] = [] + + given.sort(_.date.sortNewToOld) + + expect(given).toEqual([]) + }) + it(`given items already in order, expect unchanged`, async () => { + const given: Date[] = [new Date('2020-04-04T16:00:00.000Z'), new Date('2018-04-04T16:00:00.000Z')] + const expected = given + + given.sort(_.date.sortNewToOld) + + expect(given).toEqual(expected) + }) + it(`given items not in order, expect to sort`, async () => { + const given: Date[] = [new Date('2018-04-04T16:00:00.000Z'), new Date('2020-04-04T16:00:00.000Z')] + const expected = [given[1], given[0]] + + given.sort(_.date.sortNewToOld) + + expect(given).toEqual(expected) + }) + it(`given identical dates, expect unchanged`, async () => { + const given: Date[] = [new Date('2018-04-04T16:00:00.000Z'), new Date('2018-04-04T16:00:00.000Z')] + const expected = given + + given.sort(_.date.sortNewToOld) + + expect(given).toEqual(expected) + }) + }) +}) \ No newline at end of file diff --git a/app/underscore.ts b/app/underscore.ts new file mode 100644 index 0000000..57bcb4c --- /dev/null +++ b/app/underscore.ts @@ -0,0 +1,52 @@ +const _ = { + array: { + contains(array: T[], doesContain: (element: T) => boolean): boolean { + let result = false + + array.forEach((element) => { + if (doesContain(element)) { + result = true + } + }) + + return result + } + }, + date: { + /** + * Sort date array in order: [older date, newer date]. + * + * Use: + * ``` + * const dates: Date[] = [] + * dates.sort(_.date.sortOldToNew) + * + * const nestedDates: {date: Date}[] = [] + * nestedDates.sort((first, second) => _.date.sortOldToNew(first.date, second.date)) + * ``` + */ + sortOldToNew(first: Date, second: Date): number { + return first.getTime() - second.getTime() + }, + /** + * Sort date array in order: [newer date, older date]. + * + * Use: + * ``` + * const dates: Date[] = [] + * dates.sort(_.date.sortNewToOld) + * + * const nestedDates: {date: Date}[] = [] + * nestedDates.sort((first, second) => _.date.sortNewToOld(first.date, second.date)) + * ``` + */ + sortNewToOld(first: Date, second: Date): number { + return second.getTime() - first.getTime() + } + } +} + +/** + * @internal + */ +export default _ diff --git a/app/verify_receipt.test.ts b/app/verify_receipt.test.ts new file mode 100644 index 0000000..d7db6e0 --- /dev/null +++ b/app/verify_receipt.test.ts @@ -0,0 +1,74 @@ +import { verifyReceipt, isFailure } from "./verify_receipt" +import * as constants from "./constants" + +jest.mock('tiny-json-http') +import http from "tiny-json-http" +import { AppleVerifyReceiptErrorCode } from "./verify_receipt/apple_response" +import { AppleError } from "./verify_receipt/parse/error" + +describe("verifyReceipt", () => { + it(`given apple responds saying to use testing environment, expect to use testing environment in 2nd attempt`, async () => { + const postMock = jest.fn().mockResolvedValueOnce({ + status: AppleVerifyReceiptErrorCode.USE_TEST_ENVIRONMENT + }).mockResolvedValueOnce(require('../samples/simple')) + http.post = postMock + + await verifyReceipt("", "") + + expect(postMock.mock.calls).toHaveLength(2) + expect(postMock.mock.calls[0][0].url).toEqual(constants.appleProductionVerifyReceiptEndpoint) + expect(postMock.mock.calls[1][0].url).toEqual(constants.appleSandboxVerifyReceiptEndpoint) + }) + it(`given apple responds with non-200 response, expect receive error`, async () => { + const givenError = new Error("Non-200") + const postMock = jest.fn().mockRejectedValueOnce(givenError) + http.post = postMock + + const actual = await verifyReceipt("", "") + + expect(isFailure(actual)).toEqual(true) + expect(postMock.mock.calls).toHaveLength(1) + }) + it(`expect use production environment by default`, async () => { + const postMock = jest.fn().mockResolvedValueOnce(require('../samples/simple')) + http.post = postMock + + await verifyReceipt("", "") + + expect(postMock.mock.calls).toHaveLength(1) + expect(postMock.mock.calls[0][0].url).toEqual(constants.appleProductionVerifyReceiptEndpoint) + }) + it(`given an apple error response, expect receipt the error result`, async () => { + const postMock = jest.fn().mockResolvedValueOnce({ + status: AppleVerifyReceiptErrorCode.WRONG_SHARED_SECRET + }) + http.post = postMock + + const actual = await verifyReceipt("", "") + + expect(postMock.mock.calls).toHaveLength(1) + if (!isFailure(actual)) throw new Error('should be a failure') + expect((actual.error as AppleError).appleErrorCode).toEqual(AppleVerifyReceiptErrorCode.WRONG_SHARED_SECRET) + }) + it(`given apple returns unknown error code, expect get an error result`, async () => { + const postMock = jest.fn().mockResolvedValueOnce({ + status: 21199 + }) + http.post = postMock + + const actual = await verifyReceipt("", "") + + expect(postMock.mock.calls).toHaveLength(1) + if (!isFailure(actual)) throw new Error('should be a failure') + expect((actual.error as AppleError).appleErrorCode).toEqual(21199) + }) + it(`given successful request, expect to get parsed successful result`, async () => { + const postMock = jest.fn().mockResolvedValueOnce(require("../samples/simple.json")) + http.post = postMock + + const actual = await verifyReceipt("", "") + + expect(postMock.mock.calls).toHaveLength(1) + expect(isFailure(actual)).toEqual(false) + }) +}) diff --git a/app/verify_receipt.ts b/app/verify_receipt.ts new file mode 100644 index 0000000..0929ee5 --- /dev/null +++ b/app/verify_receipt.ts @@ -0,0 +1,92 @@ +import http from "tiny-json-http" +import { + appleProductionVerifyReceiptEndpoint, + appleSandboxVerifyReceiptEndpoint +} from "./constants" +import { + AppleVerifyReceiptErrorCode, + isAppleResponseError +} from "./verify_receipt/apple_response/error" +import { AppleVerifyReceiptResponse } from "./verify_receipt/apple_response/success" +import { handleErrorResponse } from "./verify_receipt/parse/error" +import { ParsedReceipt } from "./verify_receipt/parse/result/parsed_receipt" +import { parseSuccess } from "./verify_receipt/parse/success" + +interface VerifyReceiptOptions { + production: boolean +} + +const runVerifyReceipt = async ( + receipt: string, + sharedSecret: string, + options: VerifyReceiptOptions +): Promise => { + let url = appleSandboxVerifyReceiptEndpoint + if (options.production) { + url = appleProductionVerifyReceiptEndpoint + } + + let verifyReceiptResponse: AppleVerifyReceiptResponse + try { + verifyReceiptResponse = await http.post({ + url, + data: { + "receipt-data": receipt, + password: sharedSecret + } + }) + } catch (httpError) { // https://github.com/brianleroux/tiny-json-http/blob/0c2b3372d8b9a838ac9a63645bf5212427c0ccf3/_write.js#L123-L126 + return { + error: httpError + } + } + + if (isAppleResponseError(verifyReceiptResponse)) { + if (verifyReceiptResponse.status == AppleVerifyReceiptErrorCode.USE_TEST_ENVIRONMENT) { + return runVerifyReceipt(receipt, sharedSecret, { production: false }) + } + + const parsedError = handleErrorResponse(verifyReceiptResponse) + + return { + error: parsedError + } + } else { + return parseSuccess(verifyReceiptResponse) + } +} + +/** + * Represents a failed attempt from {@link verifyReceipt}. + */ +export type VerifyReceiptFailure = { + error: Error +} +/** + * The response type of {@link verifyReceipt}. A failure or a success. + * + * Make sure to use {@link isFailure} after you get this response. + */ +export type VerifyReceiptResponse = ParsedReceipt | VerifyReceiptFailure + +/** + * Verify an in-app purchase receipt. This is a major feature of this library. + * + * This function will send the receipt to Apple's server and parse the result into something that is much more useful for you, the developer. This allow you to implement in-app purchases faster in your app with server-side verification. + */ +export const verifyReceipt = ( + receipt: string, + sharedSecret: string +): Promise => { + return runVerifyReceipt(receipt, sharedSecret, { production: true }) +} + +/** + * Determine if a {@link VerifyReceiptResponse} was a failed response or a success. + */ +export const isFailure = (response: VerifyReceiptResponse): response is VerifyReceiptFailure => { + return (response as VerifyReceiptFailure).error !== undefined +} + +export * from "./verify_receipt/parse/result" +export * from "./verify_receipt/apple_response" diff --git a/app/verify_receipt/apple_response/error.test.ts b/app/verify_receipt/apple_response/error.test.ts new file mode 100644 index 0000000..6e0913e --- /dev/null +++ b/app/verify_receipt/apple_response/error.test.ts @@ -0,0 +1,26 @@ +import { AppleVerifyReceiptErrorCode, isAppleErrorCode, isAppleResponseError } from "./error" +import { AppleVerifyReceiptSuccessfulStatus } from "./success" + +describe("isAppleErrorCode", () => { + it(`given not error, expect false`, async () => { + expect(isAppleErrorCode(AppleVerifyReceiptSuccessfulStatus.SUCCESS)).toEqual(false) + expect(isAppleErrorCode(AppleVerifyReceiptSuccessfulStatus.VALID_BUT_SUBSCRIPTION_EXPIRED)).toEqual(false) + }) + it(`given error, expect true`, async () => { + expect(isAppleErrorCode(AppleVerifyReceiptErrorCode.WRONG_SHARED_SECRET)).toEqual(true) + }) + it(`given random error code library not aware of`, async () => { + expect(isAppleErrorCode(21199)).toEqual(true) + }) +}) + +describe("isAppleResponseError", () => { + it(`given no receipt, expect false`, async () => { + expect(isAppleResponseError({ + status: AppleVerifyReceiptErrorCode.NOT_POST + })).toEqual(true) + }) + it(`given receipt, expect true`, async () => { + expect(isAppleResponseError(require('../../../samples/subscription_first_time_purchase.json'))).toEqual(false) + }) +}) \ No newline at end of file diff --git a/app/verify_receipt/apple_response/error.ts b/app/verify_receipt/apple_response/error.ts new file mode 100644 index 0000000..c0b5942 --- /dev/null +++ b/app/verify_receipt/apple_response/error.ts @@ -0,0 +1,53 @@ +import { AppleVerifyReceiptResponse, AppleVerifyReceiptSuccessfulStatus } from "./success" + +/** + * All of the statuses that are not successful. Apple was not able to provide back receipt information. + * + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/status) + * + * > Note: There may be many more error codes then these. These are ones that are documented but Apple says that there are also codes in the range 21100-21199 that could be returned. + */ +export enum AppleVerifyReceiptErrorCode { + NOT_POST = 21000, + SHOULD_NOT_HAPPEN = 21001, + INVALID_RECEIPT_OR_DOWN = 21002, + UNAUTHORIZED = 21003, + WRONG_SHARED_SECRET = 21004, + SERVICE_DOWN = 21005, + USE_TEST_ENVIRONMENT = 21007, + USE_PRODUCTION_ENVIRONMENT = 21008, + APPLE_INTERNAL_ERROR = 21009, + CUSTOMER_NOT_FOUND = 21010 +} + +/** + * The response body of a request that had an error. The receipt was not decoded and returned. + */ +export interface AppleVerifyReceiptErrorResponse { + /** + * > Note: Type here is `number` because the status can be more then the options included in {@link AppleVerifyReceiptErrorCode}. + * + * See {@link AppleVerifyReceiptErrorCode} for documented options. + */ + status: number +} + +/** + * @internal + */ +export const isAppleResponseError = ( + response: AppleVerifyReceiptResponse +): response is AppleVerifyReceiptErrorResponse => { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + return (response as any).receipt === undefined +} + +/** + * @internal + */ +export const isAppleErrorCode = ( + code: number +): boolean => { + // Important that we only check if successful codes because there are a finite number of successful codes and unlimited error codes. + return !Object.values(AppleVerifyReceiptSuccessfulStatus).includes(code) +} diff --git a/app/verify_receipt/apple_response/index.ts b/app/verify_receipt/apple_response/index.ts new file mode 100644 index 0000000..aecfad7 --- /dev/null +++ b/app/verify_receipt/apple_response/index.ts @@ -0,0 +1,5 @@ +export * from "./error" +export * from "./latest_receipt_info" +export * from "./pending_renewal_info" +export * from "./receipt" +export * from "./success" diff --git a/app/verify_receipt/apple_response/latest_receipt_info.ts b/app/verify_receipt/apple_response/latest_receipt_info.ts new file mode 100644 index 0000000..b4e1cb3 --- /dev/null +++ b/app/verify_receipt/apple_response/latest_receipt_info.ts @@ -0,0 +1,18 @@ +import { AppleInAppPurchaseTransaction } from "./receipt" + +/** + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/responsebody/latest_receipt_info) + */ +export type AppleLatestReceiptInfo = AppleInAppPurchaseTransaction & { + /** + * Indicates a subscription has been cancelled because of an upgrade. + * + * > Note: Only present for an upgrade transaction. + */ + is_upgraded?: "true" + offer_code_ref_name?: string + /** + * > Note: Only present for an auto-renewable subscription. + */ + subscription_group_identifier?: string +} diff --git a/app/verify_receipt/apple_response/pending_renewal_info.ts b/app/verify_receipt/apple_response/pending_renewal_info.ts new file mode 100644 index 0000000..c5b21b7 --- /dev/null +++ b/app/verify_receipt/apple_response/pending_renewal_info.ts @@ -0,0 +1,89 @@ +/** + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/responsebody/pending_renewal_info) + */ +export interface ApplePendingRenewalInfo { + /** + * The product id the customer will downgrade/crossgrade to when the current subscription period is over. + * + * > Note: Present if the user is downgrading or crossgrading. + */ + auto_renew_product_id?: string + /** + * 1 - Subscription will auto-renew + * 0 - Customer turned off auto-renew + * + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/auto_renew_status) + */ + auto_renew_status: "1" | "0" + /** + * Reason subscription expired. + * + * 1 - Customer cancelled their subscription + * 2 - Billing error such as payment information not valid + * 3 - Customer did not agree to price increase + * 4 - Product not available for purchase + * 5 - Unknown error + * + * > Note: Present for a receipt that contains an expired auto-renewable subscription. + * + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/expiration_intent) + */ + expiration_intent?: "1" | "2" | "3" | "4" | "5" + /** + * When Apple will stop automatically retrying to renew the expired subscription. + * + * > Note: Present if in a grace period + */ + grace_period_expires_date?: string + /** + * When Apple will stop automatically retrying to renew the expired subscription. + * + * > Note: Present if in a grace period + */ + grace_period_expires_date_ms?: string + /** + * When Apple will stop automatically retrying to renew the expired subscription. + * + * > Note: Present if in a grace period + */ + grace_period_expires_date_pst?: string + /** + * If auto-renewable subscription is actively trying to be automatically renewed by Apple. Check the {@link expires_intent}, {@link expires_intent} to determine if Apple is trying to renew *before or after* the subscription has expired. If the subscription has expired, you are going to need to think through the logic of Grace Periods. Read more about Grace Periods to learn more about them. + * + * 1 - Apple is trying to renew the subscription. See {@link grace_period_expires_date} to determine when Apple will stop trying. + * 0 - Apple has stopped attempting to renew. + * + * > Note: Present if auto-renewable subscription has expired and Apple is or is not trying to renew it. + * + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/is_in_billing_retry_period) + */ + is_in_billing_retry_period?: "0" | "1" + /** + * > Note: Present if the customer redeemed [an offer code](https://developer.apple.com/app-store/subscriptions/#offer-codes). + * + * Resources: + * 1. [Overview of what offer codes are](https://developer.apple.com/app-store/subscriptions/#offer-codes) + * 2. [Set up offer codes](https://help.apple.com/app-store-connect/#/dev6a098e4b1) + * 3. [Implement offer codes in your app](https://developer.apple.com/documentation/storekit/in-app_purchase/subscriptions_and_offers/implementing_offer_codes_in_your_app) + */ + offer_code_ref_name?: string + /** + * The transaction ID that identifies a full payment history for a customer paying for a subscription through all of the renewals, upgrades/downgrades. + * + * [More details](https://developer.apple.com/documentation/appstorereceipts/original_transaction_id) + */ + original_transaction_id: string + /** + * 1 - customer accepted the price increase + * 0 - customer has not yet accepted the price increase + * + * > Note: Present if the customer was notified of price increase. When you enable a price increase for existing customers, Apple follows a schedule that it has set. [View the schedule here](https://help.apple.com/app-store-connect/#/devc9870599e) in the section "Increase the price of an auto-renewable subscription". + * + * See {@link expiration_intent} as the customer will have their subscription expire if they never accept the price increase. + */ + price_consent_status?: "0" | "1" + /** + * Product id this renewal information is referring to. + */ + product_id: string +} diff --git a/app/verify_receipt/apple_response/receipt.ts b/app/verify_receipt/apple_response/receipt.ts new file mode 100644 index 0000000..3bd2d82 --- /dev/null +++ b/app/verify_receipt/apple_response/receipt.ts @@ -0,0 +1,127 @@ +/** + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/responsebody/receipt/in_app) + */ +export interface AppleInAppPurchaseTransaction { + /** + * Date Apple issued a refund to customer. Refund could be because of customer asking for a refund or a subscription upgrade occurred. + * + * > Note: Only present if a refund was made. + */ + cancellation_date?: string + /** + * See {@link cancellation_date} + */ + cancellation_date_ms?: string + /** + * See {@link cancellation_date} + */ + cancellation_date_pst?: string + /** + * Reason for the refund issued for a customer asking to end their subscription immediately. + * + * 1 - Customer indicated they had a problem using your app and wanted a refund. + * 0 - Customer cancelled for another reason such as making purchase by mistake. + */ + cancellation_reason?: "0" | "1" + /** + * Time subscription expires or will renew. + * + * > Note: Only present if ths is an auto-renewable subscription purchase + */ + expires_date?: string + /** + * See {@link expires_date} + */ + expires_date_ms: string + /** + * See {@link expires_date} + */ + expires_date_pst: string + /** + * > Note: Only present if this is an auto-renewable subscription purchase + * + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/is_in_intro_offer_period) + */ + is_in_intro_offer_period?: "true" | "false" + /** + * > Note: Only present if this is an auto-renewable subscription purchase + * + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/is_trial_period) + */ + is_trial_period?: "true" | "false" + original_purchase_date: string + original_purchase_date_ms: string + original_purchase_date_pst: string + original_transaction_id: string + product_id: string + promotional_offer_id?: string + purchase_date: string + purchase_date_ms: string + purchase_date_pst: string + /** + * Number of consumable products purchased. + * + * > Note: This value is usually present with purchases that are not consumables but it's an optional field here in case Apple changes that in the future and only includes it for consumable purchases, only. + */ + quantity?: string + transaction_id: string + /** + * > Note: Only present for subscription purchases + */ + web_order_line_item_id?: string +} + +/** + * A JSON representation of the receipt that was sent for verification. + * + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/responsebody/receipt) + */ +export interface AppleReceipt { + adam_id: number + app_item_id: number + application_version: string + bundle_id: string + download_id: number + /** + * > Note: Only present if app was purchased through the Volume Purchase Program. + */ + expiration_date?: string + /** + * See {@link expiration_date}. + */ + expiration_date_ms?: string + /** + * See {@link expiration_date}. + */ + expiration_date_pst?: string + /** + * Transactions for non-consumable, non-renewing subscriptions, and auto-renewing subscriptions previously purchased by the customer. + * + * This response is similar but different then `latest_receipt_info` where that is used most often for auto-renewable subscription transactions to see if a subscription is up-to-date or not. + */ + in_app?: AppleInAppPurchaseTransaction[] + original_application_version: string + original_purchase_date: string + original_purchase_date_ms: string + original_purchase_date_pst: string + /** + * > Note: Only present if the app was ordered through pre-order. + */ + preorder_date?: string + /** + * See {@link preorder_date} + */ + preorder_date_ms?: string + /** + * See {@link preorder_date} + */ + preorder_date_pst?: string + receipt_creation_date: string + receipt_creation_date_ms: string + receipt_creation_date_pst: string + receipt_type: "Production" | "ProductionVPP" | "ProductionSandbox" | "ProductionVPPSandbox" + request_date: string + request_date_ms: string + request_date_pst: string + version_external_identifier: number +} diff --git a/app/verify_receipt/apple_response/success.ts b/app/verify_receipt/apple_response/success.ts new file mode 100644 index 0000000..4b305c7 --- /dev/null +++ b/app/verify_receipt/apple_response/success.ts @@ -0,0 +1,56 @@ +import { AppleVerifyReceiptErrorResponse } from "./error" +import { AppleLatestReceiptInfo } from "./latest_receipt_info" +import { ApplePendingRenewalInfo } from "./pending_renewal_info" +import { AppleReceipt } from "./receipt" + +/** + * All of the statuses that are successful. Apple is able to return back receipt information when any of these status codes occur. + * + * > Note: 0 means the receipt is valid. 21006 means it's valid, but the subscription is expired. You will still receive back a receipt decoded in the response with this status code. + * + * [Official Apple documentation](https://developer.apple.com/documentation/appstorereceipts/status) + */ +export enum AppleVerifyReceiptSuccessfulStatus { + SUCCESS = 0, + NOTING_PURCHASED = 2, // Not documented, but I guess it can happen: https://github.com/voltrue2/in-app-purchase/blob/e966ee1348bd4f67581779abeec59c4bbc2b2ebc/lib/apple.js#L15 + VALID_BUT_SUBSCRIPTION_EXPIRED = 21006 +} + +export interface AppleVerifyReceiptSuccessfulResponse { + /** + * The environment that Apple made the Receipt for. + */ + environment: "Sandbox" | "Production" + /** + * Base64 encoded string for the latest receipt from Apple. + * + * > Note: Only present if the receipt contains auto-renewable subscriptions. + */ + latest_receipt?: string + /** + * All in-app purchase transactions except transactions for consumable products marked as finished by your app. + * + * This is the preferred place to look for the status of subscriptions and non-consumables. The field is similar but different from the transactions list in `receipt > in_app`. This field is the *latest transactions up to this moment* while the `in_app` field contains the transactions for the given receipt that got sent to Apple to process, only. If the receipt you sent up to get verified is out-of-date then `in_app` will be out-of-date. + * + * > Note: This is only present if the customer has purchased auto-renewable subscriptions. + */ + latest_receipt_info?: AppleLatestReceiptInfo[] + /** + * All auto-renewable subscription status of their renewals. + * + * > Note: Only present if the receipt contains auto-renewable subscriptions. + */ + pending_renewal_info?: ApplePendingRenewalInfo[] + /** + * JSON version of the receipt that was sent to Apple for verifying. + */ + receipt: AppleReceipt + /** + * Determines if the receipt is a valid one, or there is some other result. Maybe the receipt is not valid, maybe you the developer made a mistake, maybe the Apple server encountered a problem. + */ + status: AppleVerifyReceiptSuccessfulStatus +} + +export type AppleVerifyReceiptResponse = + | AppleVerifyReceiptErrorResponse + | AppleVerifyReceiptSuccessfulResponse diff --git a/app/verify_receipt/parse/auto_renewable_subscription/index.test.ts b/app/verify_receipt/parse/auto_renewable_subscription/index.test.ts new file mode 100644 index 0000000..5644502 --- /dev/null +++ b/app/verify_receipt/parse/auto_renewable_subscription/index.test.ts @@ -0,0 +1,28 @@ +import { AppleVerifyReceiptSuccessfulResponse } from "../../apple_response" +import { parseSubscriptions } from "." + +describe("parseSubscriptions", () => { + it(`given no latest receipt info, expect empty`, async () => { + expect(parseSubscriptions(require("../../../../samples/simple.json"))).toEqual([]) + }) + it(`given subscription transactions, expect get parsed subscription responses`, async () => { + const actual = parseSubscriptions(require("../../../../samples/subscription_first_time_purchase.json")) + + expect(actual).toHaveLength(1) + expect(actual[0].allTransactions).toHaveLength(3) + }) + it(`given upgrade/downgrades, expect subscription maps to pending renewal info correctly`, async () => { + const actual = parseSubscriptions(require("../../../../samples/subscription_multiple_subscription_purchases.json")) + + expect(actual).toHaveLength(1) + expect(actual[0].currentProductId).toEqual("test_sub2") + expect(actual[0].willAutoRenew).toEqual(true) + }) + it(`given cannot find pending renewal info, expect no errors`, async () => { + const response: AppleVerifyReceiptSuccessfulResponse = require("../../../../samples/subscription_multiple_subscription_purchases.json") + delete response.pending_renewal_info + const actual = parseSubscriptions(response) + + expect(actual).toHaveLength(1) + }) +}) \ No newline at end of file diff --git a/app/verify_receipt/parse/auto_renewable_subscription/index.ts b/app/verify_receipt/parse/auto_renewable_subscription/index.ts new file mode 100644 index 0000000..3556a3e --- /dev/null +++ b/app/verify_receipt/parse/auto_renewable_subscription/index.ts @@ -0,0 +1,38 @@ +import { AppleVerifyReceiptSuccessfulResponse } from "../../apple_response/success" +import { + AutoRenewableSubscription, +} from "../result/auto_renewable_subscription" +import { ParsedTransactions } from "./transactions" +import { ParsedSubscription } from "./subscription" + +/** + * @internal + */ +export const parseSubscriptions = ( + response: AppleVerifyReceiptSuccessfulResponse +): AutoRenewableSubscription[] => { + if (!response.latest_receipt_info) return [] + + const parsedTransactions = new ParsedTransactions() + + response.latest_receipt_info + .filter((transaction) => ParsedTransactions.isAutoRenewableSubscriptionTransaction(transaction)) + .forEach((subscription) => parsedTransactions.addTransaction(subscription)) + + return Array.from(parsedTransactions.iterateParsedTransactions()).map((allTransactions) => { + const sampleTransaction = allTransactions[0] + const originalTransactionId = sampleTransaction.originalTransactionId + const isEligibleIntroductoryOffer = parsedTransactions.getIsEligibleIntroductoryOffer(sampleTransaction)! + + const renewalInfo = response.pending_renewal_info?.find( + (entry) => entry.original_transaction_id === originalTransactionId + ) + + return new ParsedSubscription( + originalTransactionId, + isEligibleIntroductoryOffer, + allTransactions, + renewalInfo + ).parseSubscription() + }) +} diff --git a/app/verify_receipt/parse/auto_renewable_subscription/subscription.test.ts b/app/verify_receipt/parse/auto_renewable_subscription/subscription.test.ts new file mode 100644 index 0000000..ebf3df4 --- /dev/null +++ b/app/verify_receipt/parse/auto_renewable_subscription/subscription.test.ts @@ -0,0 +1,397 @@ +import { ApplePendingRenewalInfo } from "../../apple_response" +import { ParsedSubscription } from "./subscription" +import parsedTransaction, { + parsedTransaction2 +} from "../../../../samples/parsed/auto_renewable_subscription_transaction" +import { AutoRenewableSubscriptionIssues } from "../result" + +const pendingRenewalInfo: ApplePendingRenewalInfo = require("../../../../samples/will_autorenew_pending_renewal_info.json") +const exampleDateMsString = "1605230970000" +const dateInPast = new Date("2000-04-04T16:00:00.000Z") +const dateInFuture = new Date(Date.now() + (30 * 60 * 1000)) // add 30 min + +describe("ParsedSubscription", () => { + describe("constructor", () => { + it(`given no transactions, expect throw`, async () => { + expect(() => { + new ParsedSubscription("", false, []) + }).toThrowErrorMatchingInlineSnapshot( + `"BUG! Please, file a bug report: https://github.com/levibostian/dollabill-apple/issues/new?template=BUG_REPORT.md. (It's assumed that a subscription is not created if there are no transactions recorded for it.)"` + ) + }) + it(`given transactions, expect sort`, async () => { + const actual = new ParsedSubscription("1", false, [parsedTransaction2, parsedTransaction]) + + expect(actual.allTransactions).toEqual([parsedTransaction, parsedTransaction2]) + }) + it(`given transactions, expect populate properties`, async () => { + const actual = new ParsedSubscription( + "1", + false, + [parsedTransaction2, parsedTransaction], + pendingRenewalInfo + ) + + expect(actual.allTransactions).toEqual([parsedTransaction, parsedTransaction2]) + expect(actual.latestExpireDateTransaction).toEqual(parsedTransaction) + expect(actual.renewalInfo).toEqual(pendingRenewalInfo) + }) + }) + describe("usedOfferCodes", () => { + it(`given no offer codes, expect empty`, async () => { + parsedTransaction.offerCodeRefName = undefined + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.usedOfferCodes()).toEqual([]) + }) + it(`given offer codes, expect get unique codes`, async () => { + parsedTransaction.offerCodeRefName = "code-1" + parsedTransaction2.offerCodeRefName = "code-1" + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.usedOfferCodes()).toEqual(["code-1"]) + }) + }) + describe("usedPromotionalOffers", () => { + it(`given no offer codes, expect empty`, async () => { + parsedTransaction.promotionalOfferId = undefined + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.usedPromotionalOffers()).toEqual([]) + }) + it(`given offer codes, expect get unique codes`, async () => { + parsedTransaction.promotionalOfferId = "code-1" + parsedTransaction2.promotionalOfferId = "code-1" + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.usedPromotionalOffers()).toEqual(["code-1"]) + }) + }) + describe("priceIncreaseAccepted", () => { + it(`given no renewal info, expect undefined`, async () => { + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.priceIncreaseAccepted()).toBeUndefined() + }) + it(`given no price consent info, expect undefined`, async () => { + pendingRenewalInfo.price_consent_status = undefined + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.priceIncreaseAccepted()).toBeUndefined() + }) + it(`given accepted price consent, expect true`, async () => { + pendingRenewalInfo.price_consent_status = "1" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.priceIncreaseAccepted()).toEqual(true) + }) + it(`given not accepted price consent, expect false`, async () => { + pendingRenewalInfo.price_consent_status = "0" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.priceIncreaseAccepted()).toEqual(false) + }) + }) + describe("issues", () => { + it(`given no price consent info, expect does not include not accepting price increase`, async () => { + pendingRenewalInfo.price_consent_status = undefined + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.issues().notYetAcceptingPriceIncrease).toEqual(false) + }) + it(`given no pending renewal info, expect does not include not accepting price increase`, async () => { + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.issues().notYetAcceptingPriceIncrease).toEqual(false) + }) + it(`given user not yet consented to price increase, expect does include not accepting price increase`, async () => { + pendingRenewalInfo.price_consent_status = "0" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.issues().notYetAcceptingPriceIncrease).toEqual(true) + }) + it(`given subscription expired from a billing issue, expect billing issue`, async () => { + pendingRenewalInfo.expiration_intent = "2" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.issues().billingIssue).toEqual(true) + }) + it(`given no pending renewal info, expect no billing issue`, async () => { + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.issues().billingIssue).toEqual(false) + }) + it(`given is in grace period, expect billing issue`, async () => { + pendingRenewalInfo.grace_period_expires_date_ms = exampleDateMsString + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.issues().billingIssue).toEqual(true) + }) + it(`given subscription active, expect no billing issue`, async () => { + pendingRenewalInfo.expiration_intent = undefined + pendingRenewalInfo.grace_period_expires_date_ms = undefined + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.issues().billingIssue).toEqual(false) + }) + it(`given will not auto renew, expect will voluntary cancel`, async () => { + pendingRenewalInfo.auto_renew_status = "0" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.issues().willVoluntaryCancel).toEqual(true) + }) + it(`given no pending renewal info, expect no will voluntary cancel`, async () => { + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.issues().willVoluntaryCancel).toEqual(false) + }) + it(`given will auto renew, expect no will voluntary cancel`, async () => { + pendingRenewalInfo.auto_renew_status = "1" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.issues().willVoluntaryCancel).toEqual(false) + }) + }) + describe("issuesStrings", () => { + it(`given all false, expect empty`, async () => { + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + const givenStatuses: AutoRenewableSubscriptionIssues = { + notYetAcceptingPriceIncrease: false, + billingIssue: false, + willVoluntaryCancel: false + } + + expect(actual.issuesStrings(givenStatuses)).toEqual([]) + }) + it(`given notYetAcceptingPriceIncrease, expect notYetAcceptingPriceIncrease`, async () => { + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + const givenStatuses: AutoRenewableSubscriptionIssues = { + notYetAcceptingPriceIncrease: true, + billingIssue: true, + willVoluntaryCancel: true + } + + expect(actual.issuesStrings(givenStatuses)).toMatchInlineSnapshot(` + Array [ + "not_yet_accepting_price_increase", + "billing_issue", + "will_voluntary_cancel", + ] + `) + }) + }) + describe("willAutoRenew", () => { + it(`given no pending renewal info, expect false`, async () => { + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.willAutoRenew()).toEqual(false) + }) + it(`given will not auto renew, expect false`, async () => { + pendingRenewalInfo.auto_renew_status = "0" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.willAutoRenew()).toEqual(false) + }) + it(`given will auto renew, expect true`, async () => { + pendingRenewalInfo.auto_renew_status = "1" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.willAutoRenew()).toEqual(true) + }) + }) + describe("gracePeriodExpireDate", () => { + it(`given no pending renewal info, expect undefined`, async () => { + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.gracePeriodExpireDate()).toBeUndefined() + }) + it(`given in grace period, expect get result`, async () => { + pendingRenewalInfo.grace_period_expires_date_ms = exampleDateMsString + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.gracePeriodExpireDate()).toEqual(new Date(parseInt(exampleDateMsString))) + }) + }) + describe("expireDate", () => { + it(`expect get result from latest transaction`, async () => { + parsedTransaction.expiresDate = dateInFuture + parsedTransaction2.expiresDate = dateInPast + const actual = new ParsedSubscription("1", false, [parsedTransaction2, parsedTransaction], pendingRenewalInfo) + + expect(actual.expireDate()).toEqual(dateInFuture) + }) + }) + describe("currentEndDate", () => { + it(`given customer got refund or upgraded, expect date`, async () => { + parsedTransaction.cancelledDate = dateInPast + parsedTransaction.expiresDate = dateInFuture + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.currentEndDate()).toEqual(dateInPast) + }) + it(`given customer in grace period, expect date`, async () => { + parsedTransaction.cancelledDate = undefined + parsedTransaction.expiresDate = dateInFuture + pendingRenewalInfo.grace_period_expires_date_ms = exampleDateMsString + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.currentEndDate()).toEqual(new Date(parseInt(exampleDateMsString))) + }) + it(`given customer not in grace period or cancelled, expect expire date`, async () => { + parsedTransaction.cancelledDate = undefined + parsedTransaction.expiresDate = dateInFuture + pendingRenewalInfo.grace_period_expires_date_ms = undefined + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.currentEndDate()).toEqual(dateInFuture) + }) + }) + describe("willDowngradeToProductId", () => { + it(`given no pending renewal info, expect undefined`, async () => { + const actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.willDowngradeToProductId()).toBeUndefined() + }) + it(`given will auto renew to another product, expect result`, async () => { + pendingRenewalInfo.auto_renew_product_id = "other-product" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.willDowngradeToProductId()).toEqual("other-product") + }) + }) + describe("isInBillingRetry", () => { + it(`given customer in active status, expect false`, async () => { + pendingRenewalInfo.is_in_billing_retry_period = undefined + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.isInBillingRetry()).toEqual(false) + }) + it(`given customer beyond billing retry period, expect false`, async () => { + pendingRenewalInfo.is_in_billing_retry_period = "0" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.isInBillingRetry()).toEqual(false) + }) + it(`given customer in billing retry period, expect true`, async () => { + pendingRenewalInfo.is_in_billing_retry_period = "1" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.isInBillingRetry()).toEqual(true) + }) + }) + describe("isEligibleIntroductoryOffer", () => { + it(`given in constructor, expect result`, async () => { + // Note: Putting both tests in this 1 function to make sure that I am only modifying 1 single thing for each test. I don't want any false positives by tests running in random order and changing the result. Having multiple tests in 1 test function is usually bad practice. It's done on purpose here. + let actual = new ParsedSubscription("1", false, [parsedTransaction]) + + expect(actual.parseSubscription().isEligibleIntroductoryOffer).toEqual(false) + + // Only thing changed for this test is what we pass into the constructor. + actual = new ParsedSubscription("1", true, [parsedTransaction]) + + expect(actual.parseSubscription().isEligibleIntroductoryOffer).toEqual(true) + }) + }) + describe("status", () => { + it(`given in billing retry and in grace period, expect in grace period`, async () => { + pendingRenewalInfo.grace_period_expires_date = exampleDateMsString + pendingRenewalInfo.is_in_billing_retry_period == "1" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("grace_period") + }) + it(`given not in grace period but in billing retry, expect in billing retry`, async () => { + pendingRenewalInfo.grace_period_expires_date = undefined + pendingRenewalInfo.is_in_billing_retry_period == "1" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("billing_retry_period") + }) + it(`given not in grace period and not in billing retry and date expired, expect voluntary cancelled`, async () => { + pendingRenewalInfo.grace_period_expires_date = undefined + pendingRenewalInfo.is_in_billing_retry_period = undefined + pendingRenewalInfo.expiration_intent = undefined + parsedTransaction.cancelledDate = undefined + parsedTransaction.refundReason = undefined + parsedTransaction.expiresDate = dateInPast + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("voluntary_cancel") + }) + it(`given customer had billing issue, expect involuntary cancel`, async () => { + pendingRenewalInfo.grace_period_expires_date = undefined + pendingRenewalInfo.is_in_billing_retry_period = undefined + pendingRenewalInfo.expiration_intent = "2" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("involuntary_cancel") + }) + it(`given product not available for purchase, expect involuntary cancel`, async () => { + pendingRenewalInfo.grace_period_expires_date = undefined + pendingRenewalInfo.is_in_billing_retry_period = undefined + pendingRenewalInfo.expiration_intent = "4" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("involuntary_cancel") + }) + it(`given customer voluntarily cancelled, expect voluntary cancel`, async () => { + pendingRenewalInfo.grace_period_expires_date = undefined + pendingRenewalInfo.is_in_billing_retry_period = undefined + pendingRenewalInfo.expiration_intent = "1" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("voluntary_cancel") + }) + it(`given other expiration intent, expect other reason not active`, async () => { + pendingRenewalInfo.grace_period_expires_date = undefined + pendingRenewalInfo.is_in_billing_retry_period = undefined + pendingRenewalInfo.expiration_intent = "5" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("other_not_active") + }) + it(`given customer cancelled but did not upgrade, expect refund`, async () => { + pendingRenewalInfo.grace_period_expires_date = undefined + pendingRenewalInfo.is_in_billing_retry_period = undefined + pendingRenewalInfo.expiration_intent = undefined + parsedTransaction.cancelledDate = dateInPast + parsedTransaction.refundReason = "app_issue" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("refunded") + }) + it(`given customer cancelled and upgraded, expect upgrade`, async () => { + pendingRenewalInfo.grace_period_expires_date = undefined + pendingRenewalInfo.is_in_billing_retry_period = undefined + pendingRenewalInfo.expiration_intent = undefined + parsedTransaction.cancelledDate = dateInPast + parsedTransaction.refundReason = "upgrade" + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("upgraded") + }) + it(`given customer cancelled for other reason, expect other`, async () => { + pendingRenewalInfo.grace_period_expires_date = undefined + pendingRenewalInfo.is_in_billing_retry_period = undefined + pendingRenewalInfo.expiration_intent = undefined + parsedTransaction.cancelledDate = dateInPast + parsedTransaction.refundReason = undefined + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("other_not_active") + }) + it(`given no issues detected, expect active`, async () => { + pendingRenewalInfo.grace_period_expires_date = undefined + pendingRenewalInfo.is_in_billing_retry_period == undefined + pendingRenewalInfo.expiration_intent = undefined + parsedTransaction.cancelledDate = undefined + parsedTransaction.refundReason = undefined + parsedTransaction.expiresDate = dateInFuture + const actual = new ParsedSubscription("1", false, [parsedTransaction], pendingRenewalInfo) + + expect(actual.status()).toEqual("active") + }) + }) +}) diff --git a/app/verify_receipt/parse/auto_renewable_subscription/subscription.ts b/app/verify_receipt/parse/auto_renewable_subscription/subscription.ts new file mode 100644 index 0000000..4cc7978 --- /dev/null +++ b/app/verify_receipt/parse/auto_renewable_subscription/subscription.ts @@ -0,0 +1,175 @@ +import _ from "../../../underscore" +import { ApplePendingRenewalInfo } from "../../apple_response/pending_renewal_info" +import { InternalError } from "../error" +import { + AutoRenewableSubscription, + AutoRenewableSubscriptionIssues, + AutoRenewableSubscriptionIssueString, + AutoRenewableSubscriptionStatus, + AutoRenewableSubscriptionTransaction +} from "../result/auto_renewable_subscription" + +/** + * Creates a parsed subscription from a collection of parsed transactions. + * + * @internal + */ +export class ParsedSubscription { + public allTransactions: AutoRenewableSubscriptionTransaction[] + public latestExpireDateTransaction: AutoRenewableSubscriptionTransaction + public renewalInfo?: ApplePendingRenewalInfo + + constructor( + public originalTransactionId: string, + public isEligibleIntroductoryOffer: boolean, + transactions: AutoRenewableSubscriptionTransaction[], // the transactions for the originalTransactionId. + renewalInfo?: ApplePendingRenewalInfo + ) { + if (transactions.length <= 0) + throw new InternalError( + "PrerequisiteNotMet", + "It's assumed that a subscription is not created if there are no transactions recorded for it." + ) + + this.allTransactions = transactions.sort((first, second) => + _.date.sortNewToOld(first.expiresDate, second.expiresDate) + ) + this.latestExpireDateTransaction = this.allTransactions[0] // there will be at least 1 transaction or there would not be an entry in the Map<>. + this.renewalInfo = renewalInfo + } + + parseSubscription(): AutoRenewableSubscription { + const issues = this.issues() + + return { + currentProductId: this.latestExpireDateTransaction.productId, + originalTransactionId: this.originalTransactionId, + subscriptionGroup: this.latestExpireDateTransaction.subscriptionGroupId, + isInFreeTrialPeriod: this.latestExpireDateTransaction.inTrialPeriod, + isEligibleIntroductoryOffer: this.isEligibleIntroductoryOffer, + currentEndDate: this.currentEndDate(), + status: this.status(), + usedOfferCodes: this.usedOfferCodes(), + usedPromotionalOffers: this.usedPromotionalOffers(), + allTransactions: this.allTransactions, + latestExpireDateTransaction: this.latestExpireDateTransaction, + willAutoRenew: this.willAutoRenew(), + willDowngradeToProductId: this.willDowngradeToProductId(), + issuesStrings: this.issuesStrings(issues), + issues, + priceIncreaseAccepted: this.priceIncreaseAccepted(), + gracePeriodExpireDate: this.gracePeriodExpireDate(), + cancelledDate: this.cancelledDate(), + expireDate: this.expireDate(), + isInBillingRetry: this.isInBillingRetry() + } + } + + usedOfferCodes(): string[] { + const set = new Set(this.allTransactions + .filter((transaction) => transaction.offerCodeRefName) + .map((transaction) => transaction.offerCodeRefName!)) + + return [...set] + } + + usedPromotionalOffers(): string[] { + const set = new Set(this.allTransactions + .filter((transaction) => transaction.promotionalOfferId) + .map((transaction) => transaction.promotionalOfferId!)) + + return [...set] + } + + priceIncreaseAccepted(): boolean | undefined { + let priceIncreaseAccepted: boolean | undefined + if (this.renewalInfo?.price_consent_status) { + priceIncreaseAccepted = this.renewalInfo.price_consent_status === "1" + } + + return priceIncreaseAccepted + } + + issues(): AutoRenewableSubscriptionIssues { + return { + notYetAcceptingPriceIncrease: this.priceIncreaseAccepted() === false, + billingIssue: this.renewalInfo?.expiration_intent === "2" || this.gracePeriodExpireDate() !== undefined, + willVoluntaryCancel: this.renewalInfo !== undefined && !this.willAutoRenew() + } + } + + issuesStrings(statuses: AutoRenewableSubscriptionIssues): AutoRenewableSubscriptionIssueString[] { + const strings: AutoRenewableSubscriptionIssueString[] = [] + + if (statuses.notYetAcceptingPriceIncrease) strings.push("not_yet_accepting_price_increase") + if (statuses.billingIssue) strings.push("billing_issue") + if (statuses.willVoluntaryCancel) strings.push("will_voluntary_cancel") + + return strings + } + + willAutoRenew(): boolean { + return this.renewalInfo?.auto_renew_status === "1" || false + } + + gracePeriodExpireDate(): Date | undefined { + if (this.renewalInfo?.grace_period_expires_date_ms) return new Date(parseInt(this.renewalInfo.grace_period_expires_date_ms!)) + + return undefined + } + + expireDate(): Date { + return this.latestExpireDateTransaction.expiresDate + } + + currentEndDate(): Date { + return this.cancelledDate() || this.gracePeriodExpireDate() || this.expireDate() + } + + willDowngradeToProductId(): string | undefined { + return this.renewalInfo?.auto_renew_product_id + } + + isInBillingRetry(): boolean { + return this.renewalInfo?.is_in_billing_retry_period === "1" || false + } + + cancelledDate(): Date | undefined { + return this.latestExpireDateTransaction.cancelledDate + } + + status(): AutoRenewableSubscriptionStatus { + const inGracePeriod = this.renewalInfo?.grace_period_expires_date !== undefined || false + if (inGracePeriod) return "grace_period" + + const isInBillingRetry = this.renewalInfo?.is_in_billing_retry_period === "1" || false + if (isInBillingRetry) return "billing_retry_period" + + if (this.renewalInfo?.expiration_intent !== undefined) { + const involuntaryCancelled = this.renewalInfo.expiration_intent === "2" || this.renewalInfo.expiration_intent === "4" + if (involuntaryCancelled) return "involuntary_cancel" + + const voluntaryCancelled = this.renewalInfo.expiration_intent === "1" || false + if (voluntaryCancelled) return "voluntary_cancel" + + return "other_not_active" + } + + if (this.latestExpireDateTransaction.cancelledDate !== undefined || this.latestExpireDateTransaction.refundReason !== undefined) { + const refunded = this.latestExpireDateTransaction.refundReason === "app_issue" || this.latestExpireDateTransaction.refundReason === "other" + if (refunded) return "refunded" + + const upgraded = this.latestExpireDateTransaction.refundReason === "upgrade" + if (upgraded) return "upgraded" + + return "other_not_active" + } + + // Note: this should be caught by the "expiration_intent" of the pending renewal object but just in case, we want to catch when the customer has gone beyond the billing retry period and there is still a billing issue. + if (this.latestExpireDateTransaction.expiresDate.getTime() < new Date().getTime()) { + return "voluntary_cancel" + } + + return "active" + } +} \ No newline at end of file diff --git a/app/verify_receipt/parse/auto_renewable_subscription/transaction.test.ts b/app/verify_receipt/parse/auto_renewable_subscription/transaction.test.ts new file mode 100644 index 0000000..95c5a87 --- /dev/null +++ b/app/verify_receipt/parse/auto_renewable_subscription/transaction.test.ts @@ -0,0 +1,312 @@ +import { AppleLatestReceiptInfo } from "../../apple_response" +import { ParsedTransactions } from "./transactions" + +let parsedTransactions: ParsedTransactions +let givenSubscription: AppleLatestReceiptInfo +let givenSubscription2: AppleLatestReceiptInfo +const givenProductPurchase = require("../../../../samples/consumable.json") +const exampleDateMsString = "1605230970000" + +beforeEach(() => { + parsedTransactions = new ParsedTransactions() + givenSubscription = require("../../../../samples/autorenewable_subscription.json") + givenSubscription2 = require("../../../../samples/autorenewable_subscription2.json") +}) + +describe("ParsedTransactions", () => { + describe("addTransaction", () => { + it(`given a transaction that's not an auto-renewable subscription, expect throw error`, async () => { + expect(() => { + parsedTransactions.addTransaction(givenProductPurchase) + }).toThrowErrorMatchingInlineSnapshot( + `"BUG! Please, file a bug report: https://github.com/levibostian/dollabill-apple/issues/new?template=BUG_REPORT.md. (Filter out all transactions that are *not* subscription transactions before calling this function.)"` + ) + }) + it(`given add transactions from multiple subscriptions, expect grouped by subscription`, async () => { + parsedTransactions.addTransaction(givenSubscription) + parsedTransactions.addTransaction(givenSubscription2) + + expect(parsedTransactions.getParsedTransactions(givenSubscription)).not.toEqual( + parsedTransactions.getParsedTransactions(givenSubscription2) + ) + }) + it(`given add multiple transactions for same subscription, expect add transactions`, async () => { + parsedTransactions.addTransaction(givenSubscription) + parsedTransactions.addTransaction(givenSubscription) + + expect(parsedTransactions.getParsedTransactions(givenSubscription)).toHaveLength(2) + }) + }) + describe("isAutoRenewableSubscriptionTransaction", () => { + it(`given auto-renewable subscription, expect true`, async () => { + expect(ParsedTransactions.isAutoRenewableSubscriptionTransaction(givenSubscription)).toEqual( + true + ) + }) + it(`given product purchase, expect false`, async () => { + expect( + ParsedTransactions.isAutoRenewableSubscriptionTransaction(givenProductPurchase) + ).toEqual(false) + }) + }) + describe("updateIsEligibleIntroductoryOffer", () => { + // Tests that if we find it's not eligible, we never overwrite that value + it(`given transaction in trial period and a later transaction not in trial period, expect false`, async () => { + givenSubscription.is_in_intro_offer_period = "false" + givenSubscription.is_trial_period = "true" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + expect(parsedTransactions.getIsEligibleIntroductoryOffer(givenSubscription)).toEqual(false) + + givenSubscription.is_in_intro_offer_period = "false" + givenSubscription.is_trial_period = "false" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + expect(parsedTransactions.getIsEligibleIntroductoryOffer(givenSubscription)).toEqual(false) + }) + it(`given multiple transactions not in trial period or intro offer, expect true`, async () => { + givenSubscription.is_in_intro_offer_period = "false" + givenSubscription.is_trial_period = "false" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + expect(parsedTransactions.getIsEligibleIntroductoryOffer(givenSubscription)).toEqual(true) + + givenSubscription.is_in_intro_offer_period = "false" + givenSubscription.is_trial_period = "false" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + expect(parsedTransactions.getIsEligibleIntroductoryOffer(givenSubscription)).toEqual(true) + }) + it(`given transactions from same subscription groups not eligible, expect false`, async () => { + givenSubscription.is_in_intro_offer_period = "true" + givenSubscription.is_trial_period = "false" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + + givenSubscription.is_in_intro_offer_period = "false" + givenSubscription.is_trial_period = "false" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + + expect(parsedTransactions.getIsEligibleIntroductoryOffer(givenSubscription)).toEqual(false) + }) + it(`given transactions from separate subscription groups, expect different unique results`, async () => { + givenSubscription.is_in_intro_offer_period = "true" + givenSubscription.is_trial_period = "false" + givenSubscription.subscription_group_identifier = "1" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + expect(parsedTransactions.getIsEligibleIntroductoryOffer(givenSubscription)).toEqual(false) + + givenSubscription.is_in_intro_offer_period = "false" + givenSubscription.is_trial_period = "false" + givenSubscription.subscription_group_identifier = "2" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + expect(parsedTransactions.getIsEligibleIntroductoryOffer(givenSubscription)).toEqual(true) + }) + it(`given transaction, expect populate first result`, async () => { + givenSubscription.is_in_intro_offer_period = "false" + givenSubscription.is_trial_period = "false" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + expect(parsedTransactions.getIsEligibleIntroductoryOffer(givenSubscription)).toEqual(true) + }) + it(`given second transaction, expect update with new result`, async () => { + givenSubscription.is_in_intro_offer_period = "false" + givenSubscription.is_trial_period = "false" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + expect(parsedTransactions.getIsEligibleIntroductoryOffer(givenSubscription)).toEqual(true) + + givenSubscription.is_in_intro_offer_period = "true" + givenSubscription.is_trial_period = "false" + parsedTransactions.updateIsEligibleIntroductoryOffer(givenSubscription) + expect(parsedTransactions.getIsEligibleIntroductoryOffer(givenSubscription)).toEqual(false) + }) + }) + describe("refundReason", () => { + it(`given refunded transaction for upgrade, expect get upgrade refund reason`, async () => { + givenSubscription.cancellation_date_ms = exampleDateMsString + givenSubscription.is_upgraded = "true" + + expect(parsedTransactions.refundReason(givenSubscription)).toEqual("upgrade") + }) + it(`given refunded transaction for app issue, expect get refund reason`, async () => { + givenSubscription.cancellation_date_ms = exampleDateMsString + givenSubscription.is_upgraded = undefined + givenSubscription.cancellation_reason = "1" + + expect(parsedTransactions.refundReason(givenSubscription)).toEqual("app_issue") + }) + it(`given refunded transaction for other reason, expect get refund reason`, async () => { + givenSubscription.cancellation_date_ms = exampleDateMsString + givenSubscription.is_upgraded = undefined + givenSubscription.cancellation_reason = "0" + + expect(parsedTransactions.refundReason(givenSubscription)).toEqual("other") + }) + it(`given not refunded transaction, expect no refund date`, async () => { + givenSubscription.cancellation_date_ms = undefined + + expect(parsedTransactions.refundReason(givenSubscription)).toBeUndefined() + }) + }) + describe("isRenewOrRestore", () => { + it(`given matching transaction ids, expect false`, async () => { + givenSubscription.original_transaction_id = "123" + givenSubscription.transaction_id = "123" + + expect(parsedTransactions.isRenewOrRestore(givenSubscription)).toEqual(false) + }) + it(`given different transaction ids, expect true`, async () => { + givenSubscription.original_transaction_id = "123" + givenSubscription.transaction_id = "456" + + expect(parsedTransactions.isRenewOrRestore(givenSubscription)).toEqual(true) + }) + }) + describe("parseTransaction", () => { + it(`given transaction, expect all dates to be successfully parsed`, async () => { + givenSubscription.cancellation_date_ms = exampleDateMsString + givenSubscription.expires_date_ms = exampleDateMsString + givenSubscription.original_purchase_date_ms = exampleDateMsString + givenSubscription.purchase_date_ms = exampleDateMsString + + parsedTransactions.addTransaction(givenSubscription) + + const actual = parsedTransactions.getParsedTransactions(givenSubscription)![0] + + expect(actual.cancelledDate).toBeDefined() + expect(actual.expiresDate).toBeDefined() + expect(actual.originalPurchaseDate).toBeDefined() + expect(actual.purchaseDate).toBeDefined() + }) + it(`given in intro offer period, expect true`, async () => { + givenSubscription.is_in_intro_offer_period = "true" + + parsedTransactions.addTransaction(givenSubscription) + + expect(parsedTransactions.getParsedTransactions(givenSubscription)![0].inIntroPeriod).toEqual( + true + ) + }) + it(`given no intro offer period, expect undefined`, async () => { + givenSubscription.is_in_intro_offer_period = "false" + + parsedTransactions.addTransaction(givenSubscription) + + expect(parsedTransactions.getParsedTransactions(givenSubscription)![0].inIntroPeriod).toEqual( + false + ) + }) + it(`given in trial period, expect true`, async () => { + givenSubscription.is_trial_period = "true" + + parsedTransactions.addTransaction(givenSubscription) + + expect(parsedTransactions.getParsedTransactions(givenSubscription)![0].inTrialPeriod).toEqual( + true + ) + }) + it(`given is upgraded, expect true`, async () => { + givenSubscription.is_trial_period = "false" + + parsedTransactions.addTransaction(givenSubscription) + + expect(parsedTransactions.getParsedTransactions(givenSubscription)![0].inTrialPeriod).toEqual( + false + ) + }) + it(`given offer code, expect get code`, async () => { + givenSubscription.offer_code_ref_name = "offer-code-foo" + + parsedTransactions.addTransaction(givenSubscription) + + expect( + parsedTransactions.getParsedTransactions(givenSubscription)![0].offerCodeRefName + ).toEqual("offer-code-foo") + }) + it(`given not offer code, expect undefined`, async () => { + givenSubscription.offer_code_ref_name = undefined + + parsedTransactions.addTransaction(givenSubscription) + + expect( + parsedTransactions.getParsedTransactions(givenSubscription)![0].offerCodeRefName + ).toBeUndefined() + }) + it(`given subscriptionGroupId, expect get in result`, async () => { + givenSubscription.subscription_group_identifier = "foo" + + parsedTransactions.addTransaction(givenSubscription) + + expect( + parsedTransactions.getParsedTransactions(givenSubscription)![0].subscriptionGroupId + ).toEqual("foo") + }) + it(`given transactionId, expect get in result`, async () => { + givenSubscription.transaction_id = "bar" + + parsedTransactions.addTransaction(givenSubscription) + + expect(parsedTransactions.getParsedTransactions(givenSubscription)![0].transactionId).toEqual( + "bar" + ) + }) + it(`given webOrderLineItemId, expect get in result`, async () => { + givenSubscription.web_order_line_item_id = "bar" + + parsedTransactions.addTransaction(givenSubscription) + + expect( + parsedTransactions.getParsedTransactions(givenSubscription)![0].webOrderLineItemId + ).toEqual("bar") + }) + it(`given originalTransactionId, expect get in result`, async () => { + givenSubscription.original_transaction_id = "bar" + + parsedTransactions.addTransaction(givenSubscription) + + expect( + parsedTransactions.getParsedTransactions(givenSubscription)![0].originalTransactionId + ).toEqual("bar") + }) + it(`given transaction, expect parse`, async () => { + expect(parsedTransactions.parseTransaction(givenSubscription)).toMatchInlineSnapshot(` + Object { + "cancelledDate": 2020-11-13T01:29:30.000Z, + "expiresDate": 2020-11-13T01:29:30.000Z, + "inIntroPeriod": false, + "inTrialPeriod": false, + "isFirstSubscriptionPurchase": true, + "isRenewOrRestore": false, + "isUpgradeTransaction": false, + "offerCodeRefName": undefined, + "originalPurchaseDate": 2020-11-13T01:29:30.000Z, + "originalTransactionId": "bar", + "productId": "test_subscription", + "promotionalOfferId": undefined, + "purchaseDate": 2020-11-13T01:29:30.000Z, + "rawTransaction": Object { + "cancellation_date_ms": "1605230970000", + "cancellation_reason": "0", + "expires_date": "2020-11-13 01:12:23 Etc/GMT", + "expires_date_ms": "1605230970000", + "expires_date_pst": "2020-11-12 17:12:23 America/Los_Angeles", + "is_in_intro_offer_period": "false", + "is_trial_period": "false", + "is_upgraded": undefined, + "offer_code_ref_name": undefined, + "original_purchase_date": "2020-11-13 01:09:24 Etc/GMT", + "original_purchase_date_ms": "1605230970000", + "original_purchase_date_pst": "2020-11-12 17:09:24 America/Los_Angeles", + "original_transaction_id": "bar", + "product_id": "test_subscription", + "purchase_date": "2020-11-13 01:09:23 Etc/GMT", + "purchase_date_ms": "1605230970000", + "purchase_date_pst": "2020-11-12 17:09:23 America/Los_Angeles", + "quantity": "1", + "subscription_group_identifier": "foo", + "transaction_id": "bar", + "web_order_line_item_id": "bar", + }, + "refundReason": "other", + "subscriptionGroupId": "foo", + "transactionId": "bar", + "webOrderLineItemId": "bar", + } + `) + }) + }) +}) diff --git a/app/verify_receipt/parse/auto_renewable_subscription/transactions.ts b/app/verify_receipt/parse/auto_renewable_subscription/transactions.ts new file mode 100644 index 0000000..ee15ab9 --- /dev/null +++ b/app/verify_receipt/parse/auto_renewable_subscription/transactions.ts @@ -0,0 +1,105 @@ +import { AppleLatestReceiptInfo } from "../../apple_response/latest_receipt_info" +import { InternalError } from "../error" +import { + AutoRenewableSubscriptionRefundReason, + AutoRenewableSubscriptionTransaction +} from "../result/auto_renewable_subscription" + +/** + * Takes in a response from Apple transaction and produces a parsed transaction. Keeps track of all of them by grouping them together by subscription group. + * + * @internal + */ +export class ParsedTransactions { + + // key is `original_transaction_id` as that's the way to group subscriptions + private transactions: Map = new Map() + // key is `subscription_group_identifier` as we must determine if eligible for intro offer across multiple subscriptions. + // From "Determine Eligibility" section in: https://web.archive.org/web/20200908054522if_/https://developer.apple.com/documentation/storekit/in-app_purchase/subscriptions_and_offers/implementing_introductory_offers_in_your_app + private isEligibleIntroductoryOffer: Map = new Map() + + addTransaction(transaction: AppleLatestReceiptInfo): void { + if (!ParsedTransactions.isAutoRenewableSubscriptionTransaction(transaction)) + throw new InternalError( + "PrerequisiteNotMet", + `Filter out all transactions that are *not* subscription transactions before calling this function.` + ) + + const existingTransactions = this.getParsedTransactions(transaction) || [] + + existingTransactions.push(this.parseTransaction(transaction)) + + this.transactions.set(transaction.original_transaction_id, existingTransactions) + + this.updateIsEligibleIntroductoryOffer(transaction) + } + + iterateParsedTransactions(): IterableIterator { + return this.transactions.values() + } + + getParsedTransactions(transaction: AppleLatestReceiptInfo): AutoRenewableSubscriptionTransaction[] | undefined { + return this.transactions.get(transaction.original_transaction_id) + } + + getIsEligibleIntroductoryOffer(transaction: AppleLatestReceiptInfo | AutoRenewableSubscriptionTransaction): boolean { + const key = (transaction as AppleLatestReceiptInfo).subscription_group_identifier || (transaction as AutoRenewableSubscriptionTransaction).subscriptionGroupId + + return this.isEligibleIntroductoryOffer.get(key) || false + } + + static isAutoRenewableSubscriptionTransaction(transaction: AppleLatestReceiptInfo): boolean { + return transaction.expires_date !== undefined && transaction.expires_date !== null + } + + updateIsEligibleIntroductoryOffer(transaction: AppleLatestReceiptInfo): void { + const ifNotEligible = transaction.is_trial_period === "true" || transaction.is_in_intro_offer_period === "true" + const isEligible = !ifNotEligible + const id = transaction.subscription_group_identifier! + + const existingValue = this.isEligibleIntroductoryOffer.get(id) + if (existingValue !== undefined && existingValue === false) return // we never want to overwrite value that is false already. once it's not eligible, it's never eligible. + + this.isEligibleIntroductoryOffer.set(transaction.subscription_group_identifier!, isEligible) + } + + refundReason(transaction: AppleLatestReceiptInfo): AutoRenewableSubscriptionRefundReason | undefined { + if (!transaction.cancellation_date_ms) return undefined + + if (transaction.is_upgraded) return "upgrade" + else return transaction.cancellation_reason === "1" ? "app_issue" : "other" + } + + isRenewOrRestore(transaction: AppleLatestReceiptInfo): boolean { + return transaction.original_transaction_id !== transaction.transaction_id + } + + parseTransaction( + transaction: AppleLatestReceiptInfo + ): AutoRenewableSubscriptionTransaction { + const isRenewOrRestore = this.isRenewOrRestore(transaction) + + return { + productId: transaction.product_id, + cancelledDate: transaction.cancellation_date_ms + ? new Date(parseInt(transaction.cancellation_date_ms!)) + : undefined, + refundReason: this.refundReason(transaction), + expiresDate: new Date(parseInt(transaction.expires_date_ms!)), + inIntroPeriod: transaction.is_in_intro_offer_period === "true", + inTrialPeriod: transaction.is_trial_period === "true", + isUpgradeTransaction: transaction.is_upgraded === "true", + offerCodeRefName: transaction.offer_code_ref_name, + originalPurchaseDate: new Date(parseInt(transaction.original_purchase_date_ms)), + originalTransactionId: transaction.original_transaction_id, + promotionalOfferId: transaction.promotional_offer_id, + purchaseDate: new Date(parseInt(transaction.purchase_date_ms)), + subscriptionGroupId: transaction.subscription_group_identifier!, + transactionId: transaction.transaction_id, + webOrderLineItemId: transaction.web_order_line_item_id, + isRenewOrRestore, + isFirstSubscriptionPurchase: !isRenewOrRestore, + rawTransaction: transaction + } + } +} \ No newline at end of file diff --git a/app/verify_receipt/parse/error.test.ts b/app/verify_receipt/parse/error.test.ts new file mode 100644 index 0000000..0257b59 --- /dev/null +++ b/app/verify_receipt/parse/error.test.ts @@ -0,0 +1,141 @@ +import { AppleError, handleErrorResponse, InternalError } from "./error" +import * as constants from "../../constants" +import { AppleVerifyReceiptErrorCode } from "../apple_response" + +describe("InternalError", () => { + it(`given message, expect insert bug report link`, async () => { + const actual = new InternalError("name", "added-message") + + expect(actual.message.includes(constants.bugReportLink)).toEqual(true) + }) + it(`given message, expect add message`, async () => { + const actual = new InternalError("name", "added-message") + + expect(actual.message.includes("added-message")).toEqual(true) + }) + it(`given name, expect name equal`, async () => { + const actual = new InternalError("name", "added-message") + + expect(actual.name).toEqual("name") + }) + it(`given no stack, expect set default`, async () => { + const actual = new InternalError("name", "added-message") + + expect(actual.stack).toBeDefined() + }) + it(`given given error stack, expect get error stack`, async () => { + const stack = "fake-stack" + const givenError: Error = { + name: "foo", + message: "", + stack + } + + const actual = InternalError.fromError(givenError) + + expect(actual.stack).toEqual(stack) + }) +}) + +describe("AppleError", () => { + it(`expect includes error code`, async () => { + const actual = new AppleError("message", AppleVerifyReceiptErrorCode.WRONG_SHARED_SECRET) + expect(actual.appleErrorCode).toEqual(AppleVerifyReceiptErrorCode.WRONG_SHARED_SECRET) + expect( + actual.message.includes(AppleVerifyReceiptErrorCode.WRONG_SHARED_SECRET.toString()) + ).toEqual(true) + }) +}) + +describe("handleErrorResponse", () => { + it(`given successful response, expect throw internal error`, async () => { + expect(() => { + handleErrorResponse({ + status: 0 + }) + }).toThrow() + }) + it(`given random number, expect apple error`, async () => { + const code = 2020202 + + const actual = handleErrorResponse({ + status: code + }) as AppleError + + expect(actual.appleErrorCode).toEqual(code) + }) + it(`given NOT_POST, expect internal error`, async () => { + const actual = handleErrorResponse({ + status: AppleVerifyReceiptErrorCode.NOT_POST + }) as InternalError + + expect(actual.name).toEqual("ShouldNotHappen") + }) + it(`given SHOULD_NOT_HAPPEN, expect internal error`, async () => { + const actual = handleErrorResponse({ + status: AppleVerifyReceiptErrorCode.SHOULD_NOT_HAPPEN + }) as InternalError + + expect(actual.name).toEqual("ShouldNotHappen") + }) + it(`given INVALID_RECEIPT_OR_DOWN, expect apple error`, async () => { + const actual = handleErrorResponse({ + status: AppleVerifyReceiptErrorCode.INVALID_RECEIPT_OR_DOWN + }) as AppleError + + expect(actual.message).toMatchInlineSnapshot( + `"The receipt you sent may be malformed. Your code may have modified the receipt or this is a bad request sent by someone possibly malicious. Make sure your apps work and besides that, ignore this. (error code: 21002)"` + ) + expect(actual.appleErrorCode).toEqual(AppleVerifyReceiptErrorCode.INVALID_RECEIPT_OR_DOWN) + }) + it(`given UNAUTHORIZED, expect apple error`, async () => { + const actual = handleErrorResponse({ + status: AppleVerifyReceiptErrorCode.UNAUTHORIZED + }) as AppleError + + expect(actual.message).toMatchInlineSnapshot( + `"Apple said the request was unauthorized. Perhaps you provided the wrong shared secret? (error code: 21003)"` + ) + expect(actual.appleErrorCode).toEqual(AppleVerifyReceiptErrorCode.UNAUTHORIZED) + }) + it(`given WRONG_SHARED_SECRET, expect apple error`, async () => { + const actual = handleErrorResponse({ + status: AppleVerifyReceiptErrorCode.WRONG_SHARED_SECRET + }) as AppleError + + expect(actual.message).toMatchInlineSnapshot( + `"Apple said the shared secret that you provided does not match the shared secret on file for your account. Check it to make sure it's correct. (error code: 21004)"` + ) + expect(actual.appleErrorCode).toEqual(AppleVerifyReceiptErrorCode.WRONG_SHARED_SECRET) + }) + it(`given APPLE_INTERNAL_ERROR, expect apple error`, async () => { + const actual = handleErrorResponse({ + status: AppleVerifyReceiptErrorCode.APPLE_INTERNAL_ERROR + }) as AppleError + + expect(actual.message).toMatchInlineSnapshot( + `"Unknown Apple error code. This might be an internal Apple error code or one that this library simply does not know about yet. (error code: 21009)"` + ) + expect(actual.appleErrorCode).toEqual(AppleVerifyReceiptErrorCode.APPLE_INTERNAL_ERROR) + }) + it(`given SERVICE_DOWN, expect apple error`, async () => { + const actual = handleErrorResponse({ + status: AppleVerifyReceiptErrorCode.SERVICE_DOWN + }) as AppleError + + expect(actual.message).toMatchInlineSnapshot( + `"Sorry! Apple's service seems to be down. Try the request again later. That's all we know. (error code: 21005)"` + ) + expect(actual.appleErrorCode).toEqual(AppleVerifyReceiptErrorCode.SERVICE_DOWN) + }) + it(`given CUSTOMER_NOT_FOUND, expect apple error`, async () => { + const actual = handleErrorResponse({ + status: AppleVerifyReceiptErrorCode.CUSTOMER_NOT_FOUND + }) as AppleError + + expect(actual.message).toMatchInlineSnapshot( + `"Apple could not find the customer. The customer could have been deleted? (error code: 21010)"` + ) + expect(actual.appleErrorCode).toEqual(AppleVerifyReceiptErrorCode.CUSTOMER_NOT_FOUND) + }) +}) diff --git a/app/verify_receipt/parse/error.ts b/app/verify_receipt/parse/error.ts new file mode 100644 index 0000000..f03a3c1 --- /dev/null +++ b/app/verify_receipt/parse/error.ts @@ -0,0 +1,97 @@ +import { AppleVerifyReceiptErrorCode, isAppleErrorCode } from "../apple_response/error" +import { AppleVerifyReceiptErrorResponse } from "../apple_response/error" +import { bugReportLink } from "../../constants" + +/** + * Parses the HTTP response and returns back an Error meant for a developer's purpose. These errors below are meant to be more friendly to help the developer fix the issue. + * + * @internal + */ +export const handleErrorResponse = (response: AppleVerifyReceiptErrorResponse): Error => { + const code = response.status + + if (!isAppleErrorCode(code)) throw new InternalError("ShouldNotHappen", "Function should only be called with statuses that are errors.") + + switch (code) { + case AppleVerifyReceiptErrorCode.NOT_POST: + return new InternalError( + "ShouldNotHappen", + "Error should not happen. Apple expects a correctly formatted HTTP POST which, we perform on your behalf." + ) + case AppleVerifyReceiptErrorCode.SHOULD_NOT_HAPPEN: + return new InternalError( + "ShouldNotHappen", + "Apple's documentation says this response code is no longer being used. Should not happen." + ) + case AppleVerifyReceiptErrorCode.INVALID_RECEIPT_OR_DOWN: + return new AppleError( + "The receipt you sent may be malformed. Your code may have modified the receipt or this is a bad request sent by someone possibly malicious. Make sure your apps work and besides that, ignore this.", + code + ) + case AppleVerifyReceiptErrorCode.UNAUTHORIZED: + return new AppleError( + "Apple said the request was unauthorized. Perhaps you provided the wrong shared secret?", + code + ) + case AppleVerifyReceiptErrorCode.WRONG_SHARED_SECRET: + return new AppleError( + "Apple said the shared secret that you provided does not match the shared secret on file for your account. Check it to make sure it's correct.", + code + ) + case (AppleVerifyReceiptErrorCode.APPLE_INTERNAL_ERROR, + AppleVerifyReceiptErrorCode.SERVICE_DOWN): + return new AppleError( + `Sorry! Apple's service seems to be down. Try the request again later. That's all we know.`, + code + ) + case AppleVerifyReceiptErrorCode.CUSTOMER_NOT_FOUND: + return new AppleError( + "Apple could not find the customer. The customer could have been deleted?", + code + ) + } + + return new AppleError( + "Unknown Apple error code. This might be an internal Apple error code or one that this library simply does not know about yet.", + code + ) +} + +/** + * An error came back from Apple. Apple's server returns back an error code. This class is an Error wrapper for that code to be returned to the developer. + */ +export class AppleError extends Error { + constructor(message: string, public appleErrorCode: AppleVerifyReceiptErrorCode) { + super(`${message} (error code: ${appleErrorCode})`) + } +} + +/** + * An error happened in the library itself. This Error type should not be used by a developer using this library. If this error gets thrown, it means you should open a bug report so we can fix the library. + * + * Goals of this class: + * 1. Prompt the developer to file a bug report. + * 2. Include in this error all of the information from the given error so we can fix the issue. + */ +export class InternalError implements Error { + public name: string + public message: string + public stack: string + + constructor(name: string, message: string, stack?: string) { + this.stack = stack || new Error().stack! + + this.name = name + this.message = `BUG! Please, file a bug report: ${bugReportLink}. (${message})` + } + + static fromError(error: Error): InternalError { + const errorCopy = error + const stack = error.stack! + + // so when we do stringify(), we don't include the stack. + delete errorCopy.stack + + return new InternalError(error.name, JSON.stringify(errorCopy), stack) + } +} diff --git a/app/verify_receipt/parse/purchase/index.test.ts b/app/verify_receipt/parse/purchase/index.test.ts new file mode 100644 index 0000000..5f7e32d --- /dev/null +++ b/app/verify_receipt/parse/purchase/index.test.ts @@ -0,0 +1,14 @@ +import { parsePurchases } from "." + +describe("parsePurchases", () => { + it(`given no in app purchases`, async () => { + expect(parsePurchases(require("../../../../samples/simple.json"))).toEqual([]) + }) + it(`given subscription transactions, expect get parsed subscription responses`, async () => { + const actual = parsePurchases(require("../../../../samples/bought_consumable.json")) + + expect(actual).toHaveLength(1) + expect(actual[0].transactions).toHaveLength(1) + expect(actual[0].productId).toEqual("test_consumable") + }) +}) diff --git a/app/verify_receipt/parse/purchase/index.ts b/app/verify_receipt/parse/purchase/index.ts new file mode 100644 index 0000000..1f0a783 --- /dev/null +++ b/app/verify_receipt/parse/purchase/index.ts @@ -0,0 +1,20 @@ +import { AppleVerifyReceiptSuccessfulResponse } from "../../apple_response/success" +import { ProductPurchases } from "../result" +import { ParsedPurchases } from "./transaction" + +/** + * @internal + */ +export const parsePurchases = ( + response: AppleVerifyReceiptSuccessfulResponse +): ProductPurchases[] => { + if (!response.receipt.in_app) return [] + + const parsedPurchases = new ParsedPurchases() + + response.receipt.in_app + .filter((transaction) => ParsedPurchases.isPurchaseTransaction(transaction)) + .forEach((purchase) => parsedPurchases.addTransaction(purchase)) + + return parsedPurchases.parse() +} diff --git a/app/verify_receipt/parse/purchase/transaction.test.ts b/app/verify_receipt/parse/purchase/transaction.test.ts new file mode 100644 index 0000000..565b6a8 --- /dev/null +++ b/app/verify_receipt/parse/purchase/transaction.test.ts @@ -0,0 +1,131 @@ +import { AppleInAppPurchaseTransaction } from "../../apple_response" +import { ParsedPurchases } from "./transaction" + +let parsedPurchases: ParsedPurchases +let givenPurchase: AppleInAppPurchaseTransaction +let givenPurchase2: AppleInAppPurchaseTransaction + +beforeEach(() => { + parsedPurchases = new ParsedPurchases() + givenPurchase = require("../../../../samples/consumable.json") + givenPurchase2 = require("../../../../samples/consumable2.json") +}) + +describe("ParsedPurchases", () => { + describe("isPurchaseTransaction", () => { + it(`given purchase without expires date, expect true`, async () => { + expect( + ParsedPurchases.isPurchaseTransaction({ + expires_date: undefined + } as AppleInAppPurchaseTransaction) + ).toEqual(true) + }) + it(`given purchase with expires, expect false`, async () => { + expect( + ParsedPurchases.isPurchaseTransaction({ + expires_date: "2020-11-10" + } as AppleInAppPurchaseTransaction) + ).toEqual(false) + }) + }) + + describe("addTransaction", () => { + it(`given transaction that's not a purchase, expect throw`, async () => { + expect(() => { + parsedPurchases.addTransaction({ + expires_date: "2020-11-10" + } as AppleInAppPurchaseTransaction) + }).toThrowErrorMatchingInlineSnapshot( + `"BUG! Please, file a bug report: https://github.com/levibostian/dollabill-apple/issues/new?template=BUG_REPORT.md. (Filter out all transactions that are *not* product purchase transactions before calling this function.)"` + ) + }) + it(`given add transactions from multiple purchases, expect grouped by product`, async () => { + parsedPurchases.addTransaction(givenPurchase) + parsedPurchases.addTransaction(givenPurchase2) + + expect(Array.from(parsedPurchases.transactions.keys())).toHaveLength(2) + expect(parsedPurchases.transactions.get(givenPurchase.product_id)).not.toEqual( + parsedPurchases.transactions.get(givenPurchase2.product_id) + ) + }) + it(`given add multiple transactions for same subscription, expect add transactions`, async () => { + parsedPurchases.addTransaction(givenPurchase) + parsedPurchases.addTransaction(givenPurchase) + + expect(Array.from(parsedPurchases.transactions.keys())).toHaveLength(1) + expect(parsedPurchases.transactions.get(givenPurchase.product_id)).toHaveLength(2) + }) + it(`given purchase, expect parsed transaction`, async () => { + parsedPurchases.addTransaction(givenPurchase) + + expect(parsedPurchases.transactions.get(givenPurchase.product_id)![0]).toMatchInlineSnapshot(` + Object { + "purchaseDate": 2020-11-13T01:41:41.000Z, + "quantity": 1, + "rawTransaction": Object { + "is_trial_period": "false", + "original_purchase_date": "2020-11-13 01:41:41 Etc/GMT", + "original_purchase_date_ms": "1605231701000", + "original_purchase_date_pst": "2020-11-12 17:41:41 America/Los_Angeles", + "original_transaction_id": "1000000741427785", + "product_id": "test_consumable", + "purchase_date": "2020-11-13 01:41:41 Etc/GMT", + "purchase_date_ms": "1605231701000", + "purchase_date_pst": "2020-11-12 17:41:41 America/Los_Angeles", + "quantity": "1", + "transaction_id": "1000000741427785", + }, + "transactionId": "1000000741427785", + } + `) + }) + }) + + describe("parse", () => { + it(`given multiple purchases, expect sort by purchase date`, async () => { + parsedPurchases.addTransaction(givenPurchase) + givenPurchase2.product_id = givenPurchase.product_id // to make these 2 transactions be grouped together + parsedPurchases.addTransaction(givenPurchase2) + + const actual = parsedPurchases.parse() + + expect(actual).toHaveLength(1) + expect(givenPurchase.transaction_id).not.toEqual(givenPurchase2.transaction_id) // to assert these next lines mean something. + expect(actual[0].transactions[0].transactionId).toEqual(givenPurchase.transaction_id) + expect(actual[0].transactions[1].transactionId).toEqual(givenPurchase2.transaction_id) + }) + it(`given purchases, expect result`, async () => { + parsedPurchases.addTransaction(givenPurchase) + + const actual = parsedPurchases.parse() + + expect(actual).toMatchInlineSnapshot(` + Array [ + Object { + "productId": "test_consumable", + "transactions": Array [ + Object { + "purchaseDate": 2020-11-13T01:41:41.000Z, + "quantity": 1, + "rawTransaction": Object { + "is_trial_period": "false", + "original_purchase_date": "2020-11-13 01:41:41 Etc/GMT", + "original_purchase_date_ms": "1605231701000", + "original_purchase_date_pst": "2020-11-12 17:41:41 America/Los_Angeles", + "original_transaction_id": "1000000741427785", + "product_id": "test_consumable", + "purchase_date": "2020-11-13 01:41:41 Etc/GMT", + "purchase_date_ms": "1605231701000", + "purchase_date_pst": "2020-11-12 17:41:41 America/Los_Angeles", + "quantity": "1", + "transaction_id": "1000000741427785", + }, + "transactionId": "1000000741427785", + }, + ], + }, + ] + `) + }) + }) +}) diff --git a/app/verify_receipt/parse/purchase/transaction.ts b/app/verify_receipt/parse/purchase/transaction.ts new file mode 100644 index 0000000..97b3e8f --- /dev/null +++ b/app/verify_receipt/parse/purchase/transaction.ts @@ -0,0 +1,54 @@ +import _ from "../../../underscore" +import { InternalError } from "../error" +import { AppleInAppPurchaseTransaction } from "../../apple_response/receipt" +import { ProductPurchases, ProductPurchaseTransaction } from "../result/purchase" + +/** + * Takes in a response from Apple transaction and produces a parsed transaction. Keeps track of all of them by grouping them together by subscription group. + * + * @internal + */ +export class ParsedPurchases { + public transactions: Map = new Map() + + addTransaction(purchase: AppleInAppPurchaseTransaction): void { + if (!ParsedPurchases.isPurchaseTransaction(purchase)) + throw new InternalError( + "PrerequisiteNotMet", + `Filter out all transactions that are *not* product purchase transactions before calling this function.` + ) + + const existingTransactions = this.transactions.get(purchase.product_id) || [] + + existingTransactions.push(this.parseTransaction(purchase)) + + this.transactions.set(purchase.product_id, existingTransactions) + } + + static isPurchaseTransaction(transaction: AppleInAppPurchaseTransaction): boolean { + return transaction.expires_date === undefined || transaction.expires_date === null + } + + parse(): ProductPurchases[] { + return Array.from(this.transactions.keys()).map((key) => { + const transactions = this.transactions.get(key)! + transactions.sort((first, second) => + _.date.sortNewToOld(first.purchaseDate, second.purchaseDate) + ) + + return { + productId: key, + transactions + } + }) + } + + private parseTransaction(transaction: AppleInAppPurchaseTransaction): ProductPurchaseTransaction { + return { + quantity: parseInt(transaction.quantity!), + transactionId: transaction.transaction_id, + purchaseDate: new Date(parseInt(transaction.purchase_date_ms)), + rawTransaction: transaction + } + } +} diff --git a/app/verify_receipt/parse/result/auto_renewable_subscription.ts b/app/verify_receipt/parse/result/auto_renewable_subscription.ts new file mode 100644 index 0000000..c7ba7ac --- /dev/null +++ b/app/verify_receipt/parse/result/auto_renewable_subscription.ts @@ -0,0 +1,184 @@ +import { AppleLatestReceiptInfo } from "../../apple_response" + +export type AutoRenewableSubscriptionRefundReason = "upgrade" | "app_issue" | "other" + +/** + * Receipt for an auto-renewable subscription. + */ +export type AutoRenewableSubscriptionTransaction = { + productId: string + cancelledDate?: Date + refundReason?: AutoRenewableSubscriptionRefundReason + expiresDate: Date + inIntroPeriod: boolean + inTrialPeriod: boolean + isUpgradeTransaction: boolean + offerCodeRefName?: string + originalPurchaseDate: Date + originalTransactionId: string + promotionalOfferId?: string + purchaseDate: Date + subscriptionGroupId: string + transactionId: string + webOrderLineItemId?: string + /** + * [Learn more](https://developer.apple.com/documentation/appstorereceipts/original_transaction_id) + */ + isRenewOrRestore: boolean + /** + * [Learn more](https://developer.apple.com/documentation/appstorereceipts/original_transaction_id) + */ + isFirstSubscriptionPurchase: boolean + rawTransaction: AppleLatestReceiptInfo +} + +/** + * See {@link AutoRenewableSubscriptionActionableStatus} + */ +export type AutoRenewableSubscriptionIssueString = + | "not_yet_accepting_price_increase" + | "billing_issue" + | "will_voluntary_cancel" + +/** + * Options: + * 1. **not_yet_accepting_price_increase** - You have introduced a price increase. The customer has been notified about it but has not yet accepted it. This means the customer's subscription may be expired if they do not accept it. + * 3. **billing_issue** - the subscription expired because of a billing issue. + * 3. **will_voluntary_cancel** // time to send them UI saying to downgrade to try and keep them + */ +export interface AutoRenewableSubscriptionIssues { + notYetAcceptingPriceIncrease: boolean + billingIssue: boolean + willVoluntaryCancel: boolean +} + +/** + * Options: + * 1. **active** - when no issue is detected, the subscription is active. + * 2. **billing_retry_period** - when the subscription is beyond the grace period, but apple is still trying to restore the purchase. + * 2. **voluntary_cancel** - they cancelled the subscription. You should not give the customer any access to the paid content. + * 3. **grace_period** - There is a billing issue and Apple is attempting to automatically renew the subscription. Grace period means the customer has not cancelled or refunded. You have not yet lost them as a customer. Learn more about Grace Periods in the main docs for this project. + * + * > Tip: You can open the link `https://apps.apple.com/account/billing` to send the user to their payment details page in the App Store to update their payment information. + * + * 4. **involuntary_cancel** - the customer had a billing issue and their subscription is no longer active. Maybe you (1) did not enable the *Grace Period* feature in App Store Connect and the customer has encountered a billing issue or (2) you did enable the *Grace Period* feature but Apple has since given up on attempting to renew the subscription for the customer. You should no longer give the customer access to the paid content. + * 5. **refunded** - the customer contacted Apple support and asked for a refund of their purchase and received the partial or full refund. You should not give the customer any access to the paid content. You may optionally offer another subscription plan to them to switch to a different offer that meets their needs better. + * 6. **other_not_active** - attempt to be future-proof as much as possible. *active* is only if we confirm that there are not cancellations, no expiring, no billing issues, etc. But if Apple drops on us some random new feature or expiration intent that this library doesn't know how to parse yet (or this version you have installed doesn't yet) you will get in the catch-all *other_not_active*. + */ +export type AutoRenewableSubscriptionStatus = "active" | "billing_retry_period" | "grace_period" | "voluntary_cancel" | "upgraded" | "involuntary_cancel" | "refunded" | "other_not_active" + +/** + * Auto-renewable subscription. + * + * > Note: This is auto-renewable *only*. Non aut-renewable subscriptions are up to you to handle how long they last and prompting your customers to purchase again. + */ +export interface AutoRenewableSubscription { + /** + * The `product_id` of the subscription the customer is currently paying for or did pay for. If the user upgrades or downgrades their subscription in a subscription group, this value will change to the always be the most recent `product_id` that they chose. + */ + currentProductId: string + /** + * Used to identify a subscription. + * + * When a customer upgrades/downgrades/downgrades, gets a refund then this subscription will be cancelled and a new subscription will be created. This identifier will be used through all renews and restores of the subscription. Do not rely on the {@link currentProductId} or {@link subscriptionGroup} to identify a subscription as there could be multiple subscriptions with the same product id or subscription group id (Example: one subscription is active while another was cancelled). + */ + originalTransactionId: string + /** + * The subscription group of the subscription. + */ + subscriptionGroup: string + /** + * Indicates if the customer is currently in the free trial period of their subscription. + */ + isInFreeTrialPeriod: boolean + /** + * Determine if the customer is eligible for an introductory offer. New customers are always eligible but existing customers who have not yet redeemed an offer are also eligible. This field determines if the customer has ever redeemed an offer. + * + * [Learn more about how to implement intro offers in your app](https://developer.apple.com/documentation/storekit/in-app_purchase/subscriptions_and_offers/implementing_introductory_offers_in_your_app) + */ + isEligibleIntroductoryOffer: boolean + /** + * The current end date of this subscription. This is a date that is calculated for you by evaluating the expires date, if there was a cancellation stopping the subscription early, or if the subscription is in a grace period and you should extend access. + * + * This is the date that you can use to determine how long your customer should have access to the subscription content. + * + * > Note: This date can increase and decrease at anytime. The subscription can be cancelled immediately (through refund or upgrade/crossgrade), renewed, etc. Do not assume that it only increases. + */ + currentEndDate: Date + /** + * Status of the subscription. + * + * > Note: This value does *not* determine if the customer will renew or not. It is more designed to determine their paid status *at this moment*. See {@link actionableStatuses} to get info about the future of the subscription. + * + * See {@link giveAccessToContent} for a simple true/false if you should give the customer access to the paid content or not. + */ + status: AutoRenewableSubscriptionStatus + /** + * If the customer has redeemed an offer code, this field will be populated with the reference name that you set for the offer. + * + * You determine in App Store Connect is an offer code is available for new, active, or expired users. + */ + usedOfferCodes: string[] + /** + * + */ + usedPromotionalOffers: string[] + /** + * All transactions for this subscription. Ordered by *the expiration date* of the transaction with the latest expire date first in the list. This sort order is recommended by Apple as it's the method to use to determine the subscription periods of this subscription. + * + * See {@link latestExpireDateTransaction} + */ + allTransactions: AutoRenewableSubscriptionTransaction[] + /** + * For convenience, the first entry of {@link allTransactions}. + * + * See {@link allTransactions} + */ + latestExpireDateTransaction: AutoRenewableSubscriptionTransaction + /** + * If the customer is currently planning on automatically renewing their subscription. A customer at anytime may decide to go into their Apple account and setup their subscription to cancel after their current subscription period expires. + */ + willAutoRenew: boolean + /** + * If the customer decides to downgrade their subscription, this is the product id that they will change to when the current subscription period is over. + */ + willDowngradeToProductId?: string + /** + * Strings version of {@link actionableStatuses}. + */ + issuesStrings: AutoRenewableSubscriptionIssueString[] + /** + * Various statuses about this subscription beyond if the customer should be given access to the paid content or not (see {@link subscriptionPaidStatus}). These values are meant to be helpful in notifying your customer, if you choose to do so, to reduce losing customers. Maybe the customer is having a billing issue and they need to update their credit card, maybe you have introduced a pricing increase and the customer has not yet accepted it yet. There are many scenarios and it's up to you to decide what to act upon. + * + * See {@link actionableStatusesStrings} for a string version of this that can be stored in a database. + */ + issues: AutoRenewableSubscriptionIssues + /** + * If the customer has accepted the price increase yet. + * + * This value is populated if the customer has been notified of the increase. If it's undefined, the customer has not yet been notified. + */ + priceIncreaseAccepted?: boolean + /** + * The date that the grace period should end. If a user recovers their subscription billing before this date, the user's subscription will go uninterrupted. If they do not, they can still fix the issue after the date but the subscription will go through a recovery because the subscription will be restored. + * + * > Note: This field will not be populated if the customer is not in a grace period for this subscription. + */ + gracePeriodExpireDate?: Date + /** + * Date that the customer cancelled their subscription, if they did. + */ + cancelledDate?: Date + /** + * The date that the current subscription period will end if not renewed or later on restored. If the subscription is cancelled or goes into a grace period, this Date does not change. It only changes if the subscription gets renewed or restored. + * + * See {@link currentEndDate} for a Date that puts into effect if a subscription is cancelled or is in a grace period. + */ + expireDate: Date + /** + * If the user is in billing retry state. This starts after a billing issue has occurred and lasts as long as Apple wishes (documentation currently says "up to 60 days"). + * + * This field *can* be `true` even after the {@link gracePeriodExpireDate} as lapsed. The grace period time is much shorter: 6-16 days depending on the subscription date. + */ + isInBillingRetry: boolean +} diff --git a/app/verify_receipt/parse/result/index.ts b/app/verify_receipt/parse/result/index.ts new file mode 100644 index 0000000..37b6c34 --- /dev/null +++ b/app/verify_receipt/parse/result/index.ts @@ -0,0 +1,3 @@ +export * from "./auto_renewable_subscription" +export * from "./parsed_receipt" +export * from "./purchase" diff --git a/app/verify_receipt/parse/result/parsed_receipt.ts b/app/verify_receipt/parse/result/parsed_receipt.ts new file mode 100644 index 0000000..e21bb5e --- /dev/null +++ b/app/verify_receipt/parse/result/parsed_receipt.ts @@ -0,0 +1,54 @@ +import { AppleVerifyReceiptSuccessfulResponse } from "../../apple_response/success" +import { AutoRenewableSubscription } from "./auto_renewable_subscription" +import { ProductPurchases } from "./purchase" + +export interface ParsedReceipt { + /** + * The environment for which the receipt was generated. + * + * This value is important to notice so you can determine what transactions were made against a real credit card. + */ + environment: "Sandbox" | "Production" + /** + * Bundle ID of the app this purchase was made for. + */ + bundleId: string + /** + * Version of the app used to make this purchase. (`CFBundleVersion` in iOS or `CFBundleShortVersion` in macOS). + * + * > Note: In the sandbox environment, this value is always "1.0" + */ + appVersion: string + /** + * All of the auto-renewable subscriptions for the customer. There is an object for each subscription. If the user has upgraded/downgraded a subscription that's within the same subscription group, you will still only see 1 object with that upgrade/downgrade in. This array will contain 1 object for each subscription purchase the customer has made. + * + * Note: How is this determined? Each object in this array represents a unique `original_transaction_id` + */ + autoRenewableSubscriptions: AutoRenewableSubscription[] + /** + * Consumables, non-consumables, and non auto-renewable subscriptions. + * + * * Consumable: A product that is used once, after which it becomes depleted and must be purchased again. Example: Fish food for a fishing app. + * * Non-consumable: A product that is purchased once and does not expire or decrease with use. Example: Race track for a game app. + * * Non auto-renewable subscription: A subscription type where you are responsible for determining when it expires and you prompt the customer to make a new purchase. + * + * This list is a list of all purchased delivered by Apple. It is up to you to unlock content, keep track of that has already been redeemed or expired. Consider this simply a list of receipts you must read through to determine the customer's status. + * + * This list is sorted by *the purchase date* with the most recent purchase being first. + */ + productPurchases: ProductPurchases[] + /** + * The raw JSON response back from Apple. This exists in case this library has not been updated, something new gets released from Apple, and you need to access it. + * + * The Typescript typings below are for your convenience. If Apple updates the response when they come out with a new feature and this library has not been updated to include this change from Apple, know that you can treat this field like `any` and access any property that you wish. This is simply the full, unmodified, response from Apple. + */ + rawResponse: AppleVerifyReceiptSuccessfulResponse + /** + * The latest Base64 encoded app receipt. Only returned for receipts that contain auto-renewable subscriptions. + * + * You can optionally store this value in your database to conduct *status polling*. + * + * > Note: Only returned for receipts that contain auto-renewable subscriptions. + */ + latestReceipt?: string +} diff --git a/app/verify_receipt/parse/result/purchase.ts b/app/verify_receipt/parse/result/purchase.ts new file mode 100644 index 0000000..ce35b36 --- /dev/null +++ b/app/verify_receipt/parse/result/purchase.ts @@ -0,0 +1,28 @@ +import { AppleInAppPurchaseTransaction } from "../../apple_response"; + +export interface ProductPurchaseTransaction { + /** + * The quantity purchased for the transaction. Mostly used for consumables. + */ + quantity: number + /** + * A unique identifier for a purchase. + */ + transactionId: string + /** + * When the purchase was made. + */ + purchaseDate: Date + rawTransaction: AppleInAppPurchaseTransaction +} + +export interface ProductPurchases { + /** + * The product id of the product purchased. + */ + productId: string + /** + * All transactions for the given {@link productId}. + */ + transactions: ProductPurchaseTransaction[] +} diff --git a/app/verify_receipt/parse/success.test.ts b/app/verify_receipt/parse/success.test.ts new file mode 100644 index 0000000..8a502e9 --- /dev/null +++ b/app/verify_receipt/parse/success.test.ts @@ -0,0 +1,10 @@ +import { parseSuccess } from "./success" + +describe("parseSuccess", () => { + it(`given receipt with no purchases, expect result`, async () => { + const actual = parseSuccess(require("../../../samples/simple.json")) + + expect(actual.autoRenewableSubscriptions).toEqual([]) + expect(actual.productPurchases).toEqual([]) + }) +}) \ No newline at end of file diff --git a/app/verify_receipt/parse/success.ts b/app/verify_receipt/parse/success.ts new file mode 100644 index 0000000..91f818a --- /dev/null +++ b/app/verify_receipt/parse/success.ts @@ -0,0 +1,22 @@ +import { AppleVerifyReceiptSuccessfulResponse } from "../apple_response/success" +import { parseSubscriptions } from "./auto_renewable_subscription" +import { parsePurchases } from "./purchase" +import { ParsedReceipt } from "./result/parsed_receipt" + +/** + * @internal + */ +export const parseSuccess = (response: AppleVerifyReceiptSuccessfulResponse): ParsedReceipt => { + const subscriptions = parseSubscriptions(response) + const purchases = parsePurchases(response) + + return { + environment: response.environment, + bundleId: response.receipt.bundle_id, + appVersion: response.receipt.application_version, + autoRenewableSubscriptions: subscriptions, + productPurchases: purchases, + rawResponse: response, + latestReceipt: response.latest_receipt + } +} diff --git a/docs/api/assets/css/main.css b/docs/api/assets/css/main.css index 4366900..72e752d 100644 --- a/docs/api/assets/css/main.css +++ b/docs/api/assets/css/main.css @@ -1,2534 +1 @@ -/*! normalize.css v1.1.3 | MIT License | git.io/normalize */ -article, -aside, -details, -figcaption, -figure, -footer, -header, -hgroup, -main, -nav, -section, -summary { - display: block; -} -audio, -canvas, -video { - display: inline-block; - *display: inline; - *zoom: 1; -} -audio:not([controls]) { - display: none; - height: 0; -} -[hidden] { - display: none; -} -html { - font-size: 100%; - -ms-text-size-adjust: 100%; - -webkit-text-size-adjust: 100%; - font-family: sans-serif; -} -button, -input, -select, -textarea { - font-family: sans-serif; -} -body { - margin: 0; -} -a:focus { - outline: thin dotted; -} -a:active, -a:hover { - outline: 0; -} -h1 { - font-size: 2em; - margin: 0.67em 0; -} -h2 { - font-size: 1.5em; - margin: 0.83em 0; -} -h3 { - font-size: 1.17em; - margin: 1em 0; -} -h4, -.tsd-index-panel h3 { - font-size: 1em; - margin: 1.33em 0; -} -h5 { - font-size: 0.83em; - margin: 1.67em 0; -} -h6 { - font-size: 0.67em; - margin: 2.33em 0; -} -abbr[title] { - border-bottom: 1px dotted; -} -b, -strong { - font-weight: bold; -} -blockquote { - margin: 1em 40px; -} -dfn { - font-style: italic; -} -hr { - -moz-box-sizing: content-box; - box-sizing: content-box; - height: 0; -} -mark { - background: #ff0; - color: #000; -} -p, -pre { - margin: 1em 0; -} -code, -kbd, -pre, -samp { - font-family: monospace, serif; - _font-family: "courier new", monospace; - font-size: 1em; -} -pre { - white-space: pre; - white-space: pre-wrap; - word-wrap: break-word; -} -q { - quotes: none; -} -q:before, -q:after { - content: ""; - content: none; -} -small { - font-size: 80%; -} -sub { - font-size: 75%; - line-height: 0; - position: relative; - vertical-align: baseline; -} -sup { - font-size: 75%; - line-height: 0; - position: relative; - vertical-align: baseline; - top: -0.5em; -} -sub { - bottom: -0.25em; -} -dl, -menu, -ol, -ul { - margin: 1em 0; -} -dd { - margin: 0 0 0 40px; -} -menu, -ol, -ul { - padding: 0 0 0 40px; -} -nav ul, -nav ol { - list-style: none; - list-style-image: none; -} -img { - border: 0; - -ms-interpolation-mode: bicubic; -} -svg:not(:root) { - overflow: hidden; -} -figure, -form { - margin: 0; -} -fieldset { - border: 1px solid silver; - margin: 0 2px; - padding: 0.35em 0.625em 0.75em; -} -legend { - border: 0; - padding: 0; - white-space: normal; - *margin-left: -7px; -} -button, -input, -select, -textarea { - font-size: 100%; - margin: 0; - vertical-align: baseline; - *vertical-align: middle; -} -button, -input { - line-height: normal; -} -button, -select { - text-transform: none; -} -button, -html input[type="button"] { - -webkit-appearance: button; - cursor: pointer; - *overflow: visible; -} -input[type="reset"], -input[type="submit"] { - -webkit-appearance: button; - cursor: pointer; - *overflow: visible; -} -button[disabled], -html input[disabled] { - cursor: default; -} -input[type="checkbox"], -input[type="radio"] { - box-sizing: border-box; - padding: 0; - *height: 13px; - *width: 13px; -} -input[type="search"] { - -webkit-appearance: textfield; - -moz-box-sizing: content-box; - -webkit-box-sizing: content-box; - box-sizing: content-box; -} -input[type="search"]::-webkit-search-cancel-button, -input[type="search"]::-webkit-search-decoration { - -webkit-appearance: none; -} -button::-moz-focus-inner, -input::-moz-focus-inner { - border: 0; - padding: 0; -} -textarea { - overflow: auto; - vertical-align: top; -} -table { - border-collapse: collapse; - border-spacing: 0; -} -.hljs { - display: inline-block; - padding: 0.5em; - background: #fff; - color: #000; -} -.hljs-comment, -.hljs-annotation, -.hljs-template_comment, -.diff .hljs-header, -.hljs-chunk, -.apache .hljs-cbracket { - color: green; -} -.hljs-keyword, -.hljs-id, -.hljs-built_in, -.css .smalltalk .hljs-class, -.hljs-winutils, -.bash .hljs-variable, -.tex .hljs-command, -.hljs-request, -.hljs-status, -.nginx .hljs-title { - color: blue; -} -.xml .hljs-tag { - color: blue; -} -.xml .hljs-tag .hljs-value { - color: blue; -} -.hljs-string, -.hljs-title, -.hljs-parent, -.hljs-tag .hljs-value, -.hljs-rules .hljs-value { - color: #a31515; -} -.ruby .hljs-symbol { - color: #a31515; -} -.ruby .hljs-symbol .hljs-string { - color: #a31515; -} -.hljs-template_tag, -.django .hljs-variable, -.hljs-addition, -.hljs-flow, -.hljs-stream, -.apache .hljs-tag, -.hljs-date, -.tex .hljs-formula, -.coffeescript .hljs-attribute { - color: #a31515; -} -.ruby .hljs-string, -.hljs-decorator, -.hljs-filter .hljs-argument, -.hljs-localvars, -.hljs-array, -.hljs-attr_selector, -.hljs-pseudo, -.hljs-pi, -.hljs-doctype, -.hljs-deletion, -.hljs-envvar, -.hljs-shebang, -.hljs-preprocessor, -.hljs-pragma, -.userType, -.apache .hljs-sqbracket, -.nginx .hljs-built_in, -.tex .hljs-special, -.hljs-prompt { - color: #2b91af; -} -.hljs-phpdoc, -.hljs-javadoc, -.hljs-xmlDocTag { - color: gray; -} -.vhdl .hljs-typename { - font-weight: bold; -} -.vhdl .hljs-string { - color: #666; -} -.vhdl .hljs-literal { - color: #a31515; -} -.vhdl .hljs-attribute { - color: #00b0e8; -} -.xml .hljs-attribute { - color: red; -} -ul.tsd-descriptions > li > :first-child, -.tsd-panel > :first-child, -.col > :first-child, -.col-11 > :first-child, -.col-10 > :first-child, -.col-9 > :first-child, -.col-8 > :first-child, -.col-7 > :first-child, -.col-6 > :first-child, -.col-5 > :first-child, -.col-4 > :first-child, -.col-3 > :first-child, -.col-2 > :first-child, -.col-1 > :first-child, -ul.tsd-descriptions > li > :first-child > :first-child, -.tsd-panel > :first-child > :first-child, -.col > :first-child > :first-child, -.col-11 > :first-child > :first-child, -.col-10 > :first-child > :first-child, -.col-9 > :first-child > :first-child, -.col-8 > :first-child > :first-child, -.col-7 > :first-child > :first-child, -.col-6 > :first-child > :first-child, -.col-5 > :first-child > :first-child, -.col-4 > :first-child > :first-child, -.col-3 > :first-child > :first-child, -.col-2 > :first-child > :first-child, -.col-1 > :first-child > :first-child, -ul.tsd-descriptions > li > :first-child > :first-child > :first-child, -.tsd-panel > :first-child > :first-child > :first-child, -.col > :first-child > :first-child > :first-child, -.col-11 > :first-child > :first-child > :first-child, -.col-10 > :first-child > :first-child > :first-child, -.col-9 > :first-child > :first-child > :first-child, -.col-8 > :first-child > :first-child > :first-child, -.col-7 > :first-child > :first-child > :first-child, -.col-6 > :first-child > :first-child > :first-child, -.col-5 > :first-child > :first-child > :first-child, -.col-4 > :first-child > :first-child > :first-child, -.col-3 > :first-child > :first-child > :first-child, -.col-2 > :first-child > :first-child > :first-child, -.col-1 > :first-child > :first-child > :first-child { - margin-top: 0; -} -ul.tsd-descriptions > li > :last-child, -.tsd-panel > :last-child, -.col > :last-child, -.col-11 > :last-child, -.col-10 > :last-child, -.col-9 > :last-child, -.col-8 > :last-child, -.col-7 > :last-child, -.col-6 > :last-child, -.col-5 > :last-child, -.col-4 > :last-child, -.col-3 > :last-child, -.col-2 > :last-child, -.col-1 > :last-child, -ul.tsd-descriptions > li > :last-child > :last-child, -.tsd-panel > :last-child > :last-child, -.col > :last-child > :last-child, -.col-11 > :last-child > :last-child, -.col-10 > :last-child > :last-child, -.col-9 > :last-child > :last-child, -.col-8 > :last-child > :last-child, -.col-7 > :last-child > :last-child, -.col-6 > :last-child > :last-child, -.col-5 > :last-child > :last-child, -.col-4 > :last-child > :last-child, -.col-3 > :last-child > :last-child, -.col-2 > :last-child > :last-child, -.col-1 > :last-child > :last-child, -ul.tsd-descriptions > li > :last-child > :last-child > :last-child, -.tsd-panel > :last-child > :last-child > :last-child, -.col > :last-child > :last-child > :last-child, -.col-11 > :last-child > :last-child > :last-child, -.col-10 > :last-child > :last-child > :last-child, -.col-9 > :last-child > :last-child > :last-child, -.col-8 > :last-child > :last-child > :last-child, -.col-7 > :last-child > :last-child > :last-child, -.col-6 > :last-child > :last-child > :last-child, -.col-5 > :last-child > :last-child > :last-child, -.col-4 > :last-child > :last-child > :last-child, -.col-3 > :last-child > :last-child > :last-child, -.col-2 > :last-child > :last-child > :last-child, -.col-1 > :last-child > :last-child > :last-child { - margin-bottom: 0; -} -.container { - max-width: 1200px; - margin: 0 auto; - padding: 0 40px; -} -@media (max-width: 640px) { - .container { - padding: 0 20px; - } -} -.container-main { - padding-bottom: 200px; -} -.row { - display: flex; - position: relative; - margin: 0 -10px; -} -.row:after { - visibility: hidden; - display: block; - content: ""; - clear: both; - height: 0; -} -.col, -.col-11, -.col-10, -.col-9, -.col-8, -.col-7, -.col-6, -.col-5, -.col-4, -.col-3, -.col-2, -.col-1 { - box-sizing: border-box; - float: left; - padding: 0 10px; -} -.col-1 { - width: 8.3333333333%; -} -.offset-1 { - margin-left: 8.3333333333%; -} -.col-2 { - width: 16.6666666667%; -} -.offset-2 { - margin-left: 16.6666666667%; -} -.col-3 { - width: 25%; -} -.offset-3 { - margin-left: 25%; -} -.col-4 { - width: 33.3333333333%; -} -.offset-4 { - margin-left: 33.3333333333%; -} -.col-5 { - width: 41.6666666667%; -} -.offset-5 { - margin-left: 41.6666666667%; -} -.col-6 { - width: 50%; -} -.offset-6 { - margin-left: 50%; -} -.col-7 { - width: 58.3333333333%; -} -.offset-7 { - margin-left: 58.3333333333%; -} -.col-8 { - width: 66.6666666667%; -} -.offset-8 { - margin-left: 66.6666666667%; -} -.col-9 { - width: 75%; -} -.offset-9 { - margin-left: 75%; -} -.col-10 { - width: 83.3333333333%; -} -.offset-10 { - margin-left: 83.3333333333%; -} -.col-11 { - width: 91.6666666667%; -} -.offset-11 { - margin-left: 91.6666666667%; -} -.tsd-kind-icon { - display: block; - position: relative; - padding-left: 20px; - text-indent: -20px; -} -.tsd-kind-icon:before { - content: ""; - display: inline-block; - vertical-align: middle; - width: 17px; - height: 17px; - margin: 0 3px 2px 0; - background-image: url(); -} -@media (-webkit-min-device-pixel-ratio: 1.5), (min-resolution: 144dpi) { - .tsd-kind-icon:before { - background-image: url(); - background-size: 238px 204px; - } -} -.tsd-signature.tsd-kind-icon:before { - background-position: 0 -153px; -} -.tsd-kind-object-literal > .tsd-kind-icon:before { - background-position: 0px -17px; -} -.tsd-kind-object-literal.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -17px; -} -.tsd-kind-object-literal.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -17px; -} -.tsd-kind-class > .tsd-kind-icon:before { - background-position: 0px -34px; -} -.tsd-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -34px; -} -.tsd-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -34px; -} -.tsd-kind-class.tsd-has-type-parameter > .tsd-kind-icon:before { - background-position: 0px -51px; -} -.tsd-kind-class.tsd-has-type-parameter.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -51px; -} -.tsd-kind-class.tsd-has-type-parameter.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -51px; -} -.tsd-kind-interface > .tsd-kind-icon:before { - background-position: 0px -68px; -} -.tsd-kind-interface.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -68px; -} -.tsd-kind-interface.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -68px; -} -.tsd-kind-interface.tsd-has-type-parameter > .tsd-kind-icon:before { - background-position: 0px -85px; -} -.tsd-kind-interface.tsd-has-type-parameter.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -85px; -} -.tsd-kind-interface.tsd-has-type-parameter.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -85px; -} -.tsd-kind-namespace > .tsd-kind-icon:before { - background-position: 0px -102px; -} -.tsd-kind-namespace.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -102px; -} -.tsd-kind-namespace.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -102px; -} -.tsd-kind-module > .tsd-kind-icon:before { - background-position: 0px -102px; -} -.tsd-kind-module.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -102px; -} -.tsd-kind-module.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -102px; -} -.tsd-kind-enum > .tsd-kind-icon:before { - background-position: 0px -119px; -} -.tsd-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -119px; -} -.tsd-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -119px; -} -.tsd-kind-enum-member > .tsd-kind-icon:before { - background-position: 0px -136px; -} -.tsd-kind-enum-member.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -136px; -} -.tsd-kind-enum-member.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -136px; -} -.tsd-kind-signature > .tsd-kind-icon:before { - background-position: 0px -153px; -} -.tsd-kind-signature.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -153px; -} -.tsd-kind-signature.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -153px; -} -.tsd-kind-type-alias > .tsd-kind-icon:before { - background-position: 0px -170px; -} -.tsd-kind-type-alias.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -170px; -} -.tsd-kind-type-alias.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -170px; -} -.tsd-kind-type-alias.tsd-has-type-parameter > .tsd-kind-icon:before { - background-position: 0px -187px; -} -.tsd-kind-type-alias.tsd-has-type-parameter.tsd-is-protected > .tsd-kind-icon:before { - background-position: -17px -187px; -} -.tsd-kind-type-alias.tsd-has-type-parameter.tsd-is-private > .tsd-kind-icon:before { - background-position: -34px -187px; -} -.tsd-kind-variable > .tsd-kind-icon:before { - background-position: -136px -0px; -} -.tsd-kind-variable.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -0px; -} -.tsd-kind-variable.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -0px; -} -.tsd-kind-variable.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -0px; -} -.tsd-kind-variable.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -0px; -} -.tsd-kind-variable.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -0px; -} -.tsd-kind-variable.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -102px -0px; -} -.tsd-kind-variable.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -0px; -} -.tsd-kind-variable.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -0px; -} -.tsd-kind-variable.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -0px; -} -.tsd-kind-variable.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -0px; -} -.tsd-kind-variable.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -0px; -} -.tsd-kind-variable.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -0px; -} -.tsd-kind-property > .tsd-kind-icon:before { - background-position: -136px -0px; -} -.tsd-kind-property.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -0px; -} -.tsd-kind-property.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -0px; -} -.tsd-kind-property.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -0px; -} -.tsd-kind-property.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -0px; -} -.tsd-kind-property.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -0px; -} -.tsd-kind-property.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -102px -0px; -} -.tsd-kind-property.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -0px; -} -.tsd-kind-property.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -0px; -} -.tsd-kind-property.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -0px; -} -.tsd-kind-property.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -0px; -} -.tsd-kind-property.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -0px; -} -.tsd-kind-property.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -0px; -} -.tsd-kind-get-signature > .tsd-kind-icon:before { - background-position: -136px -17px; -} -.tsd-kind-get-signature.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -17px; -} -.tsd-kind-get-signature.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -17px; -} -.tsd-kind-get-signature.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -17px; -} -.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -17px; -} -.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -17px; -} -.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -17px; -} -.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -17px; -} -.tsd-kind-get-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -17px; -} -.tsd-kind-get-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -17px; -} -.tsd-kind-get-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -17px; -} -.tsd-kind-get-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -17px; -} -.tsd-kind-get-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -17px; -} -.tsd-kind-set-signature > .tsd-kind-icon:before { - background-position: -136px -34px; -} -.tsd-kind-set-signature.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -34px; -} -.tsd-kind-set-signature.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -34px; -} -.tsd-kind-set-signature.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -34px; -} -.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -34px; -} -.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -34px; -} -.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -34px; -} -.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -34px; -} -.tsd-kind-set-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -34px; -} -.tsd-kind-set-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -34px; -} -.tsd-kind-set-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -34px; -} -.tsd-kind-set-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -34px; -} -.tsd-kind-set-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -34px; -} -.tsd-kind-accessor > .tsd-kind-icon:before { - background-position: -136px -51px; -} -.tsd-kind-accessor.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -51px; -} -.tsd-kind-accessor.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -51px; -} -.tsd-kind-accessor.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -51px; -} -.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -51px; -} -.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -51px; -} -.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -102px -51px; -} -.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -51px; -} -.tsd-kind-accessor.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -51px; -} -.tsd-kind-accessor.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -51px; -} -.tsd-kind-accessor.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -51px; -} -.tsd-kind-accessor.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -51px; -} -.tsd-kind-accessor.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -51px; -} -.tsd-kind-function > .tsd-kind-icon:before { - background-position: -136px -68px; -} -.tsd-kind-function.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -68px; -} -.tsd-kind-function.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -68px; -} -.tsd-kind-function.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -68px; -} -.tsd-kind-function.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -68px; -} -.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -68px; -} -.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -102px -68px; -} -.tsd-kind-function.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -68px; -} -.tsd-kind-function.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -68px; -} -.tsd-kind-function.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -68px; -} -.tsd-kind-function.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -68px; -} -.tsd-kind-function.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -68px; -} -.tsd-kind-function.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -68px; -} -.tsd-kind-method > .tsd-kind-icon:before { - background-position: -136px -68px; -} -.tsd-kind-method.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -68px; -} -.tsd-kind-method.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -68px; -} -.tsd-kind-method.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -68px; -} -.tsd-kind-method.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -68px; -} -.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -68px; -} -.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -102px -68px; -} -.tsd-kind-method.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -68px; -} -.tsd-kind-method.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -68px; -} -.tsd-kind-method.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -68px; -} -.tsd-kind-method.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -68px; -} -.tsd-kind-method.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -68px; -} -.tsd-kind-method.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -68px; -} -.tsd-kind-call-signature > .tsd-kind-icon:before { - background-position: -136px -68px; -} -.tsd-kind-call-signature.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -68px; -} -.tsd-kind-call-signature.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -68px; -} -.tsd-kind-call-signature.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -68px; -} -.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -68px; -} -.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -68px; -} -.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -68px; -} -.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -68px; -} -.tsd-kind-call-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -68px; -} -.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -68px; -} -.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -68px; -} -.tsd-kind-call-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -68px; -} -.tsd-kind-call-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -68px; -} -.tsd-kind-function.tsd-has-type-parameter > .tsd-kind-icon:before { - background-position: -136px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -68px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected - > .tsd-kind-icon:before { - background-position: -85px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-private - > .tsd-kind-icon:before { - background-position: -119px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-protected - > .tsd-kind-icon:before { - background-position: -187px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-private - > .tsd-kind-icon:before { - background-position: -119px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -85px; -} -.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-interface.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -221px -85px; -} -.tsd-kind-method.tsd-has-type-parameter > .tsd-kind-icon:before { - background-position: -136px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -68px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected - > .tsd-kind-icon:before { - background-position: -85px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-private - > .tsd-kind-icon:before { - background-position: -119px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-protected - > .tsd-kind-icon:before { - background-position: -187px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-private - > .tsd-kind-icon:before { - background-position: -119px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -85px; -} -.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-interface.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -221px -85px; -} -.tsd-kind-constructor > .tsd-kind-icon:before { - background-position: -136px -102px; -} -.tsd-kind-constructor.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -102px; -} -.tsd-kind-constructor.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -102px; -} -.tsd-kind-constructor.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -102px; -} -.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -102px; -} -.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -102px; -} -.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -102px; -} -.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -102px; -} -.tsd-kind-constructor.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -102px; -} -.tsd-kind-constructor.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -102px; -} -.tsd-kind-constructor.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -102px; -} -.tsd-kind-constructor.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -102px; -} -.tsd-kind-constructor.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -102px; -} -.tsd-kind-constructor-signature > .tsd-kind-icon:before { - background-position: -136px -102px; -} -.tsd-kind-constructor-signature.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -102px; -} -.tsd-kind-constructor-signature.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -102px; -} -.tsd-kind-constructor-signature.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -102px; -} -.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -102px; -} -.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -102px; -} -.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -102px; -} -.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -102px; -} -.tsd-kind-constructor-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -102px; -} -.tsd-kind-constructor-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -102px; -} -.tsd-kind-constructor-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -102px; -} -.tsd-kind-constructor-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -102px; -} -.tsd-kind-constructor-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -102px; -} -.tsd-kind-index-signature > .tsd-kind-icon:before { - background-position: -136px -119px; -} -.tsd-kind-index-signature.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -119px; -} -.tsd-kind-index-signature.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -119px; -} -.tsd-kind-index-signature.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -119px; -} -.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -119px; -} -.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -119px; -} -.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -119px; -} -.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -119px; -} -.tsd-kind-index-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -119px; -} -.tsd-kind-index-signature.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -119px; -} -.tsd-kind-index-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -119px; -} -.tsd-kind-index-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -119px; -} -.tsd-kind-index-signature.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -119px; -} -.tsd-kind-event > .tsd-kind-icon:before { - background-position: -136px -136px; -} -.tsd-kind-event.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -136px; -} -.tsd-kind-event.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -136px; -} -.tsd-kind-event.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -136px; -} -.tsd-kind-event.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -136px; -} -.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -136px; -} -.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -102px -136px; -} -.tsd-kind-event.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -136px; -} -.tsd-kind-event.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -136px; -} -.tsd-kind-event.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -136px; -} -.tsd-kind-event.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -136px; -} -.tsd-kind-event.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -136px; -} -.tsd-kind-event.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -136px; -} -.tsd-is-static > .tsd-kind-icon:before { - background-position: -136px -153px; -} -.tsd-is-static.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -153px; -} -.tsd-is-static.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -153px; -} -.tsd-is-static.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -153px; -} -.tsd-is-static.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -153px; -} -.tsd-is-static.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -153px; -} -.tsd-is-static.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -102px -153px; -} -.tsd-is-static.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -153px; -} -.tsd-is-static.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -153px; -} -.tsd-is-static.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -153px; -} -.tsd-is-static.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -153px; -} -.tsd-is-static.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -153px; -} -.tsd-is-static.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -153px; -} -.tsd-is-static.tsd-kind-function > .tsd-kind-icon:before { - background-position: -136px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -170px; -} -.tsd-is-static.tsd-kind-function.tsd-parent-kind-interface.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -221px -170px; -} -.tsd-is-static.tsd-kind-method > .tsd-kind-icon:before { - background-position: -136px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -170px; -} -.tsd-is-static.tsd-kind-method.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -170px; -} -.tsd-is-static.tsd-kind-call-signature > .tsd-kind-icon:before { - background-position: -136px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -68px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected - > .tsd-kind-icon:before { - background-position: -85px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-private - > .tsd-kind-icon:before { - background-position: -119px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-protected - > .tsd-kind-icon:before { - background-position: -187px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -170px; -} -.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-interface.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -221px -170px; -} -.tsd-is-static.tsd-kind-event > .tsd-kind-icon:before { - background-position: -136px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-is-protected > .tsd-kind-icon:before { - background-position: -153px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-parent-kind-class > .tsd-kind-icon:before { - background-position: -51px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -68px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected > .tsd-kind-icon:before { - background-position: -85px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited - > .tsd-kind-icon:before { - background-position: -102px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-parent-kind-enum > .tsd-kind-icon:before { - background-position: -170px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-parent-kind-enum.tsd-is-protected > .tsd-kind-icon:before { - background-position: -187px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-parent-kind-enum.tsd-is-private > .tsd-kind-icon:before { - background-position: -119px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-parent-kind-interface > .tsd-kind-icon:before { - background-position: -204px -187px; -} -.tsd-is-static.tsd-kind-event.tsd-parent-kind-interface.tsd-is-inherited > .tsd-kind-icon:before { - background-position: -221px -187px; -} -@keyframes fade-in { - from { - opacity: 0; - } - to { - opacity: 1; - } -} -@keyframes fade-out { - from { - opacity: 1; - visibility: visible; - } - to { - opacity: 0; - } -} -@keyframes fade-in-delayed { - 0% { - opacity: 0; - } - 33% { - opacity: 0; - } - 100% { - opacity: 1; - } -} -@keyframes fade-out-delayed { - 0% { - opacity: 1; - visibility: visible; - } - 66% { - opacity: 0; - } - 100% { - opacity: 0; - } -} -@keyframes shift-to-left { - from { - transform: translate(0, 0); - } - to { - transform: translate(-25%, 0); - } -} -@keyframes unshift-to-left { - from { - transform: translate(-25%, 0); - } - to { - transform: translate(0, 0); - } -} -@keyframes pop-in-from-right { - from { - transform: translate(100%, 0); - } - to { - transform: translate(0, 0); - } -} -@keyframes pop-out-to-right { - from { - transform: translate(0, 0); - visibility: visible; - } - to { - transform: translate(100%, 0); - } -} -body { - background: #fdfdfd; - font-family: "Segoe UI", sans-serif; - font-size: 16px; - color: #222; -} -a { - color: #4da6ff; - text-decoration: none; -} -a:hover { - text-decoration: underline; -} -code, -pre { - font-family: Menlo, Monaco, Consolas, "Courier New", monospace; - padding: 0.2em; - margin: 0; - font-size: 14px; - background-color: rgba(0, 0, 0, 0.04); -} -pre { - padding: 10px; -} -pre code { - padding: 0; - font-size: 100%; - background-color: transparent; -} -.tsd-typography { - line-height: 1.333em; -} -.tsd-typography ul { - list-style: square; - padding: 0 0 0 20px; - margin: 0; -} -.tsd-typography h4, -.tsd-typography .tsd-index-panel h3, -.tsd-index-panel .tsd-typography h3, -.tsd-typography h5, -.tsd-typography h6 { - font-size: 1em; - margin: 0; -} -.tsd-typography h5, -.tsd-typography h6 { - font-weight: normal; -} -.tsd-typography p, -.tsd-typography ul, -.tsd-typography ol { - margin: 1em 0; -} -@media (min-width: 901px) and (max-width: 1024px) { - html.default .col-content { - width: 72%; - } - html.default .col-menu { - width: 28%; - } - html.default .tsd-navigation { - padding-left: 10px; - } -} -@media (max-width: 900px) { - html.default .col-content { - float: none; - width: 100%; - } - html.default .col-menu { - position: fixed !important; - overflow: auto; - -webkit-overflow-scrolling: touch; - z-index: 1024; - top: 0 !important; - bottom: 0 !important; - left: auto !important; - right: 0 !important; - width: 100%; - padding: 20px 20px 0 0; - max-width: 450px; - visibility: hidden; - background-color: #fff; - transform: translate(100%, 0); - } - html.default .col-menu > *:last-child { - padding-bottom: 20px; - } - html.default .overlay { - content: ""; - display: block; - position: fixed; - z-index: 1023; - top: 0; - left: 0; - right: 0; - bottom: 0; - background-color: rgba(0, 0, 0, 0.75); - visibility: hidden; - } - html.default.to-has-menu .overlay { - animation: fade-in 0.4s; - } - html.default.to-has-menu header, - html.default.to-has-menu footer, - html.default.to-has-menu .col-content { - animation: shift-to-left 0.4s; - } - html.default.to-has-menu .col-menu { - animation: pop-in-from-right 0.4s; - } - html.default.from-has-menu .overlay { - animation: fade-out 0.4s; - } - html.default.from-has-menu header, - html.default.from-has-menu footer, - html.default.from-has-menu .col-content { - animation: unshift-to-left 0.4s; - } - html.default.from-has-menu .col-menu { - animation: pop-out-to-right 0.4s; - } - html.default.has-menu body { - overflow: hidden; - } - html.default.has-menu .overlay { - visibility: visible; - } - html.default.has-menu header, - html.default.has-menu footer, - html.default.has-menu .col-content { - transform: translate(-25%, 0); - } - html.default.has-menu .col-menu { - visibility: visible; - transform: translate(0, 0); - } -} -.tsd-page-title { - padding: 70px 0 20px 0; - margin: 0 0 40px 0; - background: #fff; - box-shadow: 0 0 5px rgba(0, 0, 0, 0.35); -} -.tsd-page-title h1 { - margin: 0; -} -.tsd-breadcrumb { - margin: 0; - padding: 0; - color: #707070; -} -.tsd-breadcrumb a { - color: #707070; - text-decoration: none; -} -.tsd-breadcrumb a:hover { - text-decoration: underline; -} -.tsd-breadcrumb li { - display: inline; -} -.tsd-breadcrumb li:after { - content: " / "; -} -html.minimal .container { - margin: 0; -} -html.minimal .container-main { - padding-top: 50px; - padding-bottom: 0; -} -html.minimal .content-wrap { - padding-left: 300px; -} -html.minimal .tsd-navigation { - position: fixed !important; - overflow: auto; - -webkit-overflow-scrolling: touch; - box-sizing: border-box; - z-index: 1; - left: 0; - top: 40px; - bottom: 0; - width: 300px; - padding: 20px; - margin: 0; -} -html.minimal .tsd-member .tsd-member { - margin-left: 0; -} -html.minimal .tsd-page-toolbar { - position: fixed; - z-index: 2; -} -html.minimal #tsd-filter .tsd-filter-group { - right: 0; - transform: none; -} -html.minimal footer { - background-color: transparent; -} -html.minimal footer .container { - padding: 0; -} -html.minimal .tsd-generator { - padding: 0; -} -@media (max-width: 900px) { - html.minimal .tsd-navigation { - display: none; - } - html.minimal .content-wrap { - padding-left: 0; - } -} -dl.tsd-comment-tags { - overflow: hidden; -} -dl.tsd-comment-tags dt { - float: left; - padding: 1px 5px; - margin: 0 10px 0 0; - border-radius: 4px; - border: 1px solid #707070; - color: #707070; - font-size: 0.8em; - font-weight: normal; -} -dl.tsd-comment-tags dd { - margin: 0 0 10px 0; -} -dl.tsd-comment-tags dd:before, -dl.tsd-comment-tags dd:after { - display: table; - content: " "; -} -dl.tsd-comment-tags dd pre, -dl.tsd-comment-tags dd:after { - clear: both; -} -dl.tsd-comment-tags p { - margin: 0; -} -.tsd-panel.tsd-comment .lead { - font-size: 1.1em; - line-height: 1.333em; - margin-bottom: 2em; -} -.tsd-panel.tsd-comment .lead:last-child { - margin-bottom: 0; -} -.toggle-protected .tsd-is-private { - display: none; -} -.toggle-public .tsd-is-private, -.toggle-public .tsd-is-protected, -.toggle-public .tsd-is-private-protected { - display: none; -} -.toggle-inherited .tsd-is-inherited { - display: none; -} -.toggle-only-exported .tsd-is-not-exported { - display: none; -} -.toggle-externals .tsd-is-external { - display: none; -} -#tsd-filter { - position: relative; - display: inline-block; - height: 40px; - vertical-align: bottom; -} -.no-filter #tsd-filter { - display: none; -} -#tsd-filter .tsd-filter-group { - display: inline-block; - height: 40px; - vertical-align: bottom; - white-space: nowrap; -} -#tsd-filter input { - display: none; -} -@media (max-width: 900px) { - #tsd-filter .tsd-filter-group { - display: block; - position: absolute; - top: 40px; - right: 20px; - height: auto; - background-color: #fff; - visibility: hidden; - transform: translate(50%, 0); - box-shadow: 0 0 4px rgba(0, 0, 0, 0.25); - } - .has-options #tsd-filter .tsd-filter-group { - visibility: visible; - } - .to-has-options #tsd-filter .tsd-filter-group { - animation: fade-in 0.2s; - } - .from-has-options #tsd-filter .tsd-filter-group { - animation: fade-out 0.2s; - } - #tsd-filter label, - #tsd-filter .tsd-select { - display: block; - padding-right: 20px; - } -} -footer { - border-top: 1px solid #eee; - background-color: #fff; -} -footer.with-border-bottom { - border-bottom: 1px solid #eee; -} -footer .tsd-legend-group { - font-size: 0; -} -footer .tsd-legend { - display: inline-block; - width: 25%; - padding: 0; - font-size: 16px; - list-style: none; - line-height: 1.333em; - vertical-align: top; -} -@media (max-width: 900px) { - footer .tsd-legend { - width: 50%; - } -} -.tsd-hierarchy { - list-style: square; - padding: 0 0 0 20px; - margin: 0; -} -.tsd-hierarchy .target { - font-weight: bold; -} -.tsd-index-panel .tsd-index-content { - margin-bottom: -30px !important; -} -.tsd-index-panel .tsd-index-section { - margin-bottom: 30px !important; -} -.tsd-index-panel h3 { - margin: 0 -20px 10px -20px; - padding: 0 20px 10px 20px; - border-bottom: 1px solid #eee; -} -.tsd-index-panel ul.tsd-index-list { - -webkit-column-count: 3; - -moz-column-count: 3; - -ms-column-count: 3; - -o-column-count: 3; - column-count: 3; - -webkit-column-gap: 20px; - -moz-column-gap: 20px; - -ms-column-gap: 20px; - -o-column-gap: 20px; - column-gap: 20px; - padding: 0; - list-style: none; - line-height: 1.333em; -} -@media (max-width: 900px) { - .tsd-index-panel ul.tsd-index-list { - -webkit-column-count: 1; - -moz-column-count: 1; - -ms-column-count: 1; - -o-column-count: 1; - column-count: 1; - } -} -@media (min-width: 901px) and (max-width: 1024px) { - .tsd-index-panel ul.tsd-index-list { - -webkit-column-count: 2; - -moz-column-count: 2; - -ms-column-count: 2; - -o-column-count: 2; - column-count: 2; - } -} -.tsd-index-panel ul.tsd-index-list li { - -webkit-page-break-inside: avoid; - -moz-page-break-inside: avoid; - -ms-page-break-inside: avoid; - -o-page-break-inside: avoid; - page-break-inside: avoid; -} -.tsd-index-panel a, -.tsd-index-panel .tsd-parent-kind-module a { - color: #9600ff; -} -.tsd-index-panel .tsd-parent-kind-interface a { - color: #647f1b; -} -.tsd-index-panel .tsd-parent-kind-enum a { - color: #937210; -} -.tsd-index-panel .tsd-parent-kind-class a { - color: #0672de; -} -.tsd-index-panel .tsd-kind-module a { - color: #9600ff; -} -.tsd-index-panel .tsd-kind-interface a { - color: #647f1b; -} -.tsd-index-panel .tsd-kind-enum a { - color: #937210; -} -.tsd-index-panel .tsd-kind-class a { - color: #0672de; -} -.tsd-index-panel .tsd-is-private a { - color: #707070; -} -.tsd-flag { - display: inline-block; - padding: 1px 5px; - border-radius: 4px; - color: #fff; - background-color: #707070; - text-indent: 0; - font-size: 14px; - font-weight: normal; -} -.tsd-anchor { - position: absolute; - top: -100px; -} -.tsd-member { - position: relative; -} -.tsd-member .tsd-anchor + h3 { - margin-top: 0; - margin-bottom: 0; - border-bottom: none; -} -.tsd-navigation { - margin: 0 0 0 40px; -} -.tsd-navigation a { - display: block; - padding-top: 2px; - padding-bottom: 2px; - border-left: 2px solid transparent; - color: #222; - text-decoration: none; - transition: border-left-color 0.1s; -} -.tsd-navigation a:hover { - text-decoration: underline; -} -.tsd-navigation ul { - margin: 0; - padding: 0; - list-style: none; -} -.tsd-navigation li { - padding: 0; -} -.tsd-navigation.primary { - padding-bottom: 40px; -} -.tsd-navigation.primary a { - display: block; - padding-top: 6px; - padding-bottom: 6px; -} -.tsd-navigation.primary ul li a { - padding-left: 5px; -} -.tsd-navigation.primary ul li li a { - padding-left: 25px; -} -.tsd-navigation.primary ul li li li a { - padding-left: 45px; -} -.tsd-navigation.primary ul li li li li a { - padding-left: 65px; -} -.tsd-navigation.primary ul li li li li li a { - padding-left: 85px; -} -.tsd-navigation.primary ul li li li li li li a { - padding-left: 105px; -} -.tsd-navigation.primary > ul { - border-bottom: 1px solid #eee; -} -.tsd-navigation.primary li { - border-top: 1px solid #eee; -} -.tsd-navigation.primary li.current > a { - font-weight: bold; -} -.tsd-navigation.primary li.label span { - display: block; - padding: 20px 0 6px 5px; - color: #707070; -} -.tsd-navigation.primary li.globals + li > span, -.tsd-navigation.primary li.globals + li > a { - padding-top: 20px; -} -.tsd-navigation.secondary { - max-height: calc(100vh - 1rem - 40px); - overflow: auto; - position: -webkit-sticky; - position: sticky; - top: calc(0.5rem + 40px); - transition: 0.3s; -} -.tsd-navigation.secondary.tsd-navigation--toolbar-hide { - max-height: calc(100vh - 1rem); - top: 0.5rem; -} -.tsd-navigation.secondary ul { - transition: opacity 0.2s; -} -.tsd-navigation.secondary ul li a { - padding-left: 25px; -} -.tsd-navigation.secondary ul li li a { - padding-left: 45px; -} -.tsd-navigation.secondary ul li li li a { - padding-left: 65px; -} -.tsd-navigation.secondary ul li li li li a { - padding-left: 85px; -} -.tsd-navigation.secondary ul li li li li li a { - padding-left: 105px; -} -.tsd-navigation.secondary ul li li li li li li a { - padding-left: 125px; -} -.tsd-navigation.secondary ul.current a { - border-left-color: #eee; -} -.tsd-navigation.secondary li.focus > a, -.tsd-navigation.secondary ul.current li.focus > a { - border-left-color: #000; -} -.tsd-navigation.secondary li.current { - margin-top: 20px; - margin-bottom: 20px; - border-left-color: #eee; -} -.tsd-navigation.secondary li.current > a { - font-weight: bold; -} -@media (min-width: 901px) { - .menu-sticky-wrap { - position: static; - } -} -.tsd-panel { - margin: 20px 0; - padding: 20px; - background-color: #fff; - box-shadow: 0 0 4px rgba(0, 0, 0, 0.25); -} -.tsd-panel:empty { - display: none; -} -.tsd-panel > h1, -.tsd-panel > h2, -.tsd-panel > h3 { - margin: 1.5em -20px 10px -20px; - padding: 0 20px 10px 20px; - border-bottom: 1px solid #eee; -} -.tsd-panel > h1.tsd-before-signature, -.tsd-panel > h2.tsd-before-signature, -.tsd-panel > h3.tsd-before-signature { - margin-bottom: 0; - border-bottom: 0; -} -.tsd-panel table { - display: block; - width: 100%; - overflow: auto; - margin-top: 10px; - word-break: normal; - word-break: keep-all; -} -.tsd-panel table th { - font-weight: bold; -} -.tsd-panel table th, -.tsd-panel table td { - padding: 6px 13px; - border: 1px solid #ddd; -} -.tsd-panel table tr { - background-color: #fff; - border-top: 1px solid #ccc; -} -.tsd-panel table tr:nth-child(2n) { - background-color: #f8f8f8; -} -.tsd-panel-group { - margin: 60px 0; -} -.tsd-panel-group > h1, -.tsd-panel-group > h2, -.tsd-panel-group > h3 { - padding-left: 20px; - padding-right: 20px; -} -#tsd-search { - transition: background-color 0.2s; -} -#tsd-search .title { - position: relative; - z-index: 2; -} -#tsd-search .field { - position: absolute; - left: 0; - top: 0; - right: 40px; - height: 40px; -} -#tsd-search .field input { - box-sizing: border-box; - position: relative; - top: -50px; - z-index: 1; - width: 100%; - padding: 0 10px; - opacity: 0; - outline: 0; - border: 0; - background: transparent; - color: #222; -} -#tsd-search .field label { - position: absolute; - overflow: hidden; - right: -40px; -} -#tsd-search .field input, -#tsd-search .title { - transition: opacity 0.2s; -} -#tsd-search .results { - position: absolute; - visibility: hidden; - top: 40px; - width: 100%; - margin: 0; - padding: 0; - list-style: none; - box-shadow: 0 0 4px rgba(0, 0, 0, 0.25); -} -#tsd-search .results li { - padding: 0 10px; - background-color: #fdfdfd; -} -#tsd-search .results li:nth-child(even) { - background-color: #fff; -} -#tsd-search .results li.state { - display: none; -} -#tsd-search .results li.current, -#tsd-search .results li:hover { - background-color: #eee; -} -#tsd-search .results a { - display: block; -} -#tsd-search .results a:before { - top: 10px; -} -#tsd-search .results span.parent { - color: #707070; - font-weight: normal; -} -#tsd-search.has-focus { - background-color: #eee; -} -#tsd-search.has-focus .field input { - top: 0; - opacity: 1; -} -#tsd-search.has-focus .title { - z-index: 0; - opacity: 0; -} -#tsd-search.has-focus .results { - visibility: visible; -} -#tsd-search.loading .results li.state.loading { - display: block; -} -#tsd-search.failure .results li.state.failure { - display: block; -} -.tsd-signature { - margin: 0 0 1em 0; - padding: 10px; - border: 1px solid #eee; - font-family: Menlo, Monaco, Consolas, "Courier New", monospace; - font-size: 14px; - overflow-x: auto; -} -.tsd-signature.tsd-kind-icon { - padding-left: 30px; -} -.tsd-signature.tsd-kind-icon:before { - top: 10px; - left: 10px; -} -.tsd-panel > .tsd-signature { - margin-left: -20px; - margin-right: -20px; - border-width: 1px 0; -} -.tsd-panel > .tsd-signature.tsd-kind-icon { - padding-left: 40px; -} -.tsd-panel > .tsd-signature.tsd-kind-icon:before { - left: 20px; -} -.tsd-signature-symbol { - color: #707070; - font-weight: normal; -} -.tsd-signature-type { - font-style: italic; - font-weight: normal; -} -.tsd-signatures { - padding: 0; - margin: 0 0 1em 0; - border: 1px solid #eee; -} -.tsd-signatures .tsd-signature { - margin: 0; - border-width: 1px 0 0 0; - transition: background-color 0.1s; -} -.tsd-signatures .tsd-signature:first-child { - border-top-width: 0; -} -.tsd-signatures .tsd-signature.current { - background-color: #eee; -} -.tsd-signatures.active > .tsd-signature { - cursor: pointer; -} -.tsd-panel > .tsd-signatures { - margin-left: -20px; - margin-right: -20px; - border-width: 1px 0; -} -.tsd-panel > .tsd-signatures .tsd-signature.tsd-kind-icon { - padding-left: 40px; -} -.tsd-panel > .tsd-signatures .tsd-signature.tsd-kind-icon:before { - left: 20px; -} -.tsd-panel > a.anchor + .tsd-signatures { - border-top-width: 0; - margin-top: -20px; -} -ul.tsd-descriptions { - position: relative; - overflow: hidden; - padding: 0; - list-style: none; -} -ul.tsd-descriptions.active > .tsd-description { - display: none; -} -ul.tsd-descriptions.active > .tsd-description.current { - display: block; -} -ul.tsd-descriptions.active > .tsd-description.fade-in { - animation: fade-in-delayed 0.3s; -} -ul.tsd-descriptions.active > .tsd-description.fade-out { - animation: fade-out-delayed 0.3s; - position: absolute; - display: block; - top: 0; - left: 0; - right: 0; - opacity: 0; - visibility: hidden; -} -ul.tsd-descriptions h4, -ul.tsd-descriptions .tsd-index-panel h3, -.tsd-index-panel ul.tsd-descriptions h3 { - font-size: 16px; - margin: 1em 0 0.5em 0; -} -ul.tsd-parameters, -ul.tsd-type-parameters { - list-style: square; - margin: 0; - padding-left: 20px; -} -ul.tsd-parameters > li.tsd-parameter-signature, -ul.tsd-type-parameters > li.tsd-parameter-signature { - list-style: none; - margin-left: -20px; -} -ul.tsd-parameters h5, -ul.tsd-type-parameters h5 { - font-size: 16px; - margin: 1em 0 0.5em 0; -} -ul.tsd-parameters .tsd-comment, -ul.tsd-type-parameters .tsd-comment { - margin-top: -0.5em; -} -.tsd-sources { - font-size: 14px; - color: #707070; - margin: 0 0 1em 0; -} -.tsd-sources a { - color: #707070; - text-decoration: underline; -} -.tsd-sources ul, -.tsd-sources p { - margin: 0 !important; -} -.tsd-sources ul { - list-style: none; - padding: 0; -} -.tsd-page-toolbar { - position: fixed; - z-index: 1; - top: 0; - left: 0; - width: 100%; - height: 40px; - color: #333; - background: #fff; - border-bottom: 1px solid #eee; - transition: transform 0.3s linear; -} -.tsd-page-toolbar a { - color: #333; - text-decoration: none; -} -.tsd-page-toolbar a.title { - font-weight: bold; -} -.tsd-page-toolbar a.title:hover { - text-decoration: underline; -} -.tsd-page-toolbar .table-wrap { - display: table; - width: 100%; - height: 40px; -} -.tsd-page-toolbar .table-cell { - display: table-cell; - position: relative; - white-space: nowrap; - line-height: 40px; -} -.tsd-page-toolbar .table-cell:first-child { - width: 100%; -} -.tsd-page-toolbar--hide { - transform: translateY(-100%); -} -.tsd-select .tsd-select-list li:before, -.tsd-select .tsd-select-label:before, -.tsd-widget:before { - content: ""; - display: inline-block; - width: 40px; - height: 40px; - margin: 0 -8px 0 0; - background-image: url(); - background-repeat: no-repeat; - text-indent: -1024px; - vertical-align: bottom; -} -@media (-webkit-min-device-pixel-ratio: 1.5), (min-resolution: 144dpi) { - .tsd-select .tsd-select-list li:before, - .tsd-select .tsd-select-label:before, - .tsd-widget:before { - background-image: url(); - background-size: 320px 40px; - } -} -.tsd-widget { - display: inline-block; - overflow: hidden; - opacity: 0.6; - height: 40px; - transition: opacity 0.1s, background-color 0.2s; - vertical-align: bottom; - cursor: pointer; -} -.tsd-widget:hover { - opacity: 0.8; -} -.tsd-widget.active { - opacity: 1; - background-color: #eee; -} -.tsd-widget.no-caption { - width: 40px; -} -.tsd-widget.no-caption:before { - margin: 0; -} -.tsd-widget.search:before { - background-position: 0 0; -} -.tsd-widget.menu:before { - background-position: -40px 0; -} -.tsd-widget.options:before { - background-position: -80px 0; -} -.tsd-widget.options, -.tsd-widget.menu { - display: none; -} -@media (max-width: 900px) { - .tsd-widget.options, - .tsd-widget.menu { - display: inline-block; - } -} -input[type="checkbox"] + .tsd-widget:before { - background-position: -120px 0; -} -input[type="checkbox"]:checked + .tsd-widget:before { - background-position: -160px 0; -} -.tsd-select { - position: relative; - display: inline-block; - height: 40px; - transition: opacity 0.1s, background-color 0.2s; - vertical-align: bottom; - cursor: pointer; -} -.tsd-select .tsd-select-label { - opacity: 0.6; - transition: opacity 0.2s; -} -.tsd-select .tsd-select-label:before { - background-position: -240px 0; -} -.tsd-select.active .tsd-select-label { - opacity: 0.8; -} -.tsd-select.active .tsd-select-list { - visibility: visible; - opacity: 1; - transition-delay: 0s; -} -.tsd-select .tsd-select-list { - position: absolute; - visibility: hidden; - top: 40px; - left: 0; - margin: 0; - padding: 0; - opacity: 0; - list-style: none; - box-shadow: 0 0 4px rgba(0, 0, 0, 0.25); - transition: visibility 0s 0.2s, opacity 0.2s; -} -.tsd-select .tsd-select-list li { - padding: 0 20px 0 0; - background-color: #fdfdfd; -} -.tsd-select .tsd-select-list li:before { - background-position: 40px 0; -} -.tsd-select .tsd-select-list li:nth-child(even) { - background-color: #fff; -} -.tsd-select .tsd-select-list li:hover { - background-color: #eee; -} -.tsd-select .tsd-select-list li.selected:before { - background-position: -200px 0; -} -@media (max-width: 900px) { - .tsd-select .tsd-select-list { - top: 0; - left: auto; - right: 100%; - margin-right: -5px; - } - .tsd-select .tsd-select-label:before { - background-position: -280px 0; - } -} -img { - max-width: 100%; -} +/*! normalize.css v1.1.3 | MIT License | git.io/normalize */article,aside,details,figcaption,figure,footer,header,hgroup,main,nav,section,summary{display:block}audio,canvas,video{display:inline-block;*display:inline;*zoom:1}audio:not([controls]){display:none;height:0}[hidden]{display:none}html{font-size:100%;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%;font-family:sans-serif}button,input,select,textarea{font-family:sans-serif}body{margin:0}a:focus{outline:thin dotted}a:active,a:hover{outline:0}h1{font-size:2em;margin:.67em 0}h2{font-size:1.5em;margin:.83em 0}h3{font-size:1.17em;margin:1em 0}h4,.tsd-index-panel h3{font-size:1em;margin:1.33em 0}h5{font-size:.83em;margin:1.67em 0}h6{font-size:.67em;margin:2.33em 0}abbr[title]{border-bottom:1px dotted}b,strong{font-weight:bold}blockquote{margin:1em 40px}dfn{font-style:italic}hr{-moz-box-sizing:content-box;box-sizing:content-box;height:0}mark{background:#ff0;color:#000}p,pre{margin:1em 0}code,kbd,pre,samp{font-family:monospace,serif;_font-family:"courier new",monospace;font-size:1em}pre{white-space:pre;white-space:pre-wrap;word-wrap:break-word}q{quotes:none}q:before,q:after{content:"";content:none}small{font-size:80%}sub{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline;top:-0.5em}sub{bottom:-0.25em}dl,menu,ol,ul{margin:1em 0}dd{margin:0 0 0 40px}menu,ol,ul{padding:0 0 0 40px}nav ul,nav ol{list-style:none;list-style-image:none}img{border:0;-ms-interpolation-mode:bicubic}svg:not(:root){overflow:hidden}figure,form{margin:0}fieldset{border:1px solid silver;margin:0 2px;padding:.35em .625em .75em}legend{border:0;padding:0;white-space:normal;*margin-left:-7px}button,input,select,textarea{font-size:100%;margin:0;vertical-align:baseline;*vertical-align:middle}button,input{line-height:normal}button,select{text-transform:none}button,html input[type=button]{-webkit-appearance:button;cursor:pointer;*overflow:visible}input[type=reset],input[type=submit]{-webkit-appearance:button;cursor:pointer;*overflow:visible}button[disabled],html input[disabled]{cursor:default}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0;*height:13px;*width:13px}input[type=search]{-webkit-appearance:textfield;-moz-box-sizing:content-box;-webkit-box-sizing:content-box;box-sizing:content-box}input[type=search]::-webkit-search-cancel-button,input[type=search]::-webkit-search-decoration{-webkit-appearance:none}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}textarea{overflow:auto;vertical-align:top}table{border-collapse:collapse;border-spacing:0}.hljs{display:inline-block;padding:.5em;background:#fff;color:#000}.hljs-comment,.hljs-annotation,.hljs-template_comment,.diff .hljs-header,.hljs-chunk,.apache .hljs-cbracket{color:green}.hljs-keyword,.hljs-id,.hljs-built_in,.css .smalltalk .hljs-class,.hljs-winutils,.bash .hljs-variable,.tex .hljs-command,.hljs-request,.hljs-status,.nginx .hljs-title{color:blue}.xml .hljs-tag{color:blue}.xml .hljs-tag .hljs-value{color:blue}.hljs-string,.hljs-title,.hljs-parent,.hljs-tag .hljs-value,.hljs-rules .hljs-value{color:#a31515}.ruby .hljs-symbol{color:#a31515}.ruby .hljs-symbol .hljs-string{color:#a31515}.hljs-template_tag,.django .hljs-variable,.hljs-addition,.hljs-flow,.hljs-stream,.apache .hljs-tag,.hljs-date,.tex .hljs-formula,.coffeescript .hljs-attribute{color:#a31515}.ruby .hljs-string,.hljs-decorator,.hljs-filter .hljs-argument,.hljs-localvars,.hljs-array,.hljs-attr_selector,.hljs-pseudo,.hljs-pi,.hljs-doctype,.hljs-deletion,.hljs-envvar,.hljs-shebang,.hljs-preprocessor,.hljs-pragma,.userType,.apache .hljs-sqbracket,.nginx .hljs-built_in,.tex .hljs-special,.hljs-prompt{color:#2b91af}.hljs-phpdoc,.hljs-javadoc,.hljs-xmlDocTag{color:gray}.vhdl .hljs-typename{font-weight:bold}.vhdl .hljs-string{color:#666}.vhdl .hljs-literal{color:#a31515}.vhdl .hljs-attribute{color:#00b0e8}.xml .hljs-attribute{color:red}ul.tsd-descriptions>li>:first-child,.tsd-panel>:first-child,.col>:first-child,.col-11>:first-child,.col-10>:first-child,.col-9>:first-child,.col-8>:first-child,.col-7>:first-child,.col-6>:first-child,.col-5>:first-child,.col-4>:first-child,.col-3>:first-child,.col-2>:first-child,.col-1>:first-child,ul.tsd-descriptions>li>:first-child>:first-child,.tsd-panel>:first-child>:first-child,.col>:first-child>:first-child,.col-11>:first-child>:first-child,.col-10>:first-child>:first-child,.col-9>:first-child>:first-child,.col-8>:first-child>:first-child,.col-7>:first-child>:first-child,.col-6>:first-child>:first-child,.col-5>:first-child>:first-child,.col-4>:first-child>:first-child,.col-3>:first-child>:first-child,.col-2>:first-child>:first-child,.col-1>:first-child>:first-child,ul.tsd-descriptions>li>:first-child>:first-child>:first-child,.tsd-panel>:first-child>:first-child>:first-child,.col>:first-child>:first-child>:first-child,.col-11>:first-child>:first-child>:first-child,.col-10>:first-child>:first-child>:first-child,.col-9>:first-child>:first-child>:first-child,.col-8>:first-child>:first-child>:first-child,.col-7>:first-child>:first-child>:first-child,.col-6>:first-child>:first-child>:first-child,.col-5>:first-child>:first-child>:first-child,.col-4>:first-child>:first-child>:first-child,.col-3>:first-child>:first-child>:first-child,.col-2>:first-child>:first-child>:first-child,.col-1>:first-child>:first-child>:first-child{margin-top:0}ul.tsd-descriptions>li>:last-child,.tsd-panel>:last-child,.col>:last-child,.col-11>:last-child,.col-10>:last-child,.col-9>:last-child,.col-8>:last-child,.col-7>:last-child,.col-6>:last-child,.col-5>:last-child,.col-4>:last-child,.col-3>:last-child,.col-2>:last-child,.col-1>:last-child,ul.tsd-descriptions>li>:last-child>:last-child,.tsd-panel>:last-child>:last-child,.col>:last-child>:last-child,.col-11>:last-child>:last-child,.col-10>:last-child>:last-child,.col-9>:last-child>:last-child,.col-8>:last-child>:last-child,.col-7>:last-child>:last-child,.col-6>:last-child>:last-child,.col-5>:last-child>:last-child,.col-4>:last-child>:last-child,.col-3>:last-child>:last-child,.col-2>:last-child>:last-child,.col-1>:last-child>:last-child,ul.tsd-descriptions>li>:last-child>:last-child>:last-child,.tsd-panel>:last-child>:last-child>:last-child,.col>:last-child>:last-child>:last-child,.col-11>:last-child>:last-child>:last-child,.col-10>:last-child>:last-child>:last-child,.col-9>:last-child>:last-child>:last-child,.col-8>:last-child>:last-child>:last-child,.col-7>:last-child>:last-child>:last-child,.col-6>:last-child>:last-child>:last-child,.col-5>:last-child>:last-child>:last-child,.col-4>:last-child>:last-child>:last-child,.col-3>:last-child>:last-child>:last-child,.col-2>:last-child>:last-child>:last-child,.col-1>:last-child>:last-child>:last-child{margin-bottom:0}.container{max-width:1200px;margin:0 auto;padding:0 40px}@media(max-width: 640px){.container{padding:0 20px}}.container-main{padding-bottom:200px}.row{display:flex;position:relative;margin:0 -10px}.row:after{visibility:hidden;display:block;content:"";clear:both;height:0}.col,.col-11,.col-10,.col-9,.col-8,.col-7,.col-6,.col-5,.col-4,.col-3,.col-2,.col-1{box-sizing:border-box;float:left;padding:0 10px}.col-1{width:8.3333333333%}.offset-1{margin-left:8.3333333333%}.col-2{width:16.6666666667%}.offset-2{margin-left:16.6666666667%}.col-3{width:25%}.offset-3{margin-left:25%}.col-4{width:33.3333333333%}.offset-4{margin-left:33.3333333333%}.col-5{width:41.6666666667%}.offset-5{margin-left:41.6666666667%}.col-6{width:50%}.offset-6{margin-left:50%}.col-7{width:58.3333333333%}.offset-7{margin-left:58.3333333333%}.col-8{width:66.6666666667%}.offset-8{margin-left:66.6666666667%}.col-9{width:75%}.offset-9{margin-left:75%}.col-10{width:83.3333333333%}.offset-10{margin-left:83.3333333333%}.col-11{width:91.6666666667%}.offset-11{margin-left:91.6666666667%}.tsd-kind-icon{display:block;position:relative;padding-left:20px;text-indent:-20px}.tsd-kind-icon:before{content:"";display:inline-block;vertical-align:middle;width:17px;height:17px;margin:0 3px 2px 0;background-image:url()}@media(-webkit-min-device-pixel-ratio: 1.5),(min-resolution: 144dpi){.tsd-kind-icon:before{background-image:url();background-size:238px 204px}}.tsd-signature.tsd-kind-icon:before{background-position:0 -153px}.tsd-kind-object-literal>.tsd-kind-icon:before{background-position:0px -17px}.tsd-kind-object-literal.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -17px}.tsd-kind-object-literal.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -17px}.tsd-kind-class>.tsd-kind-icon:before{background-position:0px -34px}.tsd-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -34px}.tsd-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -34px}.tsd-kind-class.tsd-has-type-parameter>.tsd-kind-icon:before{background-position:0px -51px}.tsd-kind-class.tsd-has-type-parameter.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -51px}.tsd-kind-class.tsd-has-type-parameter.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -51px}.tsd-kind-interface>.tsd-kind-icon:before{background-position:0px -68px}.tsd-kind-interface.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -68px}.tsd-kind-interface.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -68px}.tsd-kind-interface.tsd-has-type-parameter>.tsd-kind-icon:before{background-position:0px -85px}.tsd-kind-interface.tsd-has-type-parameter.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -85px}.tsd-kind-interface.tsd-has-type-parameter.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -85px}.tsd-kind-namespace>.tsd-kind-icon:before{background-position:0px -102px}.tsd-kind-namespace.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -102px}.tsd-kind-namespace.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -102px}.tsd-kind-module>.tsd-kind-icon:before{background-position:0px -102px}.tsd-kind-module.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -102px}.tsd-kind-module.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -102px}.tsd-kind-enum>.tsd-kind-icon:before{background-position:0px -119px}.tsd-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -119px}.tsd-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -119px}.tsd-kind-enum-member>.tsd-kind-icon:before{background-position:0px -136px}.tsd-kind-enum-member.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -136px}.tsd-kind-enum-member.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -136px}.tsd-kind-signature>.tsd-kind-icon:before{background-position:0px -153px}.tsd-kind-signature.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -153px}.tsd-kind-signature.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -153px}.tsd-kind-type-alias>.tsd-kind-icon:before{background-position:0px -170px}.tsd-kind-type-alias.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -170px}.tsd-kind-type-alias.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -170px}.tsd-kind-type-alias.tsd-has-type-parameter>.tsd-kind-icon:before{background-position:0px -187px}.tsd-kind-type-alias.tsd-has-type-parameter.tsd-is-protected>.tsd-kind-icon:before{background-position:-17px -187px}.tsd-kind-type-alias.tsd-has-type-parameter.tsd-is-private>.tsd-kind-icon:before{background-position:-34px -187px}.tsd-kind-variable>.tsd-kind-icon:before{background-position:-136px -0px}.tsd-kind-variable.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -0px}.tsd-kind-variable.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -0px}.tsd-kind-variable.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -0px}.tsd-kind-variable.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -0px}.tsd-kind-variable.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -0px}.tsd-kind-variable.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -0px}.tsd-kind-variable.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -0px}.tsd-kind-variable.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -0px}.tsd-kind-variable.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -0px}.tsd-kind-variable.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -0px}.tsd-kind-variable.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -0px}.tsd-kind-variable.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -0px}.tsd-kind-property>.tsd-kind-icon:before{background-position:-136px -0px}.tsd-kind-property.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -0px}.tsd-kind-property.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -0px}.tsd-kind-property.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -0px}.tsd-kind-property.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -0px}.tsd-kind-property.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -0px}.tsd-kind-property.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -0px}.tsd-kind-property.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -0px}.tsd-kind-property.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -0px}.tsd-kind-property.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -0px}.tsd-kind-property.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -0px}.tsd-kind-property.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -0px}.tsd-kind-property.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -0px}.tsd-kind-get-signature>.tsd-kind-icon:before{background-position:-136px -17px}.tsd-kind-get-signature.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -17px}.tsd-kind-get-signature.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -17px}.tsd-kind-get-signature.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -17px}.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -17px}.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -17px}.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -17px}.tsd-kind-get-signature.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -17px}.tsd-kind-get-signature.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -17px}.tsd-kind-get-signature.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -17px}.tsd-kind-get-signature.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -17px}.tsd-kind-get-signature.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -17px}.tsd-kind-get-signature.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -17px}.tsd-kind-set-signature>.tsd-kind-icon:before{background-position:-136px -34px}.tsd-kind-set-signature.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -34px}.tsd-kind-set-signature.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -34px}.tsd-kind-set-signature.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -34px}.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -34px}.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -34px}.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -34px}.tsd-kind-set-signature.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -34px}.tsd-kind-set-signature.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -34px}.tsd-kind-set-signature.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -34px}.tsd-kind-set-signature.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -34px}.tsd-kind-set-signature.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -34px}.tsd-kind-set-signature.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -34px}.tsd-kind-accessor>.tsd-kind-icon:before{background-position:-136px -51px}.tsd-kind-accessor.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -51px}.tsd-kind-accessor.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -51px}.tsd-kind-accessor.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -51px}.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -51px}.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -51px}.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -51px}.tsd-kind-accessor.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -51px}.tsd-kind-accessor.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -51px}.tsd-kind-accessor.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -51px}.tsd-kind-accessor.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -51px}.tsd-kind-accessor.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -51px}.tsd-kind-accessor.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -51px}.tsd-kind-function>.tsd-kind-icon:before{background-position:-136px -68px}.tsd-kind-function.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -68px}.tsd-kind-function.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -68px}.tsd-kind-function.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -68px}.tsd-kind-function.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -68px}.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -68px}.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -68px}.tsd-kind-function.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -68px}.tsd-kind-function.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -68px}.tsd-kind-function.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -68px}.tsd-kind-function.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -68px}.tsd-kind-function.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -68px}.tsd-kind-function.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -68px}.tsd-kind-method>.tsd-kind-icon:before{background-position:-136px -68px}.tsd-kind-method.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -68px}.tsd-kind-method.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -68px}.tsd-kind-method.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -68px}.tsd-kind-method.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -68px}.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -68px}.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -68px}.tsd-kind-method.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -68px}.tsd-kind-method.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -68px}.tsd-kind-method.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -68px}.tsd-kind-method.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -68px}.tsd-kind-method.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -68px}.tsd-kind-method.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -68px}.tsd-kind-call-signature>.tsd-kind-icon:before{background-position:-136px -68px}.tsd-kind-call-signature.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -68px}.tsd-kind-call-signature.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -68px}.tsd-kind-call-signature.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -68px}.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -68px}.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -68px}.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -68px}.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -68px}.tsd-kind-call-signature.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -68px}.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -68px}.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -68px}.tsd-kind-call-signature.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -68px}.tsd-kind-call-signature.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -68px}.tsd-kind-function.tsd-has-type-parameter>.tsd-kind-icon:before{background-position:-136px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -85px}.tsd-kind-function.tsd-has-type-parameter.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -85px}.tsd-kind-method.tsd-has-type-parameter>.tsd-kind-icon:before{background-position:-136px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -85px}.tsd-kind-method.tsd-has-type-parameter.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -85px}.tsd-kind-constructor>.tsd-kind-icon:before{background-position:-136px -102px}.tsd-kind-constructor.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -102px}.tsd-kind-constructor.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -102px}.tsd-kind-constructor.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -102px}.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -102px}.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -102px}.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -102px}.tsd-kind-constructor.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -102px}.tsd-kind-constructor.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -102px}.tsd-kind-constructor.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -102px}.tsd-kind-constructor.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -102px}.tsd-kind-constructor.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -102px}.tsd-kind-constructor.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -102px}.tsd-kind-constructor-signature>.tsd-kind-icon:before{background-position:-136px -102px}.tsd-kind-constructor-signature.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -102px}.tsd-kind-constructor-signature.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -102px}.tsd-kind-constructor-signature.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -102px}.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -102px}.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -102px}.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -102px}.tsd-kind-constructor-signature.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -102px}.tsd-kind-constructor-signature.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -102px}.tsd-kind-constructor-signature.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -102px}.tsd-kind-constructor-signature.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -102px}.tsd-kind-constructor-signature.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -102px}.tsd-kind-constructor-signature.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -102px}.tsd-kind-index-signature>.tsd-kind-icon:before{background-position:-136px -119px}.tsd-kind-index-signature.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -119px}.tsd-kind-index-signature.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -119px}.tsd-kind-index-signature.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -119px}.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -119px}.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -119px}.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -119px}.tsd-kind-index-signature.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -119px}.tsd-kind-index-signature.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -119px}.tsd-kind-index-signature.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -119px}.tsd-kind-index-signature.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -119px}.tsd-kind-index-signature.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -119px}.tsd-kind-index-signature.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -119px}.tsd-kind-event>.tsd-kind-icon:before{background-position:-136px -136px}.tsd-kind-event.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -136px}.tsd-kind-event.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -136px}.tsd-kind-event.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -136px}.tsd-kind-event.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -136px}.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -136px}.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -136px}.tsd-kind-event.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -136px}.tsd-kind-event.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -136px}.tsd-kind-event.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -136px}.tsd-kind-event.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -136px}.tsd-kind-event.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -136px}.tsd-kind-event.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -136px}.tsd-is-static>.tsd-kind-icon:before{background-position:-136px -153px}.tsd-is-static.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -153px}.tsd-is-static.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -153px}.tsd-is-static.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -153px}.tsd-is-static.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -153px}.tsd-is-static.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -153px}.tsd-is-static.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -153px}.tsd-is-static.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -153px}.tsd-is-static.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -153px}.tsd-is-static.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -153px}.tsd-is-static.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -153px}.tsd-is-static.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -153px}.tsd-is-static.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -153px}.tsd-is-static.tsd-kind-function>.tsd-kind-icon:before{background-position:-136px -170px}.tsd-is-static.tsd-kind-function.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -170px}.tsd-is-static.tsd-kind-function.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -170px}.tsd-is-static.tsd-kind-function.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -170px}.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -170px}.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -170px}.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -170px}.tsd-is-static.tsd-kind-function.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -170px}.tsd-is-static.tsd-kind-function.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -170px}.tsd-is-static.tsd-kind-function.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -170px}.tsd-is-static.tsd-kind-function.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -170px}.tsd-is-static.tsd-kind-function.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -170px}.tsd-is-static.tsd-kind-function.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -170px}.tsd-is-static.tsd-kind-method>.tsd-kind-icon:before{background-position:-136px -170px}.tsd-is-static.tsd-kind-method.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -170px}.tsd-is-static.tsd-kind-method.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -170px}.tsd-is-static.tsd-kind-method.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -170px}.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -170px}.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -170px}.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -170px}.tsd-is-static.tsd-kind-method.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -170px}.tsd-is-static.tsd-kind-method.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -170px}.tsd-is-static.tsd-kind-method.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -170px}.tsd-is-static.tsd-kind-method.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -170px}.tsd-is-static.tsd-kind-method.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -170px}.tsd-is-static.tsd-kind-method.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -170px}.tsd-is-static.tsd-kind-call-signature>.tsd-kind-icon:before{background-position:-136px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -170px}.tsd-is-static.tsd-kind-call-signature.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -170px}.tsd-is-static.tsd-kind-event>.tsd-kind-icon:before{background-position:-136px -187px}.tsd-is-static.tsd-kind-event.tsd-is-protected>.tsd-kind-icon:before{background-position:-153px -187px}.tsd-is-static.tsd-kind-event.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -187px}.tsd-is-static.tsd-kind-event.tsd-parent-kind-class>.tsd-kind-icon:before{background-position:-51px -187px}.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-inherited>.tsd-kind-icon:before{background-position:-68px -187px}.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected>.tsd-kind-icon:before{background-position:-85px -187px}.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-protected.tsd-is-inherited>.tsd-kind-icon:before{background-position:-102px -187px}.tsd-is-static.tsd-kind-event.tsd-parent-kind-class.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -187px}.tsd-is-static.tsd-kind-event.tsd-parent-kind-enum>.tsd-kind-icon:before{background-position:-170px -187px}.tsd-is-static.tsd-kind-event.tsd-parent-kind-enum.tsd-is-protected>.tsd-kind-icon:before{background-position:-187px -187px}.tsd-is-static.tsd-kind-event.tsd-parent-kind-enum.tsd-is-private>.tsd-kind-icon:before{background-position:-119px -187px}.tsd-is-static.tsd-kind-event.tsd-parent-kind-interface>.tsd-kind-icon:before{background-position:-204px -187px}.tsd-is-static.tsd-kind-event.tsd-parent-kind-interface.tsd-is-inherited>.tsd-kind-icon:before{background-position:-221px -187px}@keyframes fade-in{from{opacity:0}to{opacity:1}}@keyframes fade-out{from{opacity:1;visibility:visible}to{opacity:0}}@keyframes fade-in-delayed{0%{opacity:0}33%{opacity:0}100%{opacity:1}}@keyframes fade-out-delayed{0%{opacity:1;visibility:visible}66%{opacity:0}100%{opacity:0}}@keyframes shift-to-left{from{transform:translate(0, 0)}to{transform:translate(-25%, 0)}}@keyframes unshift-to-left{from{transform:translate(-25%, 0)}to{transform:translate(0, 0)}}@keyframes pop-in-from-right{from{transform:translate(100%, 0)}to{transform:translate(0, 0)}}@keyframes pop-out-to-right{from{transform:translate(0, 0);visibility:visible}to{transform:translate(100%, 0)}}body{background:#fdfdfd;font-family:"Segoe UI",sans-serif;font-size:16px;color:#222}a{color:#4da6ff;text-decoration:none}a:hover{text-decoration:underline}code,pre{font-family:Menlo,Monaco,Consolas,"Courier New",monospace;padding:.2em;margin:0;font-size:14px;background-color:rgba(0,0,0,.04)}pre{padding:10px}pre code{padding:0;font-size:100%;background-color:transparent}.tsd-typography{line-height:1.333em}.tsd-typography ul{list-style:square;padding:0 0 0 20px;margin:0}.tsd-typography h4,.tsd-typography .tsd-index-panel h3,.tsd-index-panel .tsd-typography h3,.tsd-typography h5,.tsd-typography h6{font-size:1em;margin:0}.tsd-typography h5,.tsd-typography h6{font-weight:normal}.tsd-typography p,.tsd-typography ul,.tsd-typography ol{margin:1em 0}@media(min-width: 901px)and (max-width: 1024px){html.default .col-content{width:72%}html.default .col-menu{width:28%}html.default .tsd-navigation{padding-left:10px}}@media(max-width: 900px){html.default .col-content{float:none;width:100%}html.default .col-menu{position:fixed !important;overflow:auto;-webkit-overflow-scrolling:touch;z-index:1024;top:0 !important;bottom:0 !important;left:auto !important;right:0 !important;width:100%;padding:20px 20px 0 0;max-width:450px;visibility:hidden;background-color:#fff;transform:translate(100%, 0)}html.default .col-menu>*:last-child{padding-bottom:20px}html.default .overlay{content:"";display:block;position:fixed;z-index:1023;top:0;left:0;right:0;bottom:0;background-color:rgba(0,0,0,.75);visibility:hidden}html.default.to-has-menu .overlay{animation:fade-in .4s}html.default.to-has-menu header,html.default.to-has-menu footer,html.default.to-has-menu .col-content{animation:shift-to-left .4s}html.default.to-has-menu .col-menu{animation:pop-in-from-right .4s}html.default.from-has-menu .overlay{animation:fade-out .4s}html.default.from-has-menu header,html.default.from-has-menu footer,html.default.from-has-menu .col-content{animation:unshift-to-left .4s}html.default.from-has-menu .col-menu{animation:pop-out-to-right .4s}html.default.has-menu body{overflow:hidden}html.default.has-menu .overlay{visibility:visible}html.default.has-menu header,html.default.has-menu footer,html.default.has-menu .col-content{transform:translate(-25%, 0)}html.default.has-menu .col-menu{visibility:visible;transform:translate(0, 0)}}.tsd-page-title{padding:70px 0 20px 0;margin:0 0 40px 0;background:#fff;box-shadow:0 0 5px rgba(0,0,0,.35)}.tsd-page-title h1{margin:0}.tsd-breadcrumb{margin:0;padding:0;color:#707070}.tsd-breadcrumb a{color:#707070;text-decoration:none}.tsd-breadcrumb a:hover{text-decoration:underline}.tsd-breadcrumb li{display:inline}.tsd-breadcrumb li:after{content:" / "}html.minimal .container{margin:0}html.minimal .container-main{padding-top:50px;padding-bottom:0}html.minimal .content-wrap{padding-left:300px}html.minimal .tsd-navigation{position:fixed !important;overflow:auto;-webkit-overflow-scrolling:touch;box-sizing:border-box;z-index:1;left:0;top:40px;bottom:0;width:300px;padding:20px;margin:0}html.minimal .tsd-member .tsd-member{margin-left:0}html.minimal .tsd-page-toolbar{position:fixed;z-index:2}html.minimal #tsd-filter .tsd-filter-group{right:0;transform:none}html.minimal footer{background-color:transparent}html.minimal footer .container{padding:0}html.minimal .tsd-generator{padding:0}@media(max-width: 900px){html.minimal .tsd-navigation{display:none}html.minimal .content-wrap{padding-left:0}}dl.tsd-comment-tags{overflow:hidden}dl.tsd-comment-tags dt{float:left;padding:1px 5px;margin:0 10px 0 0;border-radius:4px;border:1px solid #707070;color:#707070;font-size:.8em;font-weight:normal}dl.tsd-comment-tags dd{margin:0 0 10px 0}dl.tsd-comment-tags dd:before,dl.tsd-comment-tags dd:after{display:table;content:" "}dl.tsd-comment-tags dd pre,dl.tsd-comment-tags dd:after{clear:both}dl.tsd-comment-tags p{margin:0}.tsd-panel.tsd-comment .lead{font-size:1.1em;line-height:1.333em;margin-bottom:2em}.tsd-panel.tsd-comment .lead:last-child{margin-bottom:0}.toggle-protected .tsd-is-private{display:none}.toggle-public .tsd-is-private,.toggle-public .tsd-is-protected,.toggle-public .tsd-is-private-protected{display:none}.toggle-inherited .tsd-is-inherited{display:none}.toggle-only-exported .tsd-is-not-exported{display:none}.toggle-externals .tsd-is-external{display:none}#tsd-filter{position:relative;display:inline-block;height:40px;vertical-align:bottom}.no-filter #tsd-filter{display:none}#tsd-filter .tsd-filter-group{display:inline-block;height:40px;vertical-align:bottom;white-space:nowrap}#tsd-filter input{display:none}@media(max-width: 900px){#tsd-filter .tsd-filter-group{display:block;position:absolute;top:40px;right:20px;height:auto;background-color:#fff;visibility:hidden;transform:translate(50%, 0);box-shadow:0 0 4px rgba(0,0,0,.25)}.has-options #tsd-filter .tsd-filter-group{visibility:visible}.to-has-options #tsd-filter .tsd-filter-group{animation:fade-in .2s}.from-has-options #tsd-filter .tsd-filter-group{animation:fade-out .2s}#tsd-filter label,#tsd-filter .tsd-select{display:block;padding-right:20px}}footer{border-top:1px solid #eee;background-color:#fff}footer.with-border-bottom{border-bottom:1px solid #eee}footer .tsd-legend-group{font-size:0}footer .tsd-legend{display:inline-block;width:25%;padding:0;font-size:16px;list-style:none;line-height:1.333em;vertical-align:top}@media(max-width: 900px){footer .tsd-legend{width:50%}}.tsd-hierarchy{list-style:square;padding:0 0 0 20px;margin:0}.tsd-hierarchy .target{font-weight:bold}.tsd-index-panel .tsd-index-content{margin-bottom:-30px !important}.tsd-index-panel .tsd-index-section{margin-bottom:30px !important}.tsd-index-panel h3{margin:0 -20px 10px -20px;padding:0 20px 10px 20px;border-bottom:1px solid #eee}.tsd-index-panel ul.tsd-index-list{-webkit-column-count:3;-moz-column-count:3;-ms-column-count:3;-o-column-count:3;column-count:3;-webkit-column-gap:20px;-moz-column-gap:20px;-ms-column-gap:20px;-o-column-gap:20px;column-gap:20px;padding:0;list-style:none;line-height:1.333em}@media(max-width: 900px){.tsd-index-panel ul.tsd-index-list{-webkit-column-count:1;-moz-column-count:1;-ms-column-count:1;-o-column-count:1;column-count:1}}@media(min-width: 901px)and (max-width: 1024px){.tsd-index-panel ul.tsd-index-list{-webkit-column-count:2;-moz-column-count:2;-ms-column-count:2;-o-column-count:2;column-count:2}}.tsd-index-panel ul.tsd-index-list li{-webkit-page-break-inside:avoid;-moz-page-break-inside:avoid;-ms-page-break-inside:avoid;-o-page-break-inside:avoid;page-break-inside:avoid}.tsd-index-panel a,.tsd-index-panel .tsd-parent-kind-module a{color:#9600ff}.tsd-index-panel .tsd-parent-kind-interface a{color:#647f1b}.tsd-index-panel .tsd-parent-kind-enum a{color:#937210}.tsd-index-panel .tsd-parent-kind-class a{color:#0672de}.tsd-index-panel .tsd-kind-module a{color:#9600ff}.tsd-index-panel .tsd-kind-interface a{color:#647f1b}.tsd-index-panel .tsd-kind-enum a{color:#937210}.tsd-index-panel .tsd-kind-class a{color:#0672de}.tsd-index-panel .tsd-is-private a{color:#707070}.tsd-flag{display:inline-block;padding:1px 5px;border-radius:4px;color:#fff;background-color:#707070;text-indent:0;font-size:14px;font-weight:normal}.tsd-anchor{position:absolute;top:-100px}.tsd-member{position:relative}.tsd-member .tsd-anchor+h3{margin-top:0;margin-bottom:0;border-bottom:none}.tsd-navigation{margin:0 0 0 40px}.tsd-navigation a{display:block;padding-top:2px;padding-bottom:2px;border-left:2px solid transparent;color:#222;text-decoration:none;transition:border-left-color .1s}.tsd-navigation a:hover{text-decoration:underline}.tsd-navigation ul{margin:0;padding:0;list-style:none}.tsd-navigation li{padding:0}.tsd-navigation.primary{padding-bottom:40px}.tsd-navigation.primary a{display:block;padding-top:6px;padding-bottom:6px}.tsd-navigation.primary ul li a{padding-left:5px}.tsd-navigation.primary ul li li a{padding-left:25px}.tsd-navigation.primary ul li li li a{padding-left:45px}.tsd-navigation.primary ul li li li li a{padding-left:65px}.tsd-navigation.primary ul li li li li li a{padding-left:85px}.tsd-navigation.primary ul li li li li li li a{padding-left:105px}.tsd-navigation.primary>ul{border-bottom:1px solid #eee}.tsd-navigation.primary li{border-top:1px solid #eee}.tsd-navigation.primary li.current>a{font-weight:bold}.tsd-navigation.primary li.label span{display:block;padding:20px 0 6px 5px;color:#707070}.tsd-navigation.primary li.globals+li>span,.tsd-navigation.primary li.globals+li>a{padding-top:20px}.tsd-navigation.secondary{max-height:calc(100vh - 1rem - 40px);overflow:auto;position:-webkit-sticky;position:sticky;top:calc(.5rem + 40px);transition:.3s}.tsd-navigation.secondary.tsd-navigation--toolbar-hide{max-height:calc(100vh - 1rem);top:.5rem}.tsd-navigation.secondary ul{transition:opacity .2s}.tsd-navigation.secondary ul li a{padding-left:25px}.tsd-navigation.secondary ul li li a{padding-left:45px}.tsd-navigation.secondary ul li li li a{padding-left:65px}.tsd-navigation.secondary ul li li li li a{padding-left:85px}.tsd-navigation.secondary ul li li li li li a{padding-left:105px}.tsd-navigation.secondary ul li li li li li li a{padding-left:125px}.tsd-navigation.secondary ul.current a{border-left-color:#eee}.tsd-navigation.secondary li.focus>a,.tsd-navigation.secondary ul.current li.focus>a{border-left-color:#000}.tsd-navigation.secondary li.current{margin-top:20px;margin-bottom:20px;border-left-color:#eee}.tsd-navigation.secondary li.current>a{font-weight:bold}@media(min-width: 901px){.menu-sticky-wrap{position:static}}.tsd-panel{margin:20px 0;padding:20px;background-color:#fff;box-shadow:0 0 4px rgba(0,0,0,.25)}.tsd-panel:empty{display:none}.tsd-panel>h1,.tsd-panel>h2,.tsd-panel>h3{margin:1.5em -20px 10px -20px;padding:0 20px 10px 20px;border-bottom:1px solid #eee}.tsd-panel>h1.tsd-before-signature,.tsd-panel>h2.tsd-before-signature,.tsd-panel>h3.tsd-before-signature{margin-bottom:0;border-bottom:0}.tsd-panel table{display:block;width:100%;overflow:auto;margin-top:10px;word-break:normal;word-break:keep-all}.tsd-panel table th{font-weight:bold}.tsd-panel table th,.tsd-panel table td{padding:6px 13px;border:1px solid #ddd}.tsd-panel table tr{background-color:#fff;border-top:1px solid #ccc}.tsd-panel table tr:nth-child(2n){background-color:#f8f8f8}.tsd-panel-group{margin:60px 0}.tsd-panel-group>h1,.tsd-panel-group>h2,.tsd-panel-group>h3{padding-left:20px;padding-right:20px}#tsd-search{transition:background-color .2s}#tsd-search .title{position:relative;z-index:2}#tsd-search .field{position:absolute;left:0;top:0;right:40px;height:40px}#tsd-search .field input{box-sizing:border-box;position:relative;top:-50px;z-index:1;width:100%;padding:0 10px;opacity:0;outline:0;border:0;background:transparent;color:#222}#tsd-search .field label{position:absolute;overflow:hidden;right:-40px}#tsd-search .field input,#tsd-search .title{transition:opacity .2s}#tsd-search .results{position:absolute;visibility:hidden;top:40px;width:100%;margin:0;padding:0;list-style:none;box-shadow:0 0 4px rgba(0,0,0,.25)}#tsd-search .results li{padding:0 10px;background-color:#fdfdfd}#tsd-search .results li:nth-child(even){background-color:#fff}#tsd-search .results li.state{display:none}#tsd-search .results li.current,#tsd-search .results li:hover{background-color:#eee}#tsd-search .results a{display:block}#tsd-search .results a:before{top:10px}#tsd-search .results span.parent{color:#707070;font-weight:normal}#tsd-search.has-focus{background-color:#eee}#tsd-search.has-focus .field input{top:0;opacity:1}#tsd-search.has-focus .title{z-index:0;opacity:0}#tsd-search.has-focus .results{visibility:visible}#tsd-search.loading .results li.state.loading{display:block}#tsd-search.failure .results li.state.failure{display:block}.tsd-signature{margin:0 0 1em 0;padding:10px;border:1px solid #eee;font-family:Menlo,Monaco,Consolas,"Courier New",monospace;font-size:14px;overflow-x:auto}.tsd-signature.tsd-kind-icon{padding-left:30px}.tsd-signature.tsd-kind-icon:before{top:10px;left:10px}.tsd-panel>.tsd-signature{margin-left:-20px;margin-right:-20px;border-width:1px 0}.tsd-panel>.tsd-signature.tsd-kind-icon{padding-left:40px}.tsd-panel>.tsd-signature.tsd-kind-icon:before{left:20px}.tsd-signature-symbol{color:#707070;font-weight:normal}.tsd-signature-type{font-style:italic;font-weight:normal}.tsd-signatures{padding:0;margin:0 0 1em 0;border:1px solid #eee}.tsd-signatures .tsd-signature{margin:0;border-width:1px 0 0 0;transition:background-color .1s}.tsd-signatures .tsd-signature:first-child{border-top-width:0}.tsd-signatures .tsd-signature.current{background-color:#eee}.tsd-signatures.active>.tsd-signature{cursor:pointer}.tsd-panel>.tsd-signatures{margin-left:-20px;margin-right:-20px;border-width:1px 0}.tsd-panel>.tsd-signatures .tsd-signature.tsd-kind-icon{padding-left:40px}.tsd-panel>.tsd-signatures .tsd-signature.tsd-kind-icon:before{left:20px}.tsd-panel>a.anchor+.tsd-signatures{border-top-width:0;margin-top:-20px}ul.tsd-descriptions{position:relative;overflow:hidden;padding:0;list-style:none}ul.tsd-descriptions.active>.tsd-description{display:none}ul.tsd-descriptions.active>.tsd-description.current{display:block}ul.tsd-descriptions.active>.tsd-description.fade-in{animation:fade-in-delayed .3s}ul.tsd-descriptions.active>.tsd-description.fade-out{animation:fade-out-delayed .3s;position:absolute;display:block;top:0;left:0;right:0;opacity:0;visibility:hidden}ul.tsd-descriptions h4,ul.tsd-descriptions .tsd-index-panel h3,.tsd-index-panel ul.tsd-descriptions h3{font-size:16px;margin:1em 0 .5em 0}ul.tsd-parameters,ul.tsd-type-parameters{list-style:square;margin:0;padding-left:20px}ul.tsd-parameters>li.tsd-parameter-signature,ul.tsd-type-parameters>li.tsd-parameter-signature{list-style:none;margin-left:-20px}ul.tsd-parameters h5,ul.tsd-type-parameters h5{font-size:16px;margin:1em 0 .5em 0}ul.tsd-parameters .tsd-comment,ul.tsd-type-parameters .tsd-comment{margin-top:-0.5em}.tsd-sources{font-size:14px;color:#707070;margin:0 0 1em 0}.tsd-sources a{color:#707070;text-decoration:underline}.tsd-sources ul,.tsd-sources p{margin:0 !important}.tsd-sources ul{list-style:none;padding:0}.tsd-page-toolbar{position:fixed;z-index:1;top:0;left:0;width:100%;height:40px;color:#333;background:#fff;border-bottom:1px solid #eee;transition:transform .3s linear}.tsd-page-toolbar a{color:#333;text-decoration:none}.tsd-page-toolbar a.title{font-weight:bold}.tsd-page-toolbar a.title:hover{text-decoration:underline}.tsd-page-toolbar .table-wrap{display:table;width:100%;height:40px}.tsd-page-toolbar .table-cell{display:table-cell;position:relative;white-space:nowrap;line-height:40px}.tsd-page-toolbar .table-cell:first-child{width:100%}.tsd-page-toolbar--hide{transform:translateY(-100%)}.tsd-select .tsd-select-list li:before,.tsd-select .tsd-select-label:before,.tsd-widget:before{content:"";display:inline-block;width:40px;height:40px;margin:0 -8px 0 0;background-image:url();background-repeat:no-repeat;text-indent:-1024px;vertical-align:bottom}@media(-webkit-min-device-pixel-ratio: 1.5),(min-resolution: 144dpi){.tsd-select .tsd-select-list li:before,.tsd-select .tsd-select-label:before,.tsd-widget:before{background-image:url();background-size:320px 40px}}.tsd-widget{display:inline-block;overflow:hidden;opacity:.6;height:40px;transition:opacity .1s,background-color .2s;vertical-align:bottom;cursor:pointer}.tsd-widget:hover{opacity:.8}.tsd-widget.active{opacity:1;background-color:#eee}.tsd-widget.no-caption{width:40px}.tsd-widget.no-caption:before{margin:0}.tsd-widget.search:before{background-position:0 0}.tsd-widget.menu:before{background-position:-40px 0}.tsd-widget.options:before{background-position:-80px 0}.tsd-widget.options,.tsd-widget.menu{display:none}@media(max-width: 900px){.tsd-widget.options,.tsd-widget.menu{display:inline-block}}input[type=checkbox]+.tsd-widget:before{background-position:-120px 0}input[type=checkbox]:checked+.tsd-widget:before{background-position:-160px 0}.tsd-select{position:relative;display:inline-block;height:40px;transition:opacity .1s,background-color .2s;vertical-align:bottom;cursor:pointer}.tsd-select .tsd-select-label{opacity:.6;transition:opacity .2s}.tsd-select .tsd-select-label:before{background-position:-240px 0}.tsd-select.active .tsd-select-label{opacity:.8}.tsd-select.active .tsd-select-list{visibility:visible;opacity:1;transition-delay:0s}.tsd-select .tsd-select-list{position:absolute;visibility:hidden;top:40px;left:0;margin:0;padding:0;opacity:0;list-style:none;box-shadow:0 0 4px rgba(0,0,0,.25);transition:visibility 0s .2s,opacity .2s}.tsd-select .tsd-select-list li{padding:0 20px 0 0;background-color:#fdfdfd}.tsd-select .tsd-select-list li:before{background-position:40px 0}.tsd-select .tsd-select-list li:nth-child(even){background-color:#fff}.tsd-select .tsd-select-list li:hover{background-color:#eee}.tsd-select .tsd-select-list li.selected:before{background-position:-200px 0}@media(max-width: 900px){.tsd-select .tsd-select-list{top:0;left:auto;right:100%;margin-right:-5px}.tsd-select .tsd-select-label:before{background-position:-280px 0}}img{max-width:100%} diff --git a/docs/api/assets/js/main.js b/docs/api/assets/js/main.js index 6be660f..c2190a9 100644 --- a/docs/api/assets/js/main.js +++ b/docs/api/assets/js/main.js @@ -1,2280 +1,51 @@ -!(function (e) { - var t = {} - function r(n) { - if (t[n]) return t[n].exports - var i = (t[n] = { i: n, l: !1, exports: {} }) - return e[n].call(i.exports, i, i.exports, r), (i.l = !0), i.exports - } - ;(r.m = e), - (r.c = t), - (r.d = function (e, t, n) { - r.o(e, t) || Object.defineProperty(e, t, { enumerable: !0, get: n }) - }), - (r.r = function (e) { - "undefined" != typeof Symbol && - Symbol.toStringTag && - Object.defineProperty(e, Symbol.toStringTag, { value: "Module" }), - Object.defineProperty(e, "__esModule", { value: !0 }) - }), - (r.t = function (e, t) { - if ((1 & t && (e = r(e)), 8 & t)) return e - if (4 & t && "object" == typeof e && e && e.__esModule) return e - var n = Object.create(null) - if ( - (r.r(n), - Object.defineProperty(n, "default", { enumerable: !0, value: e }), - 2 & t && "string" != typeof e) - ) - for (var i in e) - r.d( - n, - i, - function (t) { - return e[t] - }.bind(null, i) - ) - return n - }), - (r.n = function (e) { - var t = - e && e.__esModule - ? function () { - return e.default - } - : function () { - return e - } - return r.d(t, "a", t), t - }), - (r.o = function (e, t) { - return Object.prototype.hasOwnProperty.call(e, t) - }), - (r.p = ""), - r((r.s = 2)) -})([ - function (e, t, r) { - var n, i - /** - * lunr - http://lunrjs.com - A bit like Solr, but much smaller and not as bright - 2.3.9 - * Copyright (C) 2020 Oliver Nightingale - * @license MIT - */ !(function () { - var s, - o, - a, - u, - l, - c, - h, - d, - f, - p, - y, - m, - v, - g, - x, - w, - L, - E, - b, - S, - k, - Q, - O, - P, - T, - _, - C = function (e) { - var t = new C.Builder() - return ( - t.pipeline.add(C.trimmer, C.stopWordFilter, C.stemmer), - t.searchPipeline.add(C.stemmer), - e.call(t, t), - t.build() - ) - } - ;(C.version = "2.3.9"), - /*! - * lunr.utils - * Copyright (C) 2020 Oliver Nightingale - */ (C.utils = {}), - (C.utils.warn = - ((s = this), - function (e) { - s.console && console.warn && console.warn(e) - })), - (C.utils.asString = function (e) { - return null == e ? "" : e.toString() - }), - (C.utils.clone = function (e) { - if (null == e) return e - for (var t = Object.create(null), r = Object.keys(e), n = 0; n < r.length; n++) { - var i = r[n], - s = e[i] - if (Array.isArray(s)) t[i] = s.slice() - else { - if ("string" != typeof s && "number" != typeof s && "boolean" != typeof s) - throw new TypeError("clone is not deep and does not support nested objects") - t[i] = s - } - } - return t - }), - (C.FieldRef = function (e, t, r) { - ;(this.docRef = e), (this.fieldName = t), (this._stringValue = r) - }), - (C.FieldRef.joiner = "/"), - (C.FieldRef.fromString = function (e) { - var t = e.indexOf(C.FieldRef.joiner) - if (-1 === t) throw "malformed field ref string" - var r = e.slice(0, t), - n = e.slice(t + 1) - return new C.FieldRef(n, r, e) - }), - (C.FieldRef.prototype.toString = function () { - return ( - null == this._stringValue && - (this._stringValue = this.fieldName + C.FieldRef.joiner + this.docRef), - this._stringValue - ) - }), - /*! - * lunr.Set - * Copyright (C) 2020 Oliver Nightingale - */ (C.Set = function (e) { - if (((this.elements = Object.create(null)), e)) { - this.length = e.length - for (var t = 0; t < this.length; t++) this.elements[e[t]] = !0 - } else this.length = 0 - }), - (C.Set.complete = { - intersect: function (e) { - return e - }, - union: function () { - return this - }, - contains: function () { - return !0 - } - }), - (C.Set.empty = { - intersect: function () { - return this - }, - union: function (e) { - return e - }, - contains: function () { - return !1 - } - }), - (C.Set.prototype.contains = function (e) { - return !!this.elements[e] - }), - (C.Set.prototype.intersect = function (e) { - var t, - r, - n, - i = [] - if (e === C.Set.complete) return this - if (e === C.Set.empty) return e - this.length < e.length ? ((t = this), (r = e)) : ((t = e), (r = this)), - (n = Object.keys(t.elements)) - for (var s = 0; s < n.length; s++) { - var o = n[s] - o in r.elements && i.push(o) - } - return new C.Set(i) - }), - (C.Set.prototype.union = function (e) { - return e === C.Set.complete - ? C.Set.complete - : e === C.Set.empty - ? this - : new C.Set(Object.keys(this.elements).concat(Object.keys(e.elements))) - }), - (C.idf = function (e, t) { - var r = 0 - for (var n in e) "_index" != n && (r += Object.keys(e[n]).length) - var i = (t - r + 0.5) / (r + 0.5) - return Math.log(1 + Math.abs(i)) - }), - (C.Token = function (e, t) { - ;(this.str = e || ""), (this.metadata = t || {}) - }), - (C.Token.prototype.toString = function () { - return this.str - }), - (C.Token.prototype.update = function (e) { - return (this.str = e(this.str, this.metadata)), this - }), - (C.Token.prototype.clone = function (e) { - return ( - (e = - e || - function (e) { - return e - }), - new C.Token(e(this.str, this.metadata), this.metadata) - ) - }), - /*! - * lunr.tokenizer - * Copyright (C) 2020 Oliver Nightingale - */ (C.tokenizer = function (e, t) { - if (null == e || null == e) return [] - if (Array.isArray(e)) - return e.map(function (e) { - return new C.Token(C.utils.asString(e).toLowerCase(), C.utils.clone(t)) - }) - for ( - var r = e.toString().toLowerCase(), n = r.length, i = [], s = 0, o = 0; - s <= n; - s++ - ) { - var a = s - o - if (r.charAt(s).match(C.tokenizer.separator) || s == n) { - if (a > 0) { - var u = C.utils.clone(t) || {} - ;(u.position = [o, a]), (u.index = i.length), i.push(new C.Token(r.slice(o, s), u)) - } - o = s + 1 - } - } - return i - }), - (C.tokenizer.separator = /[\s\-]+/), - /*! - * lunr.Pipeline - * Copyright (C) 2020 Oliver Nightingale - */ (C.Pipeline = function () { - this._stack = [] - }), - (C.Pipeline.registeredFunctions = Object.create(null)), - (C.Pipeline.registerFunction = function (e, t) { - t in this.registeredFunctions && - C.utils.warn("Overwriting existing registered function: " + t), - (e.label = t), - (C.Pipeline.registeredFunctions[e.label] = e) - }), - (C.Pipeline.warnIfFunctionNotRegistered = function (e) { - ;(e.label && e.label in this.registeredFunctions) || - C.utils.warn( - "Function is not registered with pipeline. This may cause problems when serialising the index.\n", - e - ) - }), - (C.Pipeline.load = function (e) { - var t = new C.Pipeline() - return ( - e.forEach(function (e) { - var r = C.Pipeline.registeredFunctions[e] - if (!r) throw new Error("Cannot load unregistered function: " + e) - t.add(r) - }), - t - ) - }), - (C.Pipeline.prototype.add = function () { - var e = Array.prototype.slice.call(arguments) - e.forEach(function (e) { - C.Pipeline.warnIfFunctionNotRegistered(e), this._stack.push(e) - }, this) - }), - (C.Pipeline.prototype.after = function (e, t) { - C.Pipeline.warnIfFunctionNotRegistered(t) - var r = this._stack.indexOf(e) - if (-1 == r) throw new Error("Cannot find existingFn") - ;(r += 1), this._stack.splice(r, 0, t) - }), - (C.Pipeline.prototype.before = function (e, t) { - C.Pipeline.warnIfFunctionNotRegistered(t) - var r = this._stack.indexOf(e) - if (-1 == r) throw new Error("Cannot find existingFn") - this._stack.splice(r, 0, t) - }), - (C.Pipeline.prototype.remove = function (e) { - var t = this._stack.indexOf(e) - ;-1 != t && this._stack.splice(t, 1) - }), - (C.Pipeline.prototype.run = function (e) { - for (var t = this._stack.length, r = 0; r < t; r++) { - for (var n = this._stack[r], i = [], s = 0; s < e.length; s++) { - var o = n(e[s], s, e) - if (null != o && "" !== o) - if (Array.isArray(o)) for (var a = 0; a < o.length; a++) i.push(o[a]) - else i.push(o) - } - e = i - } - return e - }), - (C.Pipeline.prototype.runString = function (e, t) { - var r = new C.Token(e, t) - return this.run([r]).map(function (e) { - return e.toString() - }) - }), - (C.Pipeline.prototype.reset = function () { - this._stack = [] - }), - (C.Pipeline.prototype.toJSON = function () { - return this._stack.map(function (e) { - return C.Pipeline.warnIfFunctionNotRegistered(e), e.label - }) - }), - /*! - * lunr.Vector - * Copyright (C) 2020 Oliver Nightingale - */ (C.Vector = function (e) { - ;(this._magnitude = 0), (this.elements = e || []) - }), - (C.Vector.prototype.positionForIndex = function (e) { - if (0 == this.elements.length) return 0 - for ( - var t = 0, - r = this.elements.length / 2, - n = r - t, - i = Math.floor(n / 2), - s = this.elements[2 * i]; - n > 1 && (s < e && (t = i), s > e && (r = i), s != e); - - ) - (n = r - t), (i = t + Math.floor(n / 2)), (s = this.elements[2 * i]) - return s == e || s > e ? 2 * i : s < e ? 2 * (i + 1) : void 0 - }), - (C.Vector.prototype.insert = function (e, t) { - this.upsert(e, t, function () { - throw "duplicate index" - }) - }), - (C.Vector.prototype.upsert = function (e, t, r) { - this._magnitude = 0 - var n = this.positionForIndex(e) - this.elements[n] == e - ? (this.elements[n + 1] = r(this.elements[n + 1], t)) - : this.elements.splice(n, 0, e, t) - }), - (C.Vector.prototype.magnitude = function () { - if (this._magnitude) return this._magnitude - for (var e = 0, t = this.elements.length, r = 1; r < t; r += 2) { - var n = this.elements[r] - e += n * n - } - return (this._magnitude = Math.sqrt(e)) - }), - (C.Vector.prototype.dot = function (e) { - for ( - var t = 0, - r = this.elements, - n = e.elements, - i = r.length, - s = n.length, - o = 0, - a = 0, - u = 0, - l = 0; - u < i && l < s; - - ) - (o = r[u]) < (a = n[l]) - ? (u += 2) - : o > a - ? (l += 2) - : o == a && ((t += r[u + 1] * n[l + 1]), (u += 2), (l += 2)) - return t - }), - (C.Vector.prototype.similarity = function (e) { - return this.dot(e) / this.magnitude() || 0 - }), - (C.Vector.prototype.toArray = function () { - for ( - var e = new Array(this.elements.length / 2), t = 1, r = 0; - t < this.elements.length; - t += 2, r++ - ) - e[r] = this.elements[t] - return e - }), - (C.Vector.prototype.toJSON = function () { - return this.elements - }), - /*! - * lunr.stemmer - * Copyright (C) 2020 Oliver Nightingale - * Includes code from - http://tartarus.org/~martin/PorterStemmer/js.txt - */ (C.stemmer = - ((o = { - ational: "ate", - tional: "tion", - enci: "ence", - anci: "ance", - izer: "ize", - bli: "ble", - alli: "al", - entli: "ent", - eli: "e", - ousli: "ous", - ization: "ize", - ation: "ate", - ator: "ate", - alism: "al", - iveness: "ive", - fulness: "ful", - ousness: "ous", - aliti: "al", - iviti: "ive", - biliti: "ble", - logi: "log" - }), - (a = { icate: "ic", ative: "", alize: "al", iciti: "ic", ical: "ic", ful: "", ness: "" }), - (u = "[aeiouy]"), - (l = "[^aeiou][^aeiouy]*"), - (c = new RegExp("^([^aeiou][^aeiouy]*)?[aeiouy][aeiou]*[^aeiou][^aeiouy]*")), - (h = new RegExp( - "^([^aeiou][^aeiouy]*)?[aeiouy][aeiou]*[^aeiou][^aeiouy]*[aeiouy][aeiou]*[^aeiou][^aeiouy]*" - )), - (d = new RegExp( - "^([^aeiou][^aeiouy]*)?[aeiouy][aeiou]*[^aeiou][^aeiouy]*([aeiouy][aeiou]*)?$" - )), - (f = new RegExp("^([^aeiou][^aeiouy]*)?[aeiouy]")), - (p = /^(.+?)(ss|i)es$/), - (y = /^(.+?)([^s])s$/), - (m = /^(.+?)eed$/), - (v = /^(.+?)(ed|ing)$/), - (g = /.$/), - (x = /(at|bl|iz)$/), - (w = new RegExp("([^aeiouylsz])\\1$")), - (L = new RegExp("^" + l + u + "[^aeiouwxy]$")), - (E = /^(.+?[^aeiou])y$/), - (b = /^(.+?)(ational|tional|enci|anci|izer|bli|alli|entli|eli|ousli|ization|ation|ator|alism|iveness|fulness|ousness|aliti|iviti|biliti|logi)$/), - (S = /^(.+?)(icate|ative|alize|iciti|ical|ful|ness)$/), - (k = /^(.+?)(al|ance|ence|er|ic|able|ible|ant|ement|ment|ent|ou|ism|ate|iti|ous|ive|ize)$/), - (Q = /^(.+?)(s|t)(ion)$/), - (O = /^(.+?)e$/), - (P = /ll$/), - (T = new RegExp("^" + l + u + "[^aeiouwxy]$")), - (_ = function (e) { - var t, r, n, i, s, u, l - if (e.length < 3) return e - if ( - ("y" == (n = e.substr(0, 1)) && (e = n.toUpperCase() + e.substr(1)), - (s = y), - (i = p).test(e) - ? (e = e.replace(i, "$1$2")) - : s.test(e) && (e = e.replace(s, "$1$2")), - (s = v), - (i = m).test(e)) - ) { - var _ = i.exec(e) - ;(i = c).test(_[1]) && ((i = g), (e = e.replace(i, ""))) - } else - s.test(e) && - ((t = (_ = s.exec(e))[1]), - (s = f).test(t) && - ((u = w), - (l = L), - (s = x).test((e = t)) - ? (e += "e") - : u.test(e) - ? ((i = g), (e = e.replace(i, ""))) - : l.test(e) && (e += "e"))) - return ( - (i = E).test(e) && (e = (t = (_ = i.exec(e))[1]) + "i"), - (i = b).test(e) && - ((t = (_ = i.exec(e))[1]), (r = _[2]), (i = c).test(t) && (e = t + o[r])), - (i = S).test(e) && - ((t = (_ = i.exec(e))[1]), (r = _[2]), (i = c).test(t) && (e = t + a[r])), - (s = Q), - (i = k).test(e) - ? ((t = (_ = i.exec(e))[1]), (i = h).test(t) && (e = t)) - : s.test(e) && ((t = (_ = s.exec(e))[1] + _[2]), (s = h).test(t) && (e = t)), - (i = O).test(e) && - ((t = (_ = i.exec(e))[1]), - (s = d), - (u = T), - ((i = h).test(t) || (s.test(t) && !u.test(t))) && (e = t)), - (s = h), - (i = P).test(e) && s.test(e) && ((i = g), (e = e.replace(i, ""))), - "y" == n && (e = n.toLowerCase() + e.substr(1)), - e - ) - }), - function (e) { - return e.update(_) - })), - C.Pipeline.registerFunction(C.stemmer, "stemmer"), - /*! - * lunr.stopWordFilter - * Copyright (C) 2020 Oliver Nightingale - */ (C.generateStopWordFilter = function (e) { - var t = e.reduce(function (e, t) { - return (e[t] = t), e - }, {}) - return function (e) { - if (e && t[e.toString()] !== e.toString()) return e - } - }), - (C.stopWordFilter = C.generateStopWordFilter([ - "a", - "able", - "about", - "across", - "after", - "all", - "almost", - "also", - "am", - "among", - "an", - "and", - "any", - "are", - "as", - "at", - "be", - "because", - "been", - "but", - "by", - "can", - "cannot", - "could", - "dear", - "did", - "do", - "does", - "either", - "else", - "ever", - "every", - "for", - "from", - "get", - "got", - "had", - "has", - "have", - "he", - "her", - "hers", - "him", - "his", - "how", - "however", - "i", - "if", - "in", - "into", - "is", - "it", - "its", - "just", - "least", - "let", - "like", - "likely", - "may", - "me", - "might", - "most", - "must", - "my", - "neither", - "no", - "nor", - "not", - "of", - "off", - "often", - "on", - "only", - "or", - "other", - "our", - "own", - "rather", - "said", - "say", - "says", - "she", - "should", - "since", - "so", - "some", - "than", - "that", - "the", - "their", - "them", - "then", - "there", - "these", - "they", - "this", - "tis", - "to", - "too", - "twas", - "us", - "wants", - "was", - "we", - "were", - "what", - "when", - "where", - "which", - "while", - "who", - "whom", - "why", - "will", - "with", - "would", - "yet", - "you", - "your" - ])), - C.Pipeline.registerFunction(C.stopWordFilter, "stopWordFilter"), - /*! - * lunr.trimmer - * Copyright (C) 2020 Oliver Nightingale - */ (C.trimmer = function (e) { - return e.update(function (e) { - return e.replace(/^\W+/, "").replace(/\W+$/, "") - }) - }), - C.Pipeline.registerFunction(C.trimmer, "trimmer"), - /*! - * lunr.TokenSet - * Copyright (C) 2020 Oliver Nightingale - */ (C.TokenSet = function () { - ;(this.final = !1), - (this.edges = {}), - (this.id = C.TokenSet._nextId), - (C.TokenSet._nextId += 1) - }), - (C.TokenSet._nextId = 1), - (C.TokenSet.fromArray = function (e) { - for (var t = new C.TokenSet.Builder(), r = 0, n = e.length; r < n; r++) t.insert(e[r]) - return t.finish(), t.root - }), - (C.TokenSet.fromClause = function (e) { - return "editDistance" in e - ? C.TokenSet.fromFuzzyString(e.term, e.editDistance) - : C.TokenSet.fromString(e.term) - }), - (C.TokenSet.fromFuzzyString = function (e, t) { - for (var r = new C.TokenSet(), n = [{ node: r, editsRemaining: t, str: e }]; n.length; ) { - var i = n.pop() - if (i.str.length > 0) { - var s, - o = i.str.charAt(0) - o in i.node.edges - ? (s = i.node.edges[o]) - : ((s = new C.TokenSet()), (i.node.edges[o] = s)), - 1 == i.str.length && (s.final = !0), - n.push({ node: s, editsRemaining: i.editsRemaining, str: i.str.slice(1) }) - } - if (0 != i.editsRemaining) { - if ("*" in i.node.edges) var a = i.node.edges["*"] - else { - a = new C.TokenSet() - i.node.edges["*"] = a - } - if ( - (0 == i.str.length && (a.final = !0), - n.push({ node: a, editsRemaining: i.editsRemaining - 1, str: i.str }), - i.str.length > 1 && - n.push({ - node: i.node, - editsRemaining: i.editsRemaining - 1, - str: i.str.slice(1) - }), - 1 == i.str.length && (i.node.final = !0), - i.str.length >= 1) - ) { - if ("*" in i.node.edges) var u = i.node.edges["*"] - else { - u = new C.TokenSet() - i.node.edges["*"] = u - } - 1 == i.str.length && (u.final = !0), - n.push({ node: u, editsRemaining: i.editsRemaining - 1, str: i.str.slice(1) }) - } - if (i.str.length > 1) { - var l, - c = i.str.charAt(0), - h = i.str.charAt(1) - h in i.node.edges - ? (l = i.node.edges[h]) - : ((l = new C.TokenSet()), (i.node.edges[h] = l)), - 1 == i.str.length && (l.final = !0), - n.push({ node: l, editsRemaining: i.editsRemaining - 1, str: c + i.str.slice(2) }) - } - } - } - return r - }), - (C.TokenSet.fromString = function (e) { - for (var t = new C.TokenSet(), r = t, n = 0, i = e.length; n < i; n++) { - var s = e[n], - o = n == i - 1 - if ("*" == s) (t.edges[s] = t), (t.final = o) - else { - var a = new C.TokenSet() - ;(a.final = o), (t.edges[s] = a), (t = a) - } - } - return r - }), - (C.TokenSet.prototype.toArray = function () { - for (var e = [], t = [{ prefix: "", node: this }]; t.length; ) { - var r = t.pop(), - n = Object.keys(r.node.edges), - i = n.length - r.node.final && (r.prefix.charAt(0), e.push(r.prefix)) - for (var s = 0; s < i; s++) { - var o = n[s] - t.push({ prefix: r.prefix.concat(o), node: r.node.edges[o] }) - } - } - return e - }), - (C.TokenSet.prototype.toString = function () { - if (this._str) return this._str - for ( - var e = this.final ? "1" : "0", t = Object.keys(this.edges).sort(), r = t.length, n = 0; - n < r; - n++ - ) { - var i = t[n] - e = e + i + this.edges[i].id - } - return e - }), - (C.TokenSet.prototype.intersect = function (e) { - for ( - var t = new C.TokenSet(), r = void 0, n = [{ qNode: e, output: t, node: this }]; - n.length; - - ) { - r = n.pop() - for ( - var i = Object.keys(r.qNode.edges), - s = i.length, - o = Object.keys(r.node.edges), - a = o.length, - u = 0; - u < s; - u++ - ) - for (var l = i[u], c = 0; c < a; c++) { - var h = o[c] - if (h == l || "*" == l) { - var d = r.node.edges[h], - f = r.qNode.edges[l], - p = d.final && f.final, - y = void 0 - h in r.output.edges - ? ((y = r.output.edges[h]).final = y.final || p) - : (((y = new C.TokenSet()).final = p), (r.output.edges[h] = y)), - n.push({ qNode: f, output: y, node: d }) - } - } - } - return t - }), - (C.TokenSet.Builder = function () { - ;(this.previousWord = ""), - (this.root = new C.TokenSet()), - (this.uncheckedNodes = []), - (this.minimizedNodes = {}) - }), - (C.TokenSet.Builder.prototype.insert = function (e) { - var t, - r = 0 - if (e < this.previousWord) throw new Error("Out of order word insertion") - for ( - var n = 0; - n < e.length && n < this.previousWord.length && e[n] == this.previousWord[n]; - n++ - ) - r++ - this.minimize(r), - (t = - 0 == this.uncheckedNodes.length - ? this.root - : this.uncheckedNodes[this.uncheckedNodes.length - 1].child) - for (n = r; n < e.length; n++) { - var i = new C.TokenSet(), - s = e[n] - ;(t.edges[s] = i), this.uncheckedNodes.push({ parent: t, char: s, child: i }), (t = i) - } - ;(t.final = !0), (this.previousWord = e) - }), - (C.TokenSet.Builder.prototype.finish = function () { - this.minimize(0) - }), - (C.TokenSet.Builder.prototype.minimize = function (e) { - for (var t = this.uncheckedNodes.length - 1; t >= e; t--) { - var r = this.uncheckedNodes[t], - n = r.child.toString() - n in this.minimizedNodes - ? (r.parent.edges[r.char] = this.minimizedNodes[n]) - : ((r.child._str = n), (this.minimizedNodes[n] = r.child)), - this.uncheckedNodes.pop() - } - }), - /*! - * lunr.Index - * Copyright (C) 2020 Oliver Nightingale - */ (C.Index = function (e) { - ;(this.invertedIndex = e.invertedIndex), - (this.fieldVectors = e.fieldVectors), - (this.tokenSet = e.tokenSet), - (this.fields = e.fields), - (this.pipeline = e.pipeline) - }), - (C.Index.prototype.search = function (e) { - return this.query(function (t) { - new C.QueryParser(e, t).parse() - }) - }), - (C.Index.prototype.query = function (e) { - for ( - var t = new C.Query(this.fields), - r = Object.create(null), - n = Object.create(null), - i = Object.create(null), - s = Object.create(null), - o = Object.create(null), - a = 0; - a < this.fields.length; - a++ - ) - n[this.fields[a]] = new C.Vector() - e.call(t, t) - for (a = 0; a < t.clauses.length; a++) { - var u = t.clauses[a], - l = null, - c = C.Set.empty - l = u.usePipeline ? this.pipeline.runString(u.term, { fields: u.fields }) : [u.term] - for (var h = 0; h < l.length; h++) { - var d = l[h] - u.term = d - var f = C.TokenSet.fromClause(u), - p = this.tokenSet.intersect(f).toArray() - if (0 === p.length && u.presence === C.Query.presence.REQUIRED) { - for (var y = 0; y < u.fields.length; y++) { - s[(R = u.fields[y])] = C.Set.empty - } - break - } - for (var m = 0; m < p.length; m++) { - var v = p[m], - g = this.invertedIndex[v], - x = g._index - for (y = 0; y < u.fields.length; y++) { - var w = g[(R = u.fields[y])], - L = Object.keys(w), - E = v + "/" + R, - b = new C.Set(L) - if ( - (u.presence == C.Query.presence.REQUIRED && - ((c = c.union(b)), void 0 === s[R] && (s[R] = C.Set.complete)), - u.presence != C.Query.presence.PROHIBITED) - ) { - if ( - (n[R].upsert(x, u.boost, function (e, t) { - return e + t - }), - !i[E]) - ) { - for (var S = 0; S < L.length; S++) { - var k, - Q = L[S], - O = new C.FieldRef(Q, R), - P = w[Q] - void 0 === (k = r[O]) ? (r[O] = new C.MatchData(v, R, P)) : k.add(v, R, P) - } - i[E] = !0 - } - } else void 0 === o[R] && (o[R] = C.Set.empty), (o[R] = o[R].union(b)) - } - } - } - if (u.presence === C.Query.presence.REQUIRED) - for (y = 0; y < u.fields.length; y++) { - s[(R = u.fields[y])] = s[R].intersect(c) - } - } - var T = C.Set.complete, - _ = C.Set.empty - for (a = 0; a < this.fields.length; a++) { - var R - s[(R = this.fields[a])] && (T = T.intersect(s[R])), o[R] && (_ = _.union(o[R])) - } - var I = Object.keys(r), - j = [], - F = Object.create(null) - if (t.isNegated()) { - I = Object.keys(this.fieldVectors) - for (a = 0; a < I.length; a++) { - O = I[a] - var N = C.FieldRef.fromString(O) - r[O] = new C.MatchData() - } - } - for (a = 0; a < I.length; a++) { - var D = (N = C.FieldRef.fromString(I[a])).docRef - if (T.contains(D) && !_.contains(D)) { - var A, - z = this.fieldVectors[N], - q = n[N.fieldName].similarity(z) - if (void 0 !== (A = F[D])) (A.score += q), A.matchData.combine(r[N]) - else { - var V = { ref: D, score: q, matchData: r[N] } - ;(F[D] = V), j.push(V) - } - } - } - return j.sort(function (e, t) { - return t.score - e.score - }) - }), - (C.Index.prototype.toJSON = function () { - var e = Object.keys(this.invertedIndex) - .sort() - .map(function (e) { - return [e, this.invertedIndex[e]] - }, this), - t = Object.keys(this.fieldVectors).map(function (e) { - return [e, this.fieldVectors[e].toJSON()] - }, this) - return { - version: C.version, - fields: this.fields, - fieldVectors: t, - invertedIndex: e, - pipeline: this.pipeline.toJSON() - } - }), - (C.Index.load = function (e) { - var t = {}, - r = {}, - n = e.fieldVectors, - i = Object.create(null), - s = e.invertedIndex, - o = new C.TokenSet.Builder(), - a = C.Pipeline.load(e.pipeline) - e.version != C.version && - C.utils.warn( - "Version mismatch when loading serialised index. Current version of lunr '" + - C.version + - "' does not match serialized index '" + - e.version + - "'" - ) - for (var u = 0; u < n.length; u++) { - var l = (h = n[u])[0], - c = h[1] - r[l] = new C.Vector(c) - } - for (u = 0; u < s.length; u++) { - var h, - d = (h = s[u])[0], - f = h[1] - o.insert(d), (i[d] = f) - } - return ( - o.finish(), - (t.fields = e.fields), - (t.fieldVectors = r), - (t.invertedIndex = i), - (t.tokenSet = o.root), - (t.pipeline = a), - new C.Index(t) - ) - }), - /*! - * lunr.Builder - * Copyright (C) 2020 Oliver Nightingale - */ (C.Builder = function () { - ;(this._ref = "id"), - (this._fields = Object.create(null)), - (this._documents = Object.create(null)), - (this.invertedIndex = Object.create(null)), - (this.fieldTermFrequencies = {}), - (this.fieldLengths = {}), - (this.tokenizer = C.tokenizer), - (this.pipeline = new C.Pipeline()), - (this.searchPipeline = new C.Pipeline()), - (this.documentCount = 0), - (this._b = 0.75), - (this._k1 = 1.2), - (this.termIndex = 0), - (this.metadataWhitelist = []) - }), - (C.Builder.prototype.ref = function (e) { - this._ref = e - }), - (C.Builder.prototype.field = function (e, t) { - if (/\//.test(e)) throw new RangeError("Field '" + e + "' contains illegal character '/'") - this._fields[e] = t || {} - }), - (C.Builder.prototype.b = function (e) { - this._b = e < 0 ? 0 : e > 1 ? 1 : e - }), - (C.Builder.prototype.k1 = function (e) { - this._k1 = e - }), - (C.Builder.prototype.add = function (e, t) { - var r = e[this._ref], - n = Object.keys(this._fields) - ;(this._documents[r] = t || {}), (this.documentCount += 1) - for (var i = 0; i < n.length; i++) { - var s = n[i], - o = this._fields[s].extractor, - a = o ? o(e) : e[s], - u = this.tokenizer(a, { fields: [s] }), - l = this.pipeline.run(u), - c = new C.FieldRef(r, s), - h = Object.create(null) - ;(this.fieldTermFrequencies[c] = h), - (this.fieldLengths[c] = 0), - (this.fieldLengths[c] += l.length) - for (var d = 0; d < l.length; d++) { - var f = l[d] - if ((null == h[f] && (h[f] = 0), (h[f] += 1), null == this.invertedIndex[f])) { - var p = Object.create(null) - ;(p._index = this.termIndex), (this.termIndex += 1) - for (var y = 0; y < n.length; y++) p[n[y]] = Object.create(null) - this.invertedIndex[f] = p - } - null == this.invertedIndex[f][s][r] && - (this.invertedIndex[f][s][r] = Object.create(null)) - for (var m = 0; m < this.metadataWhitelist.length; m++) { - var v = this.metadataWhitelist[m], - g = f.metadata[v] - null == this.invertedIndex[f][s][r][v] && (this.invertedIndex[f][s][r][v] = []), - this.invertedIndex[f][s][r][v].push(g) - } - } - } - }), - (C.Builder.prototype.calculateAverageFieldLengths = function () { - for ( - var e = Object.keys(this.fieldLengths), t = e.length, r = {}, n = {}, i = 0; - i < t; - i++ - ) { - var s = C.FieldRef.fromString(e[i]), - o = s.fieldName - n[o] || (n[o] = 0), (n[o] += 1), r[o] || (r[o] = 0), (r[o] += this.fieldLengths[s]) - } - var a = Object.keys(this._fields) - for (i = 0; i < a.length; i++) { - var u = a[i] - r[u] = r[u] / n[u] - } - this.averageFieldLength = r - }), - (C.Builder.prototype.createFieldVectors = function () { - for ( - var e = {}, - t = Object.keys(this.fieldTermFrequencies), - r = t.length, - n = Object.create(null), - i = 0; - i < r; - i++ - ) { - for ( - var s = C.FieldRef.fromString(t[i]), - o = s.fieldName, - a = this.fieldLengths[s], - u = new C.Vector(), - l = this.fieldTermFrequencies[s], - c = Object.keys(l), - h = c.length, - d = this._fields[o].boost || 1, - f = this._documents[s.docRef].boost || 1, - p = 0; - p < h; - p++ - ) { - var y, - m, - v, - g = c[p], - x = l[g], - w = this.invertedIndex[g]._index - void 0 === n[g] - ? ((y = C.idf(this.invertedIndex[g], this.documentCount)), (n[g] = y)) - : (y = n[g]), - (m = - (y * ((this._k1 + 1) * x)) / - (this._k1 * (1 - this._b + this._b * (a / this.averageFieldLength[o])) + x)), - (m *= d), - (m *= f), - (v = Math.round(1e3 * m) / 1e3), - u.insert(w, v) - } - e[s] = u - } - this.fieldVectors = e - }), - (C.Builder.prototype.createTokenSet = function () { - this.tokenSet = C.TokenSet.fromArray(Object.keys(this.invertedIndex).sort()) - }), - (C.Builder.prototype.build = function () { - return ( - this.calculateAverageFieldLengths(), - this.createFieldVectors(), - this.createTokenSet(), - new C.Index({ - invertedIndex: this.invertedIndex, - fieldVectors: this.fieldVectors, - tokenSet: this.tokenSet, - fields: Object.keys(this._fields), - pipeline: this.searchPipeline - }) - ) - }), - (C.Builder.prototype.use = function (e) { - var t = Array.prototype.slice.call(arguments, 1) - t.unshift(this), e.apply(this, t) - }), - (C.MatchData = function (e, t, r) { - for (var n = Object.create(null), i = Object.keys(r || {}), s = 0; s < i.length; s++) { - var o = i[s] - n[o] = r[o].slice() - } - ;(this.metadata = Object.create(null)), - void 0 !== e && ((this.metadata[e] = Object.create(null)), (this.metadata[e][t] = n)) - }), - (C.MatchData.prototype.combine = function (e) { - for (var t = Object.keys(e.metadata), r = 0; r < t.length; r++) { - var n = t[r], - i = Object.keys(e.metadata[n]) - null == this.metadata[n] && (this.metadata[n] = Object.create(null)) - for (var s = 0; s < i.length; s++) { - var o = i[s], - a = Object.keys(e.metadata[n][o]) - null == this.metadata[n][o] && (this.metadata[n][o] = Object.create(null)) - for (var u = 0; u < a.length; u++) { - var l = a[u] - null == this.metadata[n][o][l] - ? (this.metadata[n][o][l] = e.metadata[n][o][l]) - : (this.metadata[n][o][l] = this.metadata[n][o][l].concat(e.metadata[n][o][l])) - } - } - } - }), - (C.MatchData.prototype.add = function (e, t, r) { - if (!(e in this.metadata)) - return (this.metadata[e] = Object.create(null)), void (this.metadata[e][t] = r) - if (t in this.metadata[e]) - for (var n = Object.keys(r), i = 0; i < n.length; i++) { - var s = n[i] - s in this.metadata[e][t] - ? (this.metadata[e][t][s] = this.metadata[e][t][s].concat(r[s])) - : (this.metadata[e][t][s] = r[s]) - } - else this.metadata[e][t] = r - }), - (C.Query = function (e) { - ;(this.clauses = []), (this.allFields = e) - }), - (C.Query.wildcard = new String("*")), - (C.Query.wildcard.NONE = 0), - (C.Query.wildcard.LEADING = 1), - (C.Query.wildcard.TRAILING = 2), - (C.Query.presence = { OPTIONAL: 1, REQUIRED: 2, PROHIBITED: 3 }), - (C.Query.prototype.clause = function (e) { - return ( - "fields" in e || (e.fields = this.allFields), - "boost" in e || (e.boost = 1), - "usePipeline" in e || (e.usePipeline = !0), - "wildcard" in e || (e.wildcard = C.Query.wildcard.NONE), - e.wildcard & C.Query.wildcard.LEADING && - e.term.charAt(0) != C.Query.wildcard && - (e.term = "*" + e.term), - e.wildcard & C.Query.wildcard.TRAILING && - e.term.slice(-1) != C.Query.wildcard && - (e.term = e.term + "*"), - "presence" in e || (e.presence = C.Query.presence.OPTIONAL), - this.clauses.push(e), - this - ) - }), - (C.Query.prototype.isNegated = function () { - for (var e = 0; e < this.clauses.length; e++) - if (this.clauses[e].presence != C.Query.presence.PROHIBITED) return !1 - return !0 - }), - (C.Query.prototype.term = function (e, t) { - if (Array.isArray(e)) - return ( - e.forEach(function (e) { - this.term(e, C.utils.clone(t)) - }, this), - this - ) - var r = t || {} - return (r.term = e.toString()), this.clause(r), this - }), - (C.QueryParseError = function (e, t, r) { - ;(this.name = "QueryParseError"), (this.message = e), (this.start = t), (this.end = r) - }), - (C.QueryParseError.prototype = new Error()), - (C.QueryLexer = function (e) { - ;(this.lexemes = []), - (this.str = e), - (this.length = e.length), - (this.pos = 0), - (this.start = 0), - (this.escapeCharPositions = []) - }), - (C.QueryLexer.prototype.run = function () { - for (var e = C.QueryLexer.lexText; e; ) e = e(this) - }), - (C.QueryLexer.prototype.sliceString = function () { - for ( - var e = [], t = this.start, r = this.pos, n = 0; - n < this.escapeCharPositions.length; - n++ - ) - (r = this.escapeCharPositions[n]), e.push(this.str.slice(t, r)), (t = r + 1) - return ( - e.push(this.str.slice(t, this.pos)), (this.escapeCharPositions.length = 0), e.join("") - ) - }), - (C.QueryLexer.prototype.emit = function (e) { - this.lexemes.push({ type: e, str: this.sliceString(), start: this.start, end: this.pos }), - (this.start = this.pos) - }), - (C.QueryLexer.prototype.escapeCharacter = function () { - this.escapeCharPositions.push(this.pos - 1), (this.pos += 1) - }), - (C.QueryLexer.prototype.next = function () { - if (this.pos >= this.length) return C.QueryLexer.EOS - var e = this.str.charAt(this.pos) - return (this.pos += 1), e - }), - (C.QueryLexer.prototype.width = function () { - return this.pos - this.start - }), - (C.QueryLexer.prototype.ignore = function () { - this.start == this.pos && (this.pos += 1), (this.start = this.pos) - }), - (C.QueryLexer.prototype.backup = function () { - this.pos -= 1 - }), - (C.QueryLexer.prototype.acceptDigitRun = function () { - var e, t - do { - t = (e = this.next()).charCodeAt(0) - } while (t > 47 && t < 58) - e != C.QueryLexer.EOS && this.backup() - }), - (C.QueryLexer.prototype.more = function () { - return this.pos < this.length - }), - (C.QueryLexer.EOS = "EOS"), - (C.QueryLexer.FIELD = "FIELD"), - (C.QueryLexer.TERM = "TERM"), - (C.QueryLexer.EDIT_DISTANCE = "EDIT_DISTANCE"), - (C.QueryLexer.BOOST = "BOOST"), - (C.QueryLexer.PRESENCE = "PRESENCE"), - (C.QueryLexer.lexField = function (e) { - return e.backup(), e.emit(C.QueryLexer.FIELD), e.ignore(), C.QueryLexer.lexText - }), - (C.QueryLexer.lexTerm = function (e) { - if ((e.width() > 1 && (e.backup(), e.emit(C.QueryLexer.TERM)), e.ignore(), e.more())) - return C.QueryLexer.lexText - }), - (C.QueryLexer.lexEditDistance = function (e) { - return ( - e.ignore(), e.acceptDigitRun(), e.emit(C.QueryLexer.EDIT_DISTANCE), C.QueryLexer.lexText - ) - }), - (C.QueryLexer.lexBoost = function (e) { - return e.ignore(), e.acceptDigitRun(), e.emit(C.QueryLexer.BOOST), C.QueryLexer.lexText - }), - (C.QueryLexer.lexEOS = function (e) { - e.width() > 0 && e.emit(C.QueryLexer.TERM) - }), - (C.QueryLexer.termSeparator = C.tokenizer.separator), - (C.QueryLexer.lexText = function (e) { - for (;;) { - var t = e.next() - if (t == C.QueryLexer.EOS) return C.QueryLexer.lexEOS - if (92 != t.charCodeAt(0)) { - if (":" == t) return C.QueryLexer.lexField - if ("~" == t) - return ( - e.backup(), - e.width() > 0 && e.emit(C.QueryLexer.TERM), - C.QueryLexer.lexEditDistance - ) - if ("^" == t) - return e.backup(), e.width() > 0 && e.emit(C.QueryLexer.TERM), C.QueryLexer.lexBoost - if ("+" == t && 1 === e.width()) - return e.emit(C.QueryLexer.PRESENCE), C.QueryLexer.lexText - if ("-" == t && 1 === e.width()) - return e.emit(C.QueryLexer.PRESENCE), C.QueryLexer.lexText - if (t.match(C.QueryLexer.termSeparator)) return C.QueryLexer.lexTerm - } else e.escapeCharacter() - } - }), - (C.QueryParser = function (e, t) { - ;(this.lexer = new C.QueryLexer(e)), - (this.query = t), - (this.currentClause = {}), - (this.lexemeIdx = 0) - }), - (C.QueryParser.prototype.parse = function () { - this.lexer.run(), (this.lexemes = this.lexer.lexemes) - for (var e = C.QueryParser.parseClause; e; ) e = e(this) - return this.query - }), - (C.QueryParser.prototype.peekLexeme = function () { - return this.lexemes[this.lexemeIdx] - }), - (C.QueryParser.prototype.consumeLexeme = function () { - var e = this.peekLexeme() - return (this.lexemeIdx += 1), e - }), - (C.QueryParser.prototype.nextClause = function () { - var e = this.currentClause - this.query.clause(e), (this.currentClause = {}) - }), - (C.QueryParser.parseClause = function (e) { - var t = e.peekLexeme() - if (null != t) - switch (t.type) { - case C.QueryLexer.PRESENCE: - return C.QueryParser.parsePresence - case C.QueryLexer.FIELD: - return C.QueryParser.parseField - case C.QueryLexer.TERM: - return C.QueryParser.parseTerm - default: - var r = "expected either a field or a term, found " + t.type - throw ( - (t.str.length >= 1 && (r += " with value '" + t.str + "'"), - new C.QueryParseError(r, t.start, t.end)) - ) - } - }), - (C.QueryParser.parsePresence = function (e) { - var t = e.consumeLexeme() - if (null != t) { - switch (t.str) { - case "-": - e.currentClause.presence = C.Query.presence.PROHIBITED - break - case "+": - e.currentClause.presence = C.Query.presence.REQUIRED - break - default: - var r = "unrecognised presence operator'" + t.str + "'" - throw new C.QueryParseError(r, t.start, t.end) - } - var n = e.peekLexeme() - if (null == n) { - r = "expecting term or field, found nothing" - throw new C.QueryParseError(r, t.start, t.end) - } - switch (n.type) { - case C.QueryLexer.FIELD: - return C.QueryParser.parseField - case C.QueryLexer.TERM: - return C.QueryParser.parseTerm - default: - r = "expecting term or field, found '" + n.type + "'" - throw new C.QueryParseError(r, n.start, n.end) - } - } - }), - (C.QueryParser.parseField = function (e) { - var t = e.consumeLexeme() - if (null != t) { - if (-1 == e.query.allFields.indexOf(t.str)) { - var r = e.query.allFields - .map(function (e) { - return "'" + e + "'" - }) - .join(", "), - n = "unrecognised field '" + t.str + "', possible fields: " + r - throw new C.QueryParseError(n, t.start, t.end) - } - e.currentClause.fields = [t.str] - var i = e.peekLexeme() - if (null == i) { - n = "expecting term, found nothing" - throw new C.QueryParseError(n, t.start, t.end) - } - switch (i.type) { - case C.QueryLexer.TERM: - return C.QueryParser.parseTerm - default: - n = "expecting term, found '" + i.type + "'" - throw new C.QueryParseError(n, i.start, i.end) - } - } - }), - (C.QueryParser.parseTerm = function (e) { - var t = e.consumeLexeme() - if (null != t) { - ;(e.currentClause.term = t.str.toLowerCase()), - -1 != t.str.indexOf("*") && (e.currentClause.usePipeline = !1) - var r = e.peekLexeme() - if (null != r) - switch (r.type) { - case C.QueryLexer.TERM: - return e.nextClause(), C.QueryParser.parseTerm - case C.QueryLexer.FIELD: - return e.nextClause(), C.QueryParser.parseField - case C.QueryLexer.EDIT_DISTANCE: - return C.QueryParser.parseEditDistance - case C.QueryLexer.BOOST: - return C.QueryParser.parseBoost - case C.QueryLexer.PRESENCE: - return e.nextClause(), C.QueryParser.parsePresence - default: - var n = "Unexpected lexeme type '" + r.type + "'" - throw new C.QueryParseError(n, r.start, r.end) - } - else e.nextClause() - } - }), - (C.QueryParser.parseEditDistance = function (e) { - var t = e.consumeLexeme() - if (null != t) { - var r = parseInt(t.str, 10) - if (isNaN(r)) { - var n = "edit distance must be numeric" - throw new C.QueryParseError(n, t.start, t.end) - } - e.currentClause.editDistance = r - var i = e.peekLexeme() - if (null != i) - switch (i.type) { - case C.QueryLexer.TERM: - return e.nextClause(), C.QueryParser.parseTerm - case C.QueryLexer.FIELD: - return e.nextClause(), C.QueryParser.parseField - case C.QueryLexer.EDIT_DISTANCE: - return C.QueryParser.parseEditDistance - case C.QueryLexer.BOOST: - return C.QueryParser.parseBoost - case C.QueryLexer.PRESENCE: - return e.nextClause(), C.QueryParser.parsePresence - default: - n = "Unexpected lexeme type '" + i.type + "'" - throw new C.QueryParseError(n, i.start, i.end) - } - else e.nextClause() - } - }), - (C.QueryParser.parseBoost = function (e) { - var t = e.consumeLexeme() - if (null != t) { - var r = parseInt(t.str, 10) - if (isNaN(r)) { - var n = "boost must be numeric" - throw new C.QueryParseError(n, t.start, t.end) - } - e.currentClause.boost = r - var i = e.peekLexeme() - if (null != i) - switch (i.type) { - case C.QueryLexer.TERM: - return e.nextClause(), C.QueryParser.parseTerm - case C.QueryLexer.FIELD: - return e.nextClause(), C.QueryParser.parseField - case C.QueryLexer.EDIT_DISTANCE: - return C.QueryParser.parseEditDistance - case C.QueryLexer.BOOST: - return C.QueryParser.parseBoost - case C.QueryLexer.PRESENCE: - return e.nextClause(), C.QueryParser.parsePresence - default: - n = "Unexpected lexeme type '" + i.type + "'" - throw new C.QueryParseError(n, i.start, i.end) - } - else e.nextClause() - } - }), - void 0 === - (i = - "function" == - typeof (n = function () { - return C - }) - ? n.call(t, r, t, e) - : n) || (e.exports = i) - })() - }, - function (e, t, r) {}, - function (e, t, r) { - "use strict" - r.r(t) - var n = [] - function i(e, t) { - n.push({ selector: t, constructor: e }) - } - var s, - o, - a = (function () { - function e() { - this.createComponents(document.body) - } - return ( - (e.prototype.createComponents = function (e) { - n.forEach(function (t) { - e.querySelectorAll(t.selector).forEach(function (e) { - e.dataset.hasInstance || - (new t.constructor({ el: e }), (e.dataset.hasInstance = String(!0))) - }) - }) - }), - e - ) - })(), - u = function (e) { - this.el = e.el - }, - l = r(0), - c = - ((s = function (e, t) { - return (s = - Object.setPrototypeOf || - ({ __proto__: [] } instanceof Array && - function (e, t) { - e.__proto__ = t - }) || - function (e, t) { - for (var r in t) Object.prototype.hasOwnProperty.call(t, r) && (e[r] = t[r]) - })(e, t) - }), - function (e, t) { - function r() { - this.constructor = e - } - s(e, t), - (e.prototype = null === t ? Object.create(t) : ((r.prototype = t.prototype), new r())) - }) - !(function (e) { - ;(e[(e.Idle = 0)] = "Idle"), - (e[(e.Loading = 1)] = "Loading"), - (e[(e.Ready = 2)] = "Ready"), - (e[(e.Failure = 3)] = "Failure") - })(o || (o = {})) - var h = (function (e) { - function t(t) { - var r = e.call(this, t) || this - ;(r.query = ""), - (r.loadingState = o.Idle), - (r.hasFocus = !1), - (r.preventPress = !1), - (r.data = null), - (r.index = null), - (r.resultClicked = !1) - var n = document.querySelector("#tsd-search-field"), - i = document.querySelector(".results") - if (!n || !i) throw new Error("The input field or the result list wrapper are not found") - return ( - (r.field = n), (r.results = i), (r.base = r.el.dataset.base + "/"), r.bindEvents(), r - ) - } - return ( - c(t, e), - (t.prototype.loadIndex = function () { - var e = this - if (this.loadingState == o.Idle && !this.data) { - setTimeout(function () { - e.loadingState == o.Idle && e.setLoadingState(o.Loading) - }, 500) - var t = this.el.dataset.index - t - ? fetch(t) - .then(function (e) { - if (!e.ok) throw new Error("The search index is missing") - return e.json() - }) - .then(function (t) { - ;(e.data = t), (e.index = l.Index.load(t.index)), e.setLoadingState(o.Ready) - }) - .catch(function (t) { - console.error(t), e.setLoadingState(o.Failure) - }) - : this.setLoadingState(o.Failure) - } - }), - (t.prototype.updateResults = function () { - if ( - this.loadingState == o.Ready && - ((this.results.textContent = ""), this.query && this.index && this.data) - ) { - var e = this.index.search("*" + this.query + "*") - 0 === e.length && (e = this.index.search("*" + this.query + "~1*")) - for (var t = 0, r = Math.min(10, e.length); t < r; t++) { - var n = this.data.rows[Number(e[t].ref)], - i = n.name.replace(new RegExp(this.query, "i"), function (e) { - return "" + e + "" - }), - s = n.parent || "" - ;(s = s.replace(new RegExp(this.query, "i"), function (e) { - return "" + e + "" - })) && (i = '' + s + "." + i) - var a = document.createElement("li") - ;(a.classList.value = n.classes), - (a.innerHTML = - '\n ' + - i + - "\n "), - this.results.appendChild(a) - } - } - }), - (t.prototype.setLoadingState = function (e) { - this.loadingState != e && - (this.el.classList.remove(o[this.loadingState].toLowerCase()), - (this.loadingState = e), - this.el.classList.add(o[this.loadingState].toLowerCase()), - this.updateResults()) - }), - (t.prototype.setHasFocus = function (e) { - this.hasFocus != e && - ((this.hasFocus = e), - this.el.classList.toggle("has-focus"), - e ? (this.setQuery(""), (this.field.value = "")) : (this.field.value = this.query)) - }), - (t.prototype.setQuery = function (e) { - ;(this.query = e.trim()), this.updateResults() - }), - (t.prototype.setCurrentResult = function (e) { - var t = this.results.querySelector(".current") - if (t) { - var r = 1 == e ? t.nextElementSibling : t.previousElementSibling - r && (t.classList.remove("current"), r.classList.add("current")) - } else - (t = this.results.querySelector(1 == e ? "li:first-child" : "li:last-child")) && - t.classList.add("current") - }), - (t.prototype.gotoCurrentResult = function () { - var e = this.results.querySelector(".current") - if ((e || (e = this.results.querySelector("li:first-child")), e)) { - var t = e.querySelector("a") - t && (window.location.href = t.href), this.field.blur() - } - }), - (t.prototype.bindEvents = function () { - var e = this - this.results.addEventListener("mousedown", function () { - e.resultClicked = !0 - }), - this.results.addEventListener("mouseup", function () { - ;(e.resultClicked = !1), e.setHasFocus(!1) - }), - this.field.addEventListener("focusin", function () { - e.setHasFocus(!0), e.loadIndex() - }), - this.field.addEventListener("focusout", function () { - e.resultClicked - ? (e.resultClicked = !1) - : setTimeout(function () { - return e.setHasFocus(!1) - }, 100) - }), - this.field.addEventListener("input", function () { - e.setQuery(e.field.value) - }), - this.field.addEventListener("keydown", function (t) { - 13 == t.keyCode || 27 == t.keyCode || 38 == t.keyCode || 40 == t.keyCode - ? ((e.preventPress = !0), - t.preventDefault(), - 13 == t.keyCode - ? e.gotoCurrentResult() - : 27 == t.keyCode - ? e.field.blur() - : 38 == t.keyCode - ? e.setCurrentResult(-1) - : 40 == t.keyCode && e.setCurrentResult(1)) - : (e.preventPress = !1) - }), - this.field.addEventListener("keypress", function (t) { - e.preventPress && t.preventDefault() - }), - document.body.addEventListener("keydown", function (t) { - t.altKey || - t.ctrlKey || - t.metaKey || - (!e.hasFocus && t.keyCode > 47 && t.keyCode < 112 && e.field.focus()) - }) - }), - t - ) - })(u), - d = (function () { - function e() { - this.listeners = {} - } - return ( - (e.prototype.addEventListener = function (e, t) { - e in this.listeners || (this.listeners[e] = []), this.listeners[e].push(t) - }), - (e.prototype.removeEventListener = function (e, t) { - if (e in this.listeners) - for (var r = this.listeners[e], n = 0, i = r.length; n < i; n++) - if (r[n] === t) return void r.splice(n, 1) - }), - (e.prototype.dispatchEvent = function (e) { - if (!(e.type in this.listeners)) return !0 - for (var t = this.listeners[e.type].slice(), r = 0, n = t.length; r < n; r++) - t[r].call(this, e) - return !e.defaultPrevented - }), - e - ) - })(), - f = function (e, t) { - void 0 === t && (t = 100) - var r = Date.now() - return function () { - for (var n = [], i = 0; i < arguments.length; i++) n[i] = arguments[i] - r + t - Date.now() < 0 && (e.apply(void 0, n), (r = Date.now())) - } - }, - p = (function () { - var e = function (t, r) { - return (e = - Object.setPrototypeOf || - ({ __proto__: [] } instanceof Array && - function (e, t) { - e.__proto__ = t - }) || - function (e, t) { - for (var r in t) Object.prototype.hasOwnProperty.call(t, r) && (e[r] = t[r]) - })(t, r) - } - return function (t, r) { - function n() { - this.constructor = t - } - e(t, r), - (t.prototype = null === r ? Object.create(r) : ((n.prototype = r.prototype), new n())) - } - })(), - y = (function (e) { - function t() { - var t = e.call(this) || this - return ( - (t.scrollTop = 0), - (t.lastY = 0), - (t.width = 0), - (t.height = 0), - (t.showToolbar = !0), - (t.toolbar = document.querySelector(".tsd-page-toolbar")), - (t.secondaryNav = document.querySelector(".tsd-navigation.secondary")), - window.addEventListener( - "scroll", - f(function () { - return t.onScroll() - }, 10) - ), - window.addEventListener( - "resize", - f(function () { - return t.onResize() - }, 10) - ), - t.onResize(), - t.onScroll(), - t - ) - } - return ( - p(t, e), - (t.prototype.triggerResize = function () { - var e = new CustomEvent("resize", { - detail: { width: this.width, height: this.height } - }) - this.dispatchEvent(e) - }), - (t.prototype.onResize = function () { - ;(this.width = window.innerWidth || 0), (this.height = window.innerHeight || 0) - var e = new CustomEvent("resize", { - detail: { width: this.width, height: this.height } - }) - this.dispatchEvent(e) - }), - (t.prototype.onScroll = function () { - this.scrollTop = window.scrollY || 0 - var e = new CustomEvent("scroll", { detail: { scrollTop: this.scrollTop } }) - this.dispatchEvent(e), this.hideShowToolbar() - }), - (t.prototype.hideShowToolbar = function () { - var e = this.showToolbar - ;(this.showToolbar = this.lastY >= this.scrollTop || 0 === this.scrollTop), - e !== this.showToolbar && - (this.toolbar.classList.toggle("tsd-page-toolbar--hide"), - this.secondaryNav.classList.toggle("tsd-navigation--toolbar-hide")), - (this.lastY = this.scrollTop) - }), - (t.instance = new t()), - t - ) - })(d), - m = (function () { - var e = function (t, r) { - return (e = - Object.setPrototypeOf || - ({ __proto__: [] } instanceof Array && - function (e, t) { - e.__proto__ = t - }) || - function (e, t) { - for (var r in t) Object.prototype.hasOwnProperty.call(t, r) && (e[r] = t[r]) - })(t, r) - } - return function (t, r) { - function n() { - this.constructor = t - } - e(t, r), - (t.prototype = null === r ? Object.create(r) : ((n.prototype = r.prototype), new n())) - } - })(), - v = (function (e) { - function t(t) { - var r = e.call(this, t) || this - return ( - (r.anchors = []), - (r.index = -1), - y.instance.addEventListener("resize", function () { - return r.onResize() - }), - y.instance.addEventListener("scroll", function (e) { - return r.onScroll(e) - }), - r.createAnchors(), - r - ) - } - return ( - m(t, e), - (t.prototype.createAnchors = function () { - var e = this, - t = window.location.href - ;-1 != t.indexOf("#") && (t = t.substr(0, t.indexOf("#"))), - this.el.querySelectorAll("a").forEach(function (r) { - var n = r.href - if (-1 != n.indexOf("#") && n.substr(0, t.length) == t) { - var i = n.substr(n.indexOf("#") + 1), - s = document.querySelector("a.tsd-anchor[name=" + i + "]"), - o = r.parentNode - s && o && e.anchors.push({ link: o, anchor: s, position: 0 }) - } - }), - this.onResize() - }), - (t.prototype.onResize = function () { - for (var e, t = 0, r = this.anchors.length; t < r; t++) { - var n = (e = this.anchors[t]).anchor.getBoundingClientRect() - e.position = n.top + document.body.scrollTop - } - this.anchors.sort(function (e, t) { - return e.position - t.position - }) - var i = new CustomEvent("scroll", { detail: { scrollTop: y.instance.scrollTop } }) - this.onScroll(i) - }), - (t.prototype.onScroll = function (e) { - for ( - var t = e.detail.scrollTop + 5, r = this.anchors, n = r.length - 1, i = this.index; - i > -1 && r[i].position > t; - - ) - i -= 1 - for (; i < n && r[i + 1].position < t; ) i += 1 - this.index != i && - (this.index > -1 && this.anchors[this.index].link.classList.remove("focus"), - (this.index = i), - this.index > -1 && this.anchors[this.index].link.classList.add("focus")) - }), - t - ) - })(u), - g = (function () { - var e = function (t, r) { - return (e = - Object.setPrototypeOf || - ({ __proto__: [] } instanceof Array && - function (e, t) { - e.__proto__ = t - }) || - function (e, t) { - for (var r in t) Object.prototype.hasOwnProperty.call(t, r) && (e[r] = t[r]) - })(t, r) - } - return function (t, r) { - function n() { - this.constructor = t - } - e(t, r), - (t.prototype = null === r ? Object.create(r) : ((n.prototype = r.prototype), new n())) - } - })(), - x = (function () { - function e(e, t) { - ;(this.signature = e), (this.description = t) - } - return ( - (e.prototype.addClass = function (e) { - return this.signature.classList.add(e), this.description.classList.add(e), this - }), - (e.prototype.removeClass = function (e) { - return this.signature.classList.remove(e), this.description.classList.remove(e), this - }), - e - ) - })(), - w = (function (e) { - function t(t) { - var r = e.call(this, t) || this - return ( - (r.groups = []), - (r.index = -1), - r.createGroups(), - r.container && - (r.el.classList.add("active"), - Array.from(r.el.children).forEach(function (e) { - e.addEventListener("touchstart", function (e) { - return r.onClick(e) - }), - e.addEventListener("click", function (e) { - return r.onClick(e) - }) - }), - r.container.classList.add("active"), - r.setIndex(0)), - r - ) - } - return ( - g(t, e), - (t.prototype.setIndex = function (e) { - if ( - (e < 0 && (e = 0), - e > this.groups.length - 1 && (e = this.groups.length - 1), - this.index != e) - ) { - var t = this.groups[e] - if (this.index > -1) { - var r = this.groups[this.index] - r.removeClass("current").addClass("fade-out"), - t.addClass("current"), - t.addClass("fade-in"), - y.instance.triggerResize(), - setTimeout(function () { - r.removeClass("fade-out"), t.removeClass("fade-in") - }, 300) - } else t.addClass("current"), y.instance.triggerResize() - this.index = e - } - }), - (t.prototype.createGroups = function () { - var e = this.el.children - if (!(e.length < 2)) { - this.container = this.el.nextElementSibling - var t = this.container.children - this.groups = [] - for (var r = 0; r < e.length; r++) this.groups.push(new x(e[r], t[r])) - } - }), - (t.prototype.onClick = function (e) { - var t = this - this.groups.forEach(function (r, n) { - r.signature === e.currentTarget && t.setIndex(n) - }) - }), - t - ) - })(u), - L = "mousedown", - E = "mousemove", - b = "mouseup", - S = { x: 0, y: 0 }, - k = !1, - Q = !1, - O = !1, - P = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent) - document.documentElement.classList.add(P ? "is-mobile" : "not-mobile"), - P && - "ontouchstart" in document.documentElement && - (!0, (L = "touchstart"), (E = "touchmove"), (b = "touchend")), - document.addEventListener(L, function (e) { - ;(Q = !0), (O = !1) - var t = "touchstart" == L ? e.targetTouches[0] : e - ;(S.y = t.pageY || 0), (S.x = t.pageX || 0) - }), - document.addEventListener(E, function (e) { - if (Q && !O) { - var t = "touchstart" == L ? e.targetTouches[0] : e, - r = S.x - (t.pageX || 0), - n = S.y - (t.pageY || 0) - O = Math.sqrt(r * r + n * n) > 10 - } - }), - document.addEventListener(b, function () { - Q = !1 - }), - document.addEventListener("click", function (e) { - k && (e.preventDefault(), e.stopImmediatePropagation(), (k = !1)) - }) - var T = (function () { - var e = function (t, r) { - return (e = - Object.setPrototypeOf || - ({ __proto__: [] } instanceof Array && - function (e, t) { - e.__proto__ = t - }) || - function (e, t) { - for (var r in t) Object.prototype.hasOwnProperty.call(t, r) && (e[r] = t[r]) - })(t, r) - } - return function (t, r) { - function n() { - this.constructor = t - } - e(t, r), - (t.prototype = null === r ? Object.create(r) : ((n.prototype = r.prototype), new n())) - } - })(), - _ = (function (e) { - function t(t) { - var r = e.call(this, t) || this - return ( - (r.className = r.el.dataset.toggle || ""), - r.el.addEventListener(b, function (e) { - return r.onPointerUp(e) - }), - r.el.addEventListener("click", function (e) { - return e.preventDefault() - }), - document.addEventListener(L, function (e) { - return r.onDocumentPointerDown(e) - }), - document.addEventListener(b, function (e) { - return r.onDocumentPointerUp(e) - }), - r - ) - } - return ( - T(t, e), - (t.prototype.setActive = function (e) { - if (this.active != e) { - ;(this.active = e), - document.documentElement.classList.toggle("has-" + this.className, e), - this.el.classList.toggle("active", e) - var t = (this.active ? "to-has-" : "from-has-") + this.className - document.documentElement.classList.add(t), - setTimeout(function () { - return document.documentElement.classList.remove(t) - }, 500) - } - }), - (t.prototype.onPointerUp = function (e) { - O || (this.setActive(!0), e.preventDefault()) - }), - (t.prototype.onDocumentPointerDown = function (e) { - if (this.active) { - if (e.target.closest(".col-menu, .tsd-filter-group")) return - this.setActive(!1) - } - }), - (t.prototype.onDocumentPointerUp = function (e) { - var t = this - if (!O && this.active && e.target.closest(".col-menu")) { - var r = e.target.closest("a") - if (r) { - var n = window.location.href - ;-1 != n.indexOf("#") && (n = n.substr(0, n.indexOf("#"))), - r.href.substr(0, n.length) == n && - setTimeout(function () { - return t.setActive(!1) - }, 250) - } - } - }), - t - ) - })(u), - C = (function () { - var e = function (t, r) { - return (e = - Object.setPrototypeOf || - ({ __proto__: [] } instanceof Array && - function (e, t) { - e.__proto__ = t - }) || - function (e, t) { - for (var r in t) Object.prototype.hasOwnProperty.call(t, r) && (e[r] = t[r]) - })(t, r) - } - return function (t, r) { - function n() { - this.constructor = t - } - e(t, r), - (t.prototype = null === r ? Object.create(r) : ((n.prototype = r.prototype), new n())) - } - })(), - R = (function () { - function e(e, t) { - ;(this.key = e), - (this.value = t), - (this.defaultValue = t), - this.initialize(), - window.localStorage[this.key] && - this.setValue(this.fromLocalStorage(window.localStorage[this.key])) - } - return ( - (e.prototype.initialize = function () {}), - (e.prototype.setValue = function (e) { - if (this.value != e) { - var t = this.value - ;(this.value = e), - (window.localStorage[this.key] = this.toLocalStorage(e)), - this.handleValueChange(t, e) - } - }), - e - ) - })(), - I = (function (e) { - function t() { - return (null !== e && e.apply(this, arguments)) || this - } - return ( - C(t, e), - (t.prototype.initialize = function () { - var e = this, - t = document.querySelector("#tsd-filter-" + this.key) - t && - ((this.checkbox = t), - this.checkbox.addEventListener("change", function () { - e.setValue(e.checkbox.checked) - })) - }), - (t.prototype.handleValueChange = function (e, t) { - this.checkbox && - ((this.checkbox.checked = this.value), - document.documentElement.classList.toggle( - "toggle-" + this.key, - this.value != this.defaultValue - )) - }), - (t.prototype.fromLocalStorage = function (e) { - return "true" == e - }), - (t.prototype.toLocalStorage = function (e) { - return e ? "true" : "false" - }), - t - ) - })(R), - j = (function (e) { - function t() { - return (null !== e && e.apply(this, arguments)) || this - } - return ( - C(t, e), - (t.prototype.initialize = function () { - var e = this - document.documentElement.classList.add("toggle-" + this.key + this.value) - var t = document.querySelector("#tsd-filter-" + this.key) - if (t) { - this.select = t - var r = function () { - e.select.classList.add("active") - } - this.select.addEventListener(L, r), - this.select.addEventListener("mouseover", r), - this.select.addEventListener("mouseleave", function () { - e.select.classList.remove("active") - }), - this.select.querySelectorAll("li").forEach(function (r) { - r.addEventListener(b, function (r) { - t.classList.remove("active"), e.setValue(r.target.dataset.value || "") - }) - }), - document.addEventListener(L, function (t) { - e.select.contains(t.target) || e.select.classList.remove("active") - }) - } - }), - (t.prototype.handleValueChange = function (e, t) { - this.select.querySelectorAll("li.selected").forEach(function (e) { - e.classList.remove("selected") - }) - var r = this.select.querySelector('li[data-value="' + t + '"]'), - n = this.select.querySelector(".tsd-select-label") - r && n && (r.classList.add("selected"), (n.textContent = r.textContent)), - document.documentElement.classList.remove("toggle-" + e), - document.documentElement.classList.add("toggle-" + t) - }), - (t.prototype.fromLocalStorage = function (e) { - return e - }), - (t.prototype.toLocalStorage = function (e) { - return e - }), - t - ) - })(R), - F = (function (e) { - function t(t) { - var r = e.call(this, t) || this - return ( - (r.optionVisibility = new j("visibility", "private")), - (r.optionInherited = new I("inherited", !0)), - (r.optionExternals = new I("externals", !0)), - (r.optionOnlyExported = new I("only-exported", !1)), - r - ) - } - return ( - C(t, e), - (t.isSupported = function () { - try { - return void 0 !== window.localStorage - } catch (e) { - return !1 - } - }), - t - ) - })(u) - r(1) - i(h, "#tsd-search"), - i(v, ".menu-highlight"), - i(w, ".tsd-signatures"), - i(_, "a[data-toggle]"), - F.isSupported() ? i(F, "#tsd-filter") : document.documentElement.classList.add("no-filter") - var N = new a() - Object.defineProperty(window, "app", { value: N }) - } -]) +!function(e){var t={};function r(n){if(t[n])return t[n].exports;var i=t[n]={i:n,l:!1,exports:{}};return e[n].call(i.exports,i,i.exports,r),i.l=!0,i.exports}r.m=e,r.c=t,r.d=function(e,t,n){r.o(e,t)||Object.defineProperty(e,t,{enumerable:!0,get:n})},r.r=function(e){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},r.t=function(e,t){if(1&t&&(e=r(e)),8&t)return e;if(4&t&&"object"==typeof e&&e&&e.__esModule)return e;var n=Object.create(null);if(r.r(n),Object.defineProperty(n,"default",{enumerable:!0,value:e}),2&t&&"string"!=typeof e)for(var i in e)r.d(n,i,function(t){return e[t]}.bind(null,i));return n},r.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return r.d(t,"a",t),t},r.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},r.p="",r(r.s=2)}([function(e,t,r){var n,i; +/** + * lunr - http://lunrjs.com - A bit like Solr, but much smaller and not as bright - 2.3.9 + * Copyright (C) 2020 Oliver Nightingale + * @license MIT + */!function(){var s,o,a,u,l,c,h,d,f,p,y,m,v,g,x,w,L,E,b,S,k,Q,O,P,T,_,C=function(e){var t=new C.Builder;return t.pipeline.add(C.trimmer,C.stopWordFilter,C.stemmer),t.searchPipeline.add(C.stemmer),e.call(t,t),t.build()};C.version="2.3.9" +/*! + * lunr.utils + * Copyright (C) 2020 Oliver Nightingale + */,C.utils={},C.utils.warn=(s=this,function(e){s.console&&console.warn&&console.warn(e)}),C.utils.asString=function(e){return null==e?"":e.toString()},C.utils.clone=function(e){if(null==e)return e;for(var t=Object.create(null),r=Object.keys(e),n=0;n0){var u=C.utils.clone(t)||{};u.position=[o,a],u.index=i.length,i.push(new C.Token(r.slice(o,s),u))}o=s+1}}return i},C.tokenizer.separator=/[\s\-]+/ +/*! + * lunr.Pipeline + * Copyright (C) 2020 Oliver Nightingale + */,C.Pipeline=function(){this._stack=[]},C.Pipeline.registeredFunctions=Object.create(null),C.Pipeline.registerFunction=function(e,t){t in this.registeredFunctions&&C.utils.warn("Overwriting existing registered function: "+t),e.label=t,C.Pipeline.registeredFunctions[e.label]=e},C.Pipeline.warnIfFunctionNotRegistered=function(e){e.label&&e.label in this.registeredFunctions||C.utils.warn("Function is not registered with pipeline. This may cause problems when serialising the index.\n",e)},C.Pipeline.load=function(e){var t=new C.Pipeline;return e.forEach((function(e){var r=C.Pipeline.registeredFunctions[e];if(!r)throw new Error("Cannot load unregistered function: "+e);t.add(r)})),t},C.Pipeline.prototype.add=function(){var e=Array.prototype.slice.call(arguments);e.forEach((function(e){C.Pipeline.warnIfFunctionNotRegistered(e),this._stack.push(e)}),this)},C.Pipeline.prototype.after=function(e,t){C.Pipeline.warnIfFunctionNotRegistered(t);var r=this._stack.indexOf(e);if(-1==r)throw new Error("Cannot find existingFn");r+=1,this._stack.splice(r,0,t)},C.Pipeline.prototype.before=function(e,t){C.Pipeline.warnIfFunctionNotRegistered(t);var r=this._stack.indexOf(e);if(-1==r)throw new Error("Cannot find existingFn");this._stack.splice(r,0,t)},C.Pipeline.prototype.remove=function(e){var t=this._stack.indexOf(e);-1!=t&&this._stack.splice(t,1)},C.Pipeline.prototype.run=function(e){for(var t=this._stack.length,r=0;r1&&(se&&(r=i),s!=e);)n=r-t,i=t+Math.floor(n/2),s=this.elements[2*i];return s==e||s>e?2*i:sa?l+=2:o==a&&(t+=r[u+1]*n[l+1],u+=2,l+=2);return t},C.Vector.prototype.similarity=function(e){return this.dot(e)/this.magnitude()||0},C.Vector.prototype.toArray=function(){for(var e=new Array(this.elements.length/2),t=1,r=0;t0){var s,o=i.str.charAt(0);o in i.node.edges?s=i.node.edges[o]:(s=new C.TokenSet,i.node.edges[o]=s),1==i.str.length&&(s.final=!0),n.push({node:s,editsRemaining:i.editsRemaining,str:i.str.slice(1)})}if(0!=i.editsRemaining){if("*"in i.node.edges)var a=i.node.edges["*"];else{a=new C.TokenSet;i.node.edges["*"]=a}if(0==i.str.length&&(a.final=!0),n.push({node:a,editsRemaining:i.editsRemaining-1,str:i.str}),i.str.length>1&&n.push({node:i.node,editsRemaining:i.editsRemaining-1,str:i.str.slice(1)}),1==i.str.length&&(i.node.final=!0),i.str.length>=1){if("*"in i.node.edges)var u=i.node.edges["*"];else{u=new C.TokenSet;i.node.edges["*"]=u}1==i.str.length&&(u.final=!0),n.push({node:u,editsRemaining:i.editsRemaining-1,str:i.str.slice(1)})}if(i.str.length>1){var l,c=i.str.charAt(0),h=i.str.charAt(1);h in i.node.edges?l=i.node.edges[h]:(l=new C.TokenSet,i.node.edges[h]=l),1==i.str.length&&(l.final=!0),n.push({node:l,editsRemaining:i.editsRemaining-1,str:c+i.str.slice(2)})}}}return r},C.TokenSet.fromString=function(e){for(var t=new C.TokenSet,r=t,n=0,i=e.length;n=e;t--){var r=this.uncheckedNodes[t],n=r.child.toString();n in this.minimizedNodes?r.parent.edges[r.char]=this.minimizedNodes[n]:(r.child._str=n,this.minimizedNodes[n]=r.child),this.uncheckedNodes.pop()}} +/*! + * lunr.Index + * Copyright (C) 2020 Oliver Nightingale + */,C.Index=function(e){this.invertedIndex=e.invertedIndex,this.fieldVectors=e.fieldVectors,this.tokenSet=e.tokenSet,this.fields=e.fields,this.pipeline=e.pipeline},C.Index.prototype.search=function(e){return this.query((function(t){new C.QueryParser(e,t).parse()}))},C.Index.prototype.query=function(e){for(var t=new C.Query(this.fields),r=Object.create(null),n=Object.create(null),i=Object.create(null),s=Object.create(null),o=Object.create(null),a=0;a1?1:e},C.Builder.prototype.k1=function(e){this._k1=e},C.Builder.prototype.add=function(e,t){var r=e[this._ref],n=Object.keys(this._fields);this._documents[r]=t||{},this.documentCount+=1;for(var i=0;i=this.length)return C.QueryLexer.EOS;var e=this.str.charAt(this.pos);return this.pos+=1,e},C.QueryLexer.prototype.width=function(){return this.pos-this.start},C.QueryLexer.prototype.ignore=function(){this.start==this.pos&&(this.pos+=1),this.start=this.pos},C.QueryLexer.prototype.backup=function(){this.pos-=1},C.QueryLexer.prototype.acceptDigitRun=function(){var e,t;do{t=(e=this.next()).charCodeAt(0)}while(t>47&&t<58);e!=C.QueryLexer.EOS&&this.backup()},C.QueryLexer.prototype.more=function(){return this.pos1&&(e.backup(),e.emit(C.QueryLexer.TERM)),e.ignore(),e.more())return C.QueryLexer.lexText},C.QueryLexer.lexEditDistance=function(e){return e.ignore(),e.acceptDigitRun(),e.emit(C.QueryLexer.EDIT_DISTANCE),C.QueryLexer.lexText},C.QueryLexer.lexBoost=function(e){return e.ignore(),e.acceptDigitRun(),e.emit(C.QueryLexer.BOOST),C.QueryLexer.lexText},C.QueryLexer.lexEOS=function(e){e.width()>0&&e.emit(C.QueryLexer.TERM)},C.QueryLexer.termSeparator=C.tokenizer.separator,C.QueryLexer.lexText=function(e){for(;;){var t=e.next();if(t==C.QueryLexer.EOS)return C.QueryLexer.lexEOS;if(92!=t.charCodeAt(0)){if(":"==t)return C.QueryLexer.lexField;if("~"==t)return e.backup(),e.width()>0&&e.emit(C.QueryLexer.TERM),C.QueryLexer.lexEditDistance;if("^"==t)return e.backup(),e.width()>0&&e.emit(C.QueryLexer.TERM),C.QueryLexer.lexBoost;if("+"==t&&1===e.width())return e.emit(C.QueryLexer.PRESENCE),C.QueryLexer.lexText;if("-"==t&&1===e.width())return e.emit(C.QueryLexer.PRESENCE),C.QueryLexer.lexText;if(t.match(C.QueryLexer.termSeparator))return C.QueryLexer.lexTerm}else e.escapeCharacter()}},C.QueryParser=function(e,t){this.lexer=new C.QueryLexer(e),this.query=t,this.currentClause={},this.lexemeIdx=0},C.QueryParser.prototype.parse=function(){this.lexer.run(),this.lexemes=this.lexer.lexemes;for(var e=C.QueryParser.parseClause;e;)e=e(this);return this.query},C.QueryParser.prototype.peekLexeme=function(){return this.lexemes[this.lexemeIdx]},C.QueryParser.prototype.consumeLexeme=function(){var e=this.peekLexeme();return this.lexemeIdx+=1,e},C.QueryParser.prototype.nextClause=function(){var e=this.currentClause;this.query.clause(e),this.currentClause={}},C.QueryParser.parseClause=function(e){var t=e.peekLexeme();if(null!=t)switch(t.type){case C.QueryLexer.PRESENCE:return C.QueryParser.parsePresence;case C.QueryLexer.FIELD:return C.QueryParser.parseField;case C.QueryLexer.TERM:return C.QueryParser.parseTerm;default:var r="expected either a field or a term, found "+t.type;throw t.str.length>=1&&(r+=" with value '"+t.str+"'"),new C.QueryParseError(r,t.start,t.end)}},C.QueryParser.parsePresence=function(e){var t=e.consumeLexeme();if(null!=t){switch(t.str){case"-":e.currentClause.presence=C.Query.presence.PROHIBITED;break;case"+":e.currentClause.presence=C.Query.presence.REQUIRED;break;default:var r="unrecognised presence operator'"+t.str+"'";throw new C.QueryParseError(r,t.start,t.end)}var n=e.peekLexeme();if(null==n){r="expecting term or field, found nothing";throw new C.QueryParseError(r,t.start,t.end)}switch(n.type){case C.QueryLexer.FIELD:return C.QueryParser.parseField;case C.QueryLexer.TERM:return C.QueryParser.parseTerm;default:r="expecting term or field, found '"+n.type+"'";throw new C.QueryParseError(r,n.start,n.end)}}},C.QueryParser.parseField=function(e){var t=e.consumeLexeme();if(null!=t){if(-1==e.query.allFields.indexOf(t.str)){var r=e.query.allFields.map((function(e){return"'"+e+"'"})).join(", "),n="unrecognised field '"+t.str+"', possible fields: "+r;throw new C.QueryParseError(n,t.start,t.end)}e.currentClause.fields=[t.str];var i=e.peekLexeme();if(null==i){n="expecting term, found nothing";throw new C.QueryParseError(n,t.start,t.end)}switch(i.type){case C.QueryLexer.TERM:return C.QueryParser.parseTerm;default:n="expecting term, found '"+i.type+"'";throw new C.QueryParseError(n,i.start,i.end)}}},C.QueryParser.parseTerm=function(e){var t=e.consumeLexeme();if(null!=t){e.currentClause.term=t.str.toLowerCase(),-1!=t.str.indexOf("*")&&(e.currentClause.usePipeline=!1);var r=e.peekLexeme();if(null!=r)switch(r.type){case C.QueryLexer.TERM:return e.nextClause(),C.QueryParser.parseTerm;case C.QueryLexer.FIELD:return e.nextClause(),C.QueryParser.parseField;case C.QueryLexer.EDIT_DISTANCE:return C.QueryParser.parseEditDistance;case C.QueryLexer.BOOST:return C.QueryParser.parseBoost;case C.QueryLexer.PRESENCE:return e.nextClause(),C.QueryParser.parsePresence;default:var n="Unexpected lexeme type '"+r.type+"'";throw new C.QueryParseError(n,r.start,r.end)}else e.nextClause()}},C.QueryParser.parseEditDistance=function(e){var t=e.consumeLexeme();if(null!=t){var r=parseInt(t.str,10);if(isNaN(r)){var n="edit distance must be numeric";throw new C.QueryParseError(n,t.start,t.end)}e.currentClause.editDistance=r;var i=e.peekLexeme();if(null!=i)switch(i.type){case C.QueryLexer.TERM:return e.nextClause(),C.QueryParser.parseTerm;case C.QueryLexer.FIELD:return e.nextClause(),C.QueryParser.parseField;case C.QueryLexer.EDIT_DISTANCE:return C.QueryParser.parseEditDistance;case C.QueryLexer.BOOST:return C.QueryParser.parseBoost;case C.QueryLexer.PRESENCE:return e.nextClause(),C.QueryParser.parsePresence;default:n="Unexpected lexeme type '"+i.type+"'";throw new C.QueryParseError(n,i.start,i.end)}else e.nextClause()}},C.QueryParser.parseBoost=function(e){var t=e.consumeLexeme();if(null!=t){var r=parseInt(t.str,10);if(isNaN(r)){var n="boost must be numeric";throw new C.QueryParseError(n,t.start,t.end)}e.currentClause.boost=r;var i=e.peekLexeme();if(null!=i)switch(i.type){case C.QueryLexer.TERM:return e.nextClause(),C.QueryParser.parseTerm;case C.QueryLexer.FIELD:return e.nextClause(),C.QueryParser.parseField;case C.QueryLexer.EDIT_DISTANCE:return C.QueryParser.parseEditDistance;case C.QueryLexer.BOOST:return C.QueryParser.parseBoost;case C.QueryLexer.PRESENCE:return e.nextClause(),C.QueryParser.parsePresence;default:n="Unexpected lexeme type '"+i.type+"'";throw new C.QueryParseError(n,i.start,i.end)}else e.nextClause()}},void 0===(i="function"==typeof(n=function(){return C})?n.call(t,r,t,e):n)||(e.exports=i)}()},function(e,t,r){},function(e,t,r){"use strict";r.r(t);var n=[];function i(e,t){n.push({selector:t,constructor:e})}var s,o,a=function(){function e(){this.createComponents(document.body)}return e.prototype.createComponents=function(e){n.forEach((function(t){e.querySelectorAll(t.selector).forEach((function(e){e.dataset.hasInstance||(new t.constructor({el:e}),e.dataset.hasInstance=String(!0))}))}))},e}(),u=function(e){this.el=e.el},l=r(0),c=(s=function(e,t){return(s=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(e,t){e.__proto__=t}||function(e,t){for(var r in t)Object.prototype.hasOwnProperty.call(t,r)&&(e[r]=t[r])})(e,t)},function(e,t){function r(){this.constructor=e}s(e,t),e.prototype=null===t?Object.create(t):(r.prototype=t.prototype,new r)});!function(e){e[e.Idle=0]="Idle",e[e.Loading=1]="Loading",e[e.Ready=2]="Ready",e[e.Failure=3]="Failure"}(o||(o={}));var h=function(e){function t(t){var r=e.call(this,t)||this;r.query="",r.loadingState=o.Idle,r.hasFocus=!1,r.preventPress=!1,r.data=null,r.index=null,r.resultClicked=!1;var n=document.querySelector("#tsd-search-field"),i=document.querySelector(".results");if(!n||!i)throw new Error("The input field or the result list wrapper are not found");return r.field=n,r.results=i,r.base=r.el.dataset.base+"/",r.bindEvents(),r}return c(t,e),t.prototype.loadIndex=function(){var e=this;if(this.loadingState==o.Idle&&!this.data){setTimeout((function(){e.loadingState==o.Idle&&e.setLoadingState(o.Loading)}),500);var t=this.el.dataset.index;t?fetch(t).then((function(e){if(!e.ok)throw new Error("The search index is missing");return e.json()})).then((function(t){e.data=t,e.index=l.Index.load(t.index),e.setLoadingState(o.Ready)})).catch((function(t){console.error(t),e.setLoadingState(o.Failure)})):this.setLoadingState(o.Failure)}},t.prototype.updateResults=function(){if(this.loadingState==o.Ready&&(this.results.textContent="",this.query&&this.index&&this.data)){var e=this.index.search("*"+this.query+"*");0===e.length&&(e=this.index.search("*"+this.query+"~1*"));for(var t=0,r=Math.min(10,e.length);t"+e+""})),s=n.parent||"";(s=s.replace(new RegExp(this.query,"i"),(function(e){return""+e+""})))&&(i=''+s+"."+i);var a=document.createElement("li");a.classList.value=n.classes,a.innerHTML='\n '+i+"\n ",this.results.appendChild(a)}}},t.prototype.setLoadingState=function(e){this.loadingState!=e&&(this.el.classList.remove(o[this.loadingState].toLowerCase()),this.loadingState=e,this.el.classList.add(o[this.loadingState].toLowerCase()),this.updateResults())},t.prototype.setHasFocus=function(e){this.hasFocus!=e&&(this.hasFocus=e,this.el.classList.toggle("has-focus"),e?(this.setQuery(""),this.field.value=""):this.field.value=this.query)},t.prototype.setQuery=function(e){this.query=e.trim(),this.updateResults()},t.prototype.setCurrentResult=function(e){var t=this.results.querySelector(".current");if(t){var r=1==e?t.nextElementSibling:t.previousElementSibling;r&&(t.classList.remove("current"),r.classList.add("current"))}else(t=this.results.querySelector(1==e?"li:first-child":"li:last-child"))&&t.classList.add("current")},t.prototype.gotoCurrentResult=function(){var e=this.results.querySelector(".current");if(e||(e=this.results.querySelector("li:first-child")),e){var t=e.querySelector("a");t&&(window.location.href=t.href),this.field.blur()}},t.prototype.bindEvents=function(){var e=this;this.results.addEventListener("mousedown",(function(){e.resultClicked=!0})),this.results.addEventListener("mouseup",(function(){e.resultClicked=!1,e.setHasFocus(!1)})),this.field.addEventListener("focusin",(function(){e.setHasFocus(!0),e.loadIndex()})),this.field.addEventListener("focusout",(function(){e.resultClicked?e.resultClicked=!1:setTimeout((function(){return e.setHasFocus(!1)}),100)})),this.field.addEventListener("input",(function(){e.setQuery(e.field.value)})),this.field.addEventListener("keydown",(function(t){13==t.keyCode||27==t.keyCode||38==t.keyCode||40==t.keyCode?(e.preventPress=!0,t.preventDefault(),13==t.keyCode?e.gotoCurrentResult():27==t.keyCode?e.field.blur():38==t.keyCode?e.setCurrentResult(-1):40==t.keyCode&&e.setCurrentResult(1)):e.preventPress=!1})),this.field.addEventListener("keypress",(function(t){e.preventPress&&t.preventDefault()})),document.body.addEventListener("keydown",(function(t){t.altKey||t.ctrlKey||t.metaKey||!e.hasFocus&&t.keyCode>47&&t.keyCode<112&&e.field.focus()}))},t}(u),d=function(){function e(){this.listeners={}}return e.prototype.addEventListener=function(e,t){e in this.listeners||(this.listeners[e]=[]),this.listeners[e].push(t)},e.prototype.removeEventListener=function(e,t){if(e in this.listeners)for(var r=this.listeners[e],n=0,i=r.length;n=this.scrollTop||0===this.scrollTop,e!==this.showToolbar&&(this.toolbar.classList.toggle("tsd-page-toolbar--hide"),this.secondaryNav.classList.toggle("tsd-navigation--toolbar-hide")),this.lastY=this.scrollTop},t.instance=new t,t}(d),m=function(){var e=function(t,r){return(e=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(e,t){e.__proto__=t}||function(e,t){for(var r in t)Object.prototype.hasOwnProperty.call(t,r)&&(e[r]=t[r])})(t,r)};return function(t,r){function n(){this.constructor=t}e(t,r),t.prototype=null===r?Object.create(r):(n.prototype=r.prototype,new n)}}(),v=function(e){function t(t){var r=e.call(this,t)||this;return r.anchors=[],r.index=-1,y.instance.addEventListener("resize",(function(){return r.onResize()})),y.instance.addEventListener("scroll",(function(e){return r.onScroll(e)})),r.createAnchors(),r}return m(t,e),t.prototype.createAnchors=function(){var e=this,t=window.location.href;-1!=t.indexOf("#")&&(t=t.substr(0,t.indexOf("#"))),this.el.querySelectorAll("a").forEach((function(r){var n=r.href;if(-1!=n.indexOf("#")&&n.substr(0,t.length)==t){var i=n.substr(n.indexOf("#")+1),s=document.querySelector("a.tsd-anchor[name="+i+"]"),o=r.parentNode;s&&o&&e.anchors.push({link:o,anchor:s,position:0})}})),this.onResize()},t.prototype.onResize=function(){for(var e,t=0,r=this.anchors.length;t-1&&r[i].position>t;)i-=1;for(;i-1&&this.anchors[this.index].link.classList.remove("focus"),this.index=i,this.index>-1&&this.anchors[this.index].link.classList.add("focus"))},t}(u),g=function(){var e=function(t,r){return(e=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(e,t){e.__proto__=t}||function(e,t){for(var r in t)Object.prototype.hasOwnProperty.call(t,r)&&(e[r]=t[r])})(t,r)};return function(t,r){function n(){this.constructor=t}e(t,r),t.prototype=null===r?Object.create(r):(n.prototype=r.prototype,new n)}}(),x=function(){function e(e,t){this.signature=e,this.description=t}return e.prototype.addClass=function(e){return this.signature.classList.add(e),this.description.classList.add(e),this},e.prototype.removeClass=function(e){return this.signature.classList.remove(e),this.description.classList.remove(e),this},e}(),w=function(e){function t(t){var r=e.call(this,t)||this;return r.groups=[],r.index=-1,r.createGroups(),r.container&&(r.el.classList.add("active"),Array.from(r.el.children).forEach((function(e){e.addEventListener("touchstart",(function(e){return r.onClick(e)})),e.addEventListener("click",(function(e){return r.onClick(e)}))})),r.container.classList.add("active"),r.setIndex(0)),r}return g(t,e),t.prototype.setIndex=function(e){if(e<0&&(e=0),e>this.groups.length-1&&(e=this.groups.length-1),this.index!=e){var t=this.groups[e];if(this.index>-1){var r=this.groups[this.index];r.removeClass("current").addClass("fade-out"),t.addClass("current"),t.addClass("fade-in"),y.instance.triggerResize(),setTimeout((function(){r.removeClass("fade-out"),t.removeClass("fade-in")}),300)}else t.addClass("current"),y.instance.triggerResize();this.index=e}},t.prototype.createGroups=function(){var e=this.el.children;if(!(e.length<2)){this.container=this.el.nextElementSibling;var t=this.container.children;this.groups=[];for(var r=0;r10}})),document.addEventListener(b,(function(){Q=!1})),document.addEventListener("click",(function(e){k&&(e.preventDefault(),e.stopImmediatePropagation(),k=!1)}));var T=function(){var e=function(t,r){return(e=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(e,t){e.__proto__=t}||function(e,t){for(var r in t)Object.prototype.hasOwnProperty.call(t,r)&&(e[r]=t[r])})(t,r)};return function(t,r){function n(){this.constructor=t}e(t,r),t.prototype=null===r?Object.create(r):(n.prototype=r.prototype,new n)}}(),_=function(e){function t(t){var r=e.call(this,t)||this;return r.className=r.el.dataset.toggle||"",r.el.addEventListener(b,(function(e){return r.onPointerUp(e)})),r.el.addEventListener("click",(function(e){return e.preventDefault()})),document.addEventListener(L,(function(e){return r.onDocumentPointerDown(e)})),document.addEventListener(b,(function(e){return r.onDocumentPointerUp(e)})),r}return T(t,e),t.prototype.setActive=function(e){if(this.active!=e){this.active=e,document.documentElement.classList.toggle("has-"+this.className,e),this.el.classList.toggle("active",e);var t=(this.active?"to-has-":"from-has-")+this.className;document.documentElement.classList.add(t),setTimeout((function(){return document.documentElement.classList.remove(t)}),500)}},t.prototype.onPointerUp=function(e){O||(this.setActive(!0),e.preventDefault())},t.prototype.onDocumentPointerDown=function(e){if(this.active){if(e.target.closest(".col-menu, .tsd-filter-group"))return;this.setActive(!1)}},t.prototype.onDocumentPointerUp=function(e){var t=this;if(!O&&this.active&&e.target.closest(".col-menu")){var r=e.target.closest("a");if(r){var n=window.location.href;-1!=n.indexOf("#")&&(n=n.substr(0,n.indexOf("#"))),r.href.substr(0,n.length)==n&&setTimeout((function(){return t.setActive(!1)}),250)}}},t}(u),C=function(){var e=function(t,r){return(e=Object.setPrototypeOf||{__proto__:[]}instanceof Array&&function(e,t){e.__proto__=t}||function(e,t){for(var r in t)Object.prototype.hasOwnProperty.call(t,r)&&(e[r]=t[r])})(t,r)};return function(t,r){function n(){this.constructor=t}e(t,r),t.prototype=null===r?Object.create(r):(n.prototype=r.prototype,new n)}}(),R=function(){function e(e,t){this.key=e,this.value=t,this.defaultValue=t,this.initialize(),window.localStorage[this.key]&&this.setValue(this.fromLocalStorage(window.localStorage[this.key]))}return e.prototype.initialize=function(){},e.prototype.setValue=function(e){if(this.value!=e){var t=this.value;this.value=e,window.localStorage[this.key]=this.toLocalStorage(e),this.handleValueChange(t,e)}},e}(),I=function(e){function t(){return null!==e&&e.apply(this,arguments)||this}return C(t,e),t.prototype.initialize=function(){var e=this,t=document.querySelector("#tsd-filter-"+this.key);t&&(this.checkbox=t,this.checkbox.addEventListener("change",(function(){e.setValue(e.checkbox.checked)})))},t.prototype.handleValueChange=function(e,t){this.checkbox&&(this.checkbox.checked=this.value,document.documentElement.classList.toggle("toggle-"+this.key,this.value!=this.defaultValue))},t.prototype.fromLocalStorage=function(e){return"true"==e},t.prototype.toLocalStorage=function(e){return e?"true":"false"},t}(R),j=function(e){function t(){return null!==e&&e.apply(this,arguments)||this}return C(t,e),t.prototype.initialize=function(){var e=this;document.documentElement.classList.add("toggle-"+this.key+this.value);var t=document.querySelector("#tsd-filter-"+this.key);if(t){this.select=t;var r=function(){e.select.classList.add("active")};this.select.addEventListener(L,r),this.select.addEventListener("mouseover",r),this.select.addEventListener("mouseleave",(function(){e.select.classList.remove("active")})),this.select.querySelectorAll("li").forEach((function(r){r.addEventListener(b,(function(r){t.classList.remove("active"),e.setValue(r.target.dataset.value||"")}))})),document.addEventListener(L,(function(t){e.select.contains(t.target)||e.select.classList.remove("active")}))}},t.prototype.handleValueChange=function(e,t){this.select.querySelectorAll("li.selected").forEach((function(e){e.classList.remove("selected")}));var r=this.select.querySelector('li[data-value="'+t+'"]'),n=this.select.querySelector(".tsd-select-label");r&&n&&(r.classList.add("selected"),n.textContent=r.textContent),document.documentElement.classList.remove("toggle-"+e),document.documentElement.classList.add("toggle-"+t)},t.prototype.fromLocalStorage=function(e){return e},t.prototype.toLocalStorage=function(e){return e},t}(R),F=function(e){function t(t){var r=e.call(this,t)||this;return r.optionVisibility=new j("visibility","private"),r.optionInherited=new I("inherited",!0),r.optionExternals=new I("externals",!0),r.optionOnlyExported=new I("only-exported",!1),r}return C(t,e),t.isSupported=function(){try{return void 0!==window.localStorage}catch(e){return!1}},t}(u);r(1);i(h,"#tsd-search"),i(v,".menu-highlight"),i(w,".tsd-signatures"),i(_,"a[data-toggle]"),F.isSupported()?i(F,"#tsd-filter"):document.documentElement.classList.add("no-filter");var N=new a;Object.defineProperty(window,"app",{value:N})}]); \ No newline at end of file diff --git a/docs/api/assets/js/search.json b/docs/api/assets/js/search.json index 386a3a9..4785944 100644 --- a/docs/api/assets/js/search.json +++ b/docs/api/assets/js/search.json @@ -1,35 +1 @@ -{ - "kinds": { "1": "Module", "64": "Function" }, - "rows": [ - { - "id": 0, - "kind": 1, - "name": "\"app/index\"", - "url": "modules/_app_index_.html", - "classes": "tsd-kind-module" - }, - { - "id": 1, - "kind": 64, - "name": "hello", - "url": "modules/_app_index_.html#hello", - "classes": "tsd-kind-function tsd-parent-kind-module", - "parent": "\"app/index\"" - } - ], - "index": { - "version": "2.3.9", - "fields": ["name", "parent"], - "fieldVectors": [ - ["name/0", [0, 1.823]], - ["parent/0", []], - ["name/1", [1, 6.931]], - ["parent/1", [0, 0.129]] - ], - "invertedIndex": [ - ["app/index", { "_index": 0, "name": { "0": {} }, "parent": { "1": {} } }], - ["hello", { "_index": 1, "name": { "1": {} }, "parent": {} }] - ], - "pipeline": [] - } -} +{"kinds":{"4":"Enumeration","16":"Enumeration member","32":"Variable","64":"Function","128":"Class","256":"Interface","512":"Constructor","1024":"Property","2048":"Method","65536":"Type literal","2097152":"Object literal","4194304":"Type alias"},"rows":[{"id":0,"kind":2097152,"name":"_","url":"globals.html#_","classes":"tsd-kind-object-literal"},{"id":1,"kind":2097152,"name":"array","url":"globals.html#_.array","classes":"tsd-kind-object-literal tsd-parent-kind-object-literal","parent":"_"},{"id":2,"kind":64,"name":"contains","url":"globals.html#_.array.contains","classes":"tsd-kind-function tsd-parent-kind-object-literal tsd-has-type-parameter","parent":"_.array"},{"id":3,"kind":2097152,"name":"date","url":"globals.html#_.date","classes":"tsd-kind-object-literal tsd-parent-kind-object-literal","parent":"_"},{"id":4,"kind":64,"name":"sortOldToNew","url":"globals.html#_.date.sortoldtonew","classes":"tsd-kind-function tsd-parent-kind-object-literal","parent":"_.date"},{"id":5,"kind":64,"name":"sortNewToOld","url":"globals.html#_.date.sortnewtoold","classes":"tsd-kind-function tsd-parent-kind-object-literal","parent":"_.date"},{"id":6,"kind":256,"name":"AppleInAppPurchaseTransaction","url":"interfaces/appleinapppurchasetransaction.html","classes":"tsd-kind-interface"},{"id":7,"kind":1024,"name":"cancellation_date","url":"interfaces/appleinapppurchasetransaction.html#cancellation_date","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":8,"kind":1024,"name":"cancellation_date_ms","url":"interfaces/appleinapppurchasetransaction.html#cancellation_date_ms","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":9,"kind":1024,"name":"cancellation_date_pst","url":"interfaces/appleinapppurchasetransaction.html#cancellation_date_pst","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":10,"kind":1024,"name":"cancellation_reason","url":"interfaces/appleinapppurchasetransaction.html#cancellation_reason","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":11,"kind":1024,"name":"expires_date","url":"interfaces/appleinapppurchasetransaction.html#expires_date","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":12,"kind":1024,"name":"expires_date_ms","url":"interfaces/appleinapppurchasetransaction.html#expires_date_ms","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":13,"kind":1024,"name":"expires_date_pst","url":"interfaces/appleinapppurchasetransaction.html#expires_date_pst","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":14,"kind":1024,"name":"is_in_intro_offer_period","url":"interfaces/appleinapppurchasetransaction.html#is_in_intro_offer_period","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":15,"kind":1024,"name":"is_trial_period","url":"interfaces/appleinapppurchasetransaction.html#is_trial_period","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":16,"kind":1024,"name":"original_purchase_date","url":"interfaces/appleinapppurchasetransaction.html#original_purchase_date","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":17,"kind":1024,"name":"original_purchase_date_ms","url":"interfaces/appleinapppurchasetransaction.html#original_purchase_date_ms","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":18,"kind":1024,"name":"original_purchase_date_pst","url":"interfaces/appleinapppurchasetransaction.html#original_purchase_date_pst","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":19,"kind":1024,"name":"original_transaction_id","url":"interfaces/appleinapppurchasetransaction.html#original_transaction_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":20,"kind":1024,"name":"product_id","url":"interfaces/appleinapppurchasetransaction.html#product_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":21,"kind":1024,"name":"promotional_offer_id","url":"interfaces/appleinapppurchasetransaction.html#promotional_offer_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":22,"kind":1024,"name":"purchase_date","url":"interfaces/appleinapppurchasetransaction.html#purchase_date","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":23,"kind":1024,"name":"purchase_date_ms","url":"interfaces/appleinapppurchasetransaction.html#purchase_date_ms","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":24,"kind":1024,"name":"purchase_date_pst","url":"interfaces/appleinapppurchasetransaction.html#purchase_date_pst","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":25,"kind":1024,"name":"quantity","url":"interfaces/appleinapppurchasetransaction.html#quantity","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":26,"kind":1024,"name":"transaction_id","url":"interfaces/appleinapppurchasetransaction.html#transaction_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":27,"kind":1024,"name":"web_order_line_item_id","url":"interfaces/appleinapppurchasetransaction.html#web_order_line_item_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleInAppPurchaseTransaction"},{"id":28,"kind":256,"name":"AppleReceipt","url":"interfaces/applereceipt.html","classes":"tsd-kind-interface"},{"id":29,"kind":1024,"name":"adam_id","url":"interfaces/applereceipt.html#adam_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":30,"kind":1024,"name":"app_item_id","url":"interfaces/applereceipt.html#app_item_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":31,"kind":1024,"name":"application_version","url":"interfaces/applereceipt.html#application_version","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":32,"kind":1024,"name":"bundle_id","url":"interfaces/applereceipt.html#bundle_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":33,"kind":1024,"name":"download_id","url":"interfaces/applereceipt.html#download_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":34,"kind":1024,"name":"expiration_date","url":"interfaces/applereceipt.html#expiration_date","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":35,"kind":1024,"name":"expiration_date_ms","url":"interfaces/applereceipt.html#expiration_date_ms","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":36,"kind":1024,"name":"expiration_date_pst","url":"interfaces/applereceipt.html#expiration_date_pst","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":37,"kind":1024,"name":"in_app","url":"interfaces/applereceipt.html#in_app","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":38,"kind":1024,"name":"original_application_version","url":"interfaces/applereceipt.html#original_application_version","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":39,"kind":1024,"name":"original_purchase_date","url":"interfaces/applereceipt.html#original_purchase_date","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":40,"kind":1024,"name":"original_purchase_date_ms","url":"interfaces/applereceipt.html#original_purchase_date_ms","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":41,"kind":1024,"name":"original_purchase_date_pst","url":"interfaces/applereceipt.html#original_purchase_date_pst","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":42,"kind":1024,"name":"preorder_date","url":"interfaces/applereceipt.html#preorder_date","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":43,"kind":1024,"name":"preorder_date_ms","url":"interfaces/applereceipt.html#preorder_date_ms","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":44,"kind":1024,"name":"preorder_date_pst","url":"interfaces/applereceipt.html#preorder_date_pst","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":45,"kind":1024,"name":"receipt_creation_date","url":"interfaces/applereceipt.html#receipt_creation_date","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":46,"kind":1024,"name":"receipt_creation_date_ms","url":"interfaces/applereceipt.html#receipt_creation_date_ms","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":47,"kind":1024,"name":"receipt_creation_date_pst","url":"interfaces/applereceipt.html#receipt_creation_date_pst","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":48,"kind":1024,"name":"receipt_type","url":"interfaces/applereceipt.html#receipt_type","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":49,"kind":1024,"name":"request_date","url":"interfaces/applereceipt.html#request_date","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":50,"kind":1024,"name":"request_date_ms","url":"interfaces/applereceipt.html#request_date_ms","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":51,"kind":1024,"name":"request_date_pst","url":"interfaces/applereceipt.html#request_date_pst","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":52,"kind":1024,"name":"version_external_identifier","url":"interfaces/applereceipt.html#version_external_identifier","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleReceipt"},{"id":53,"kind":4194304,"name":"AppleLatestReceiptInfo","url":"globals.html#applelatestreceiptinfo","classes":"tsd-kind-type-alias"},{"id":54,"kind":256,"name":"ApplePendingRenewalInfo","url":"interfaces/applependingrenewalinfo.html","classes":"tsd-kind-interface"},{"id":55,"kind":1024,"name":"auto_renew_product_id","url":"interfaces/applependingrenewalinfo.html#auto_renew_product_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":56,"kind":1024,"name":"auto_renew_status","url":"interfaces/applependingrenewalinfo.html#auto_renew_status","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":57,"kind":1024,"name":"expiration_intent","url":"interfaces/applependingrenewalinfo.html#expiration_intent","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":58,"kind":1024,"name":"grace_period_expires_date","url":"interfaces/applependingrenewalinfo.html#grace_period_expires_date","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":59,"kind":1024,"name":"grace_period_expires_date_ms","url":"interfaces/applependingrenewalinfo.html#grace_period_expires_date_ms","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":60,"kind":1024,"name":"grace_period_expires_date_pst","url":"interfaces/applependingrenewalinfo.html#grace_period_expires_date_pst","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":61,"kind":1024,"name":"is_in_billing_retry_period","url":"interfaces/applependingrenewalinfo.html#is_in_billing_retry_period","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":62,"kind":1024,"name":"offer_code_ref_name","url":"interfaces/applependingrenewalinfo.html#offer_code_ref_name","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":63,"kind":1024,"name":"original_transaction_id","url":"interfaces/applependingrenewalinfo.html#original_transaction_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":64,"kind":1024,"name":"price_consent_status","url":"interfaces/applependingrenewalinfo.html#price_consent_status","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":65,"kind":1024,"name":"product_id","url":"interfaces/applependingrenewalinfo.html#product_id","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ApplePendingRenewalInfo"},{"id":66,"kind":256,"name":"AppleVerifyReceiptSuccessfulResponse","url":"interfaces/appleverifyreceiptsuccessfulresponse.html","classes":"tsd-kind-interface"},{"id":67,"kind":1024,"name":"environment","url":"interfaces/appleverifyreceiptsuccessfulresponse.html#environment","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleVerifyReceiptSuccessfulResponse"},{"id":68,"kind":1024,"name":"is-retryable","url":"interfaces/appleverifyreceiptsuccessfulresponse.html#is_retryable","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleVerifyReceiptSuccessfulResponse"},{"id":69,"kind":1024,"name":"latest_receipt","url":"interfaces/appleverifyreceiptsuccessfulresponse.html#latest_receipt","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleVerifyReceiptSuccessfulResponse"},{"id":70,"kind":1024,"name":"latest_receipt_info","url":"interfaces/appleverifyreceiptsuccessfulresponse.html#latest_receipt_info","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleVerifyReceiptSuccessfulResponse"},{"id":71,"kind":1024,"name":"pending_renewal_info","url":"interfaces/appleverifyreceiptsuccessfulresponse.html#pending_renewal_info","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleVerifyReceiptSuccessfulResponse"},{"id":72,"kind":1024,"name":"receipt","url":"interfaces/appleverifyreceiptsuccessfulresponse.html#receipt","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleVerifyReceiptSuccessfulResponse"},{"id":73,"kind":1024,"name":"status","url":"interfaces/appleverifyreceiptsuccessfulresponse.html#status","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleVerifyReceiptSuccessfulResponse"},{"id":74,"kind":4194304,"name":"AppleVerifyReceiptSuccessfulStatus","url":"globals.html#appleverifyreceiptsuccessfulstatus","classes":"tsd-kind-type-alias"},{"id":75,"kind":4194304,"name":"AppleVerifyReceiptResponse","url":"globals.html#appleverifyreceiptresponse","classes":"tsd-kind-type-alias"},{"id":76,"kind":4,"name":"AppleVerifyReceiptErrorCode","url":"enums/appleverifyreceipterrorcode.html","classes":"tsd-kind-enum"},{"id":77,"kind":16,"name":"NOT_POST","url":"enums/appleverifyreceipterrorcode.html#not_post","classes":"tsd-kind-enum-member tsd-parent-kind-enum","parent":"AppleVerifyReceiptErrorCode"},{"id":78,"kind":16,"name":"SHOULD_NOT_HAPPEN","url":"enums/appleverifyreceipterrorcode.html#should_not_happen","classes":"tsd-kind-enum-member tsd-parent-kind-enum","parent":"AppleVerifyReceiptErrorCode"},{"id":79,"kind":16,"name":"INVALID_RECEIPT_OR_DOWN","url":"enums/appleverifyreceipterrorcode.html#invalid_receipt_or_down","classes":"tsd-kind-enum-member tsd-parent-kind-enum","parent":"AppleVerifyReceiptErrorCode"},{"id":80,"kind":16,"name":"UNAUTHORIZED","url":"enums/appleverifyreceipterrorcode.html#unauthorized","classes":"tsd-kind-enum-member tsd-parent-kind-enum","parent":"AppleVerifyReceiptErrorCode"},{"id":81,"kind":16,"name":"WRONG_SHARED_SECRET","url":"enums/appleverifyreceipterrorcode.html#wrong_shared_secret","classes":"tsd-kind-enum-member tsd-parent-kind-enum","parent":"AppleVerifyReceiptErrorCode"},{"id":82,"kind":16,"name":"SERVICE_DOWN","url":"enums/appleverifyreceipterrorcode.html#service_down","classes":"tsd-kind-enum-member tsd-parent-kind-enum","parent":"AppleVerifyReceiptErrorCode"},{"id":83,"kind":16,"name":"USE_TEST_ENVIRONMENT","url":"enums/appleverifyreceipterrorcode.html#use_test_environment","classes":"tsd-kind-enum-member tsd-parent-kind-enum","parent":"AppleVerifyReceiptErrorCode"},{"id":84,"kind":16,"name":"USE_PRODUCTION_ENVIRONMENT","url":"enums/appleverifyreceipterrorcode.html#use_production_environment","classes":"tsd-kind-enum-member tsd-parent-kind-enum","parent":"AppleVerifyReceiptErrorCode"},{"id":85,"kind":16,"name":"APPLE_INTERNAL_ERROR","url":"enums/appleverifyreceipterrorcode.html#apple_internal_error","classes":"tsd-kind-enum-member tsd-parent-kind-enum","parent":"AppleVerifyReceiptErrorCode"},{"id":86,"kind":16,"name":"CUSTOMER_NOT_FOUND","url":"enums/appleverifyreceipterrorcode.html#customer_not_found","classes":"tsd-kind-enum-member tsd-parent-kind-enum","parent":"AppleVerifyReceiptErrorCode"},{"id":87,"kind":256,"name":"AppleVerifyReceiptErrorResponse","url":"interfaces/appleverifyreceipterrorresponse.html","classes":"tsd-kind-interface"},{"id":88,"kind":1024,"name":"status","url":"interfaces/appleverifyreceipterrorresponse.html#status","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AppleVerifyReceiptErrorResponse"},{"id":89,"kind":128,"name":"AppleError","url":"classes/appleerror.html","classes":"tsd-kind-class"},{"id":90,"kind":512,"name":"constructor","url":"classes/appleerror.html#constructor","classes":"tsd-kind-constructor tsd-parent-kind-class","parent":"AppleError"},{"id":91,"kind":1024,"name":"appleErrorCode","url":"classes/appleerror.html#appleerrorcode","classes":"tsd-kind-property tsd-parent-kind-class","parent":"AppleError"},{"id":92,"kind":1024,"name":"name","url":"classes/appleerror.html#name","classes":"tsd-kind-property tsd-parent-kind-class tsd-is-inherited","parent":"AppleError"},{"id":93,"kind":1024,"name":"message","url":"classes/appleerror.html#message","classes":"tsd-kind-property tsd-parent-kind-class tsd-is-inherited","parent":"AppleError"},{"id":94,"kind":1024,"name":"stack","url":"classes/appleerror.html#stack","classes":"tsd-kind-property tsd-parent-kind-class tsd-is-inherited","parent":"AppleError"},{"id":95,"kind":1024,"name":"Error","url":"classes/appleerror.html#error","classes":"tsd-kind-property tsd-parent-kind-class tsd-is-static","parent":"AppleError"},{"id":96,"kind":128,"name":"InternalError","url":"classes/internalerror.html","classes":"tsd-kind-class"},{"id":97,"kind":1024,"name":"name","url":"classes/internalerror.html#name","classes":"tsd-kind-property tsd-parent-kind-class","parent":"InternalError"},{"id":98,"kind":1024,"name":"message","url":"classes/internalerror.html#message","classes":"tsd-kind-property tsd-parent-kind-class","parent":"InternalError"},{"id":99,"kind":1024,"name":"stack","url":"classes/internalerror.html#stack","classes":"tsd-kind-property tsd-parent-kind-class","parent":"InternalError"},{"id":100,"kind":512,"name":"constructor","url":"classes/internalerror.html#constructor","classes":"tsd-kind-constructor tsd-parent-kind-class","parent":"InternalError"},{"id":101,"kind":2048,"name":"fromError","url":"classes/internalerror.html#fromerror","classes":"tsd-kind-method tsd-parent-kind-class tsd-is-static","parent":"InternalError"},{"id":102,"kind":256,"name":"AutoRenewableSubscription","url":"interfaces/autorenewablesubscription.html","classes":"tsd-kind-interface"},{"id":103,"kind":1024,"name":"currentProductId","url":"interfaces/autorenewablesubscription.html#currentproductid","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":104,"kind":1024,"name":"originalTransactionId","url":"interfaces/autorenewablesubscription.html#originaltransactionid","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":105,"kind":1024,"name":"subscriptionGroup","url":"interfaces/autorenewablesubscription.html#subscriptiongroup","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":106,"kind":1024,"name":"isInFreeTrialPeriod","url":"interfaces/autorenewablesubscription.html#isinfreetrialperiod","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":107,"kind":1024,"name":"isEligibleIntroductoryOffer","url":"interfaces/autorenewablesubscription.html#iseligibleintroductoryoffer","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":108,"kind":1024,"name":"currentSubscriptionPeriodExpires","url":"interfaces/autorenewablesubscription.html#currentsubscriptionperiodexpires","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":109,"kind":1024,"name":"inactiveStatus","url":"interfaces/autorenewablesubscription.html#inactivestatus","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":110,"kind":1024,"name":"usedOfferCodes","url":"interfaces/autorenewablesubscription.html#usedoffercodes","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":111,"kind":1024,"name":"usedPromotionalOffers","url":"interfaces/autorenewablesubscription.html#usedpromotionaloffers","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":112,"kind":1024,"name":"allTransactions","url":"interfaces/autorenewablesubscription.html#alltransactions","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":113,"kind":1024,"name":"mostRecentTransaction","url":"interfaces/autorenewablesubscription.html#mostrecenttransaction","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":114,"kind":1024,"name":"willAutoRenew","url":"interfaces/autorenewablesubscription.html#willautorenew","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":115,"kind":1024,"name":"willDowngradeToProductId","url":"interfaces/autorenewablesubscription.html#willdowngradetoproductid","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":116,"kind":1024,"name":"actionableStatuses","url":"interfaces/autorenewablesubscription.html#actionablestatuses","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":117,"kind":1024,"name":"priceIncreaseAccepted","url":"interfaces/autorenewablesubscription.html#priceincreaseaccepted","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":118,"kind":1024,"name":"gracePeriod","url":"interfaces/autorenewablesubscription.html#graceperiod","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"AutoRenewableSubscription"},{"id":119,"kind":4194304,"name":"AutoRenewableSubscriptionRefundReason","url":"globals.html#autorenewablesubscriptionrefundreason","classes":"tsd-kind-type-alias"},{"id":120,"kind":4194304,"name":"AutoRenewableSubscriptionTransaction","url":"globals.html#autorenewablesubscriptiontransaction","classes":"tsd-kind-type-alias"},{"id":121,"kind":65536,"name":"__type","url":"globals.html#autorenewablesubscriptiontransaction.__type","classes":"tsd-kind-type-literal tsd-parent-kind-type-alias","parent":"AutoRenewableSubscriptionTransaction"},{"id":122,"kind":32,"name":"productId","url":"globals.html#autorenewablesubscriptiontransaction.__type.productid","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":123,"kind":32,"name":"refundedDate","url":"globals.html#autorenewablesubscriptiontransaction.__type.refundeddate","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":124,"kind":32,"name":"refundReason","url":"globals.html#autorenewablesubscriptiontransaction.__type.refundreason","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":125,"kind":32,"name":"expiresDate","url":"globals.html#autorenewablesubscriptiontransaction.__type.expiresdate","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":126,"kind":32,"name":"inIntroPeriod","url":"globals.html#autorenewablesubscriptiontransaction.__type.inintroperiod","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":127,"kind":32,"name":"inTrialPeriod","url":"globals.html#autorenewablesubscriptiontransaction.__type.intrialperiod","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":128,"kind":32,"name":"isUpgradeTransaction","url":"globals.html#autorenewablesubscriptiontransaction.__type.isupgradetransaction","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":129,"kind":32,"name":"offerCodeRefName","url":"globals.html#autorenewablesubscriptiontransaction.__type.offercoderefname","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":130,"kind":32,"name":"originalPurchaseDate","url":"globals.html#autorenewablesubscriptiontransaction.__type.originalpurchasedate","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":131,"kind":32,"name":"originalTransactionId","url":"globals.html#autorenewablesubscriptiontransaction.__type.originaltransactionid","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":132,"kind":32,"name":"promotionalOfferId","url":"globals.html#autorenewablesubscriptiontransaction.__type.promotionalofferid","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":133,"kind":32,"name":"purchaseDate","url":"globals.html#autorenewablesubscriptiontransaction.__type.purchasedate","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":134,"kind":32,"name":"subscriptionGroupId","url":"globals.html#autorenewablesubscriptiontransaction.__type.subscriptiongroupid","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":135,"kind":32,"name":"transactionId","url":"globals.html#autorenewablesubscriptiontransaction.__type.transactionid","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":136,"kind":32,"name":"webOrderLineItemId","url":"globals.html#autorenewablesubscriptiontransaction.__type.weborderlineitemid","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"AutoRenewableSubscriptionTransaction.__type"},{"id":137,"kind":4194304,"name":"AutoRenewableSubscriptionActionableStatus","url":"globals.html#autorenewablesubscriptionactionablestatus","classes":"tsd-kind-type-alias"},{"id":138,"kind":4194304,"name":"AutoRenewableSubscriptionInactiveStatus","url":"globals.html#autorenewablesubscriptioninactivestatus","classes":"tsd-kind-type-alias"},{"id":139,"kind":256,"name":"ProductPurchaseTransaction","url":"interfaces/productpurchasetransaction.html","classes":"tsd-kind-interface"},{"id":140,"kind":1024,"name":"quantity","url":"interfaces/productpurchasetransaction.html#quantity","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ProductPurchaseTransaction"},{"id":141,"kind":1024,"name":"transactionId","url":"interfaces/productpurchasetransaction.html#transactionid","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ProductPurchaseTransaction"},{"id":142,"kind":1024,"name":"purchaseDate","url":"interfaces/productpurchasetransaction.html#purchasedate","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ProductPurchaseTransaction"},{"id":143,"kind":256,"name":"ProductPurchase","url":"interfaces/productpurchase.html","classes":"tsd-kind-interface"},{"id":144,"kind":1024,"name":"productId","url":"interfaces/productpurchase.html#productid","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ProductPurchase"},{"id":145,"kind":1024,"name":"transactions","url":"interfaces/productpurchase.html#transactions","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ProductPurchase"},{"id":146,"kind":256,"name":"ParsedReceipt","url":"interfaces/parsedreceipt.html","classes":"tsd-kind-interface"},{"id":147,"kind":1024,"name":"environment","url":"interfaces/parsedreceipt.html#environment","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ParsedReceipt"},{"id":148,"kind":1024,"name":"auto_renewable_subscriptions","url":"interfaces/parsedreceipt.html#auto_renewable_subscriptions","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ParsedReceipt"},{"id":149,"kind":1024,"name":"product_purchases","url":"interfaces/parsedreceipt.html#product_purchases","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ParsedReceipt"},{"id":150,"kind":1024,"name":"rawResponse","url":"interfaces/parsedreceipt.html#rawresponse","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ParsedReceipt"},{"id":151,"kind":1024,"name":"latestReceipt","url":"interfaces/parsedreceipt.html#latestreceipt","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"ParsedReceipt"},{"id":152,"kind":64,"name":"parseSuccess","url":"globals.html#parsesuccess","classes":"tsd-kind-function"},{"id":153,"kind":256,"name":"VerifyReceiptOptions","url":"interfaces/verifyreceiptoptions.html","classes":"tsd-kind-interface"},{"id":154,"kind":1024,"name":"production","url":"interfaces/verifyreceiptoptions.html#production","classes":"tsd-kind-property tsd-parent-kind-interface","parent":"VerifyReceiptOptions"},{"id":155,"kind":32,"name":"maxNumberHttpAttempts","url":"globals.html#maxnumberhttpattempts","classes":"tsd-kind-variable"},{"id":156,"kind":64,"name":"runVerifyReceipt","url":"globals.html#runverifyreceipt","classes":"tsd-kind-function"},{"id":157,"kind":4194304,"name":"VerifyReceiptFailure","url":"globals.html#verifyreceiptfailure","classes":"tsd-kind-type-alias"},{"id":158,"kind":65536,"name":"__type","url":"globals.html#verifyreceiptfailure.__type","classes":"tsd-kind-type-literal tsd-parent-kind-type-alias","parent":"VerifyReceiptFailure"},{"id":159,"kind":32,"name":"error","url":"globals.html#verifyreceiptfailure.__type.error","classes":"tsd-kind-variable tsd-parent-kind-type-literal","parent":"VerifyReceiptFailure.__type"},{"id":160,"kind":4194304,"name":"VerifyReceiptResponse","url":"globals.html#verifyreceiptresponse","classes":"tsd-kind-type-alias"},{"id":161,"kind":64,"name":"verifyReceipt","url":"globals.html#verifyreceipt","classes":"tsd-kind-function"},{"id":162,"kind":64,"name":"isFailure","url":"globals.html#isfailure","classes":"tsd-kind-function"}],"index":{"version":"2.3.9","fields":["name","parent"],"fieldVectors":[["name/0",[0,38.567]],["parent/0",[]],["name/1",[1,47.061]],["parent/1",[0,3.546]],["name/2",[2,47.061]],["parent/2",[3,4.327]],["name/3",[4,47.061]],["parent/3",[0,3.546]],["name/4",[5,47.061]],["parent/4",[6,3.856]],["name/5",[7,47.061]],["parent/5",[6,3.856]],["name/6",[8,19.913]],["parent/6",[]],["name/7",[9,47.061]],["parent/7",[8,1.831]],["name/8",[10,47.061]],["parent/8",[8,1.831]],["name/9",[11,47.061]],["parent/9",[8,1.831]],["name/10",[12,47.061]],["parent/10",[8,1.831]],["name/11",[13,47.061]],["parent/11",[8,1.831]],["name/12",[14,47.061]],["parent/12",[8,1.831]],["name/13",[15,47.061]],["parent/13",[8,1.831]],["name/14",[16,47.061]],["parent/14",[8,1.831]],["name/15",[17,47.061]],["parent/15",[8,1.831]],["name/16",[18,41.94]],["parent/16",[8,1.831]],["name/17",[19,41.94]],["parent/17",[8,1.831]],["name/18",[20,41.94]],["parent/18",[8,1.831]],["name/19",[21,41.94]],["parent/19",[8,1.831]],["name/20",[22,41.94]],["parent/20",[8,1.831]],["name/21",[23,47.061]],["parent/21",[8,1.831]],["name/22",[24,47.061]],["parent/22",[8,1.831]],["name/23",[25,47.061]],["parent/23",[8,1.831]],["name/24",[26,47.061]],["parent/24",[8,1.831]],["name/25",[27,41.94]],["parent/25",[8,1.831]],["name/26",[28,47.061]],["parent/26",[8,1.831]],["name/27",[29,47.061]],["parent/27",[8,1.831]],["name/28",[30,18.658]],["parent/28",[]],["name/29",[31,47.061]],["parent/29",[30,1.716]],["name/30",[32,47.061]],["parent/30",[30,1.716]],["name/31",[33,47.061]],["parent/31",[30,1.716]],["name/32",[34,47.061]],["parent/32",[30,1.716]],["name/33",[35,47.061]],["parent/33",[30,1.716]],["name/34",[36,47.061]],["parent/34",[30,1.716]],["name/35",[37,47.061]],["parent/35",[30,1.716]],["name/36",[38,47.061]],["parent/36",[30,1.716]],["name/37",[39,47.061]],["parent/37",[30,1.716]],["name/38",[40,47.061]],["parent/38",[30,1.716]],["name/39",[18,41.94]],["parent/39",[30,1.716]],["name/40",[19,41.94]],["parent/40",[30,1.716]],["name/41",[20,41.94]],["parent/41",[30,1.716]],["name/42",[41,47.061]],["parent/42",[30,1.716]],["name/43",[42,47.061]],["parent/43",[30,1.716]],["name/44",[43,47.061]],["parent/44",[30,1.716]],["name/45",[44,47.061]],["parent/45",[30,1.716]],["name/46",[45,47.061]],["parent/46",[30,1.716]],["name/47",[46,47.061]],["parent/47",[30,1.716]],["name/48",[47,47.061]],["parent/48",[30,1.716]],["name/49",[48,47.061]],["parent/49",[30,1.716]],["name/50",[49,47.061]],["parent/50",[30,1.716]],["name/51",[50,47.061]],["parent/51",[30,1.716]],["name/52",[51,47.061]],["parent/52",[30,1.716]],["name/53",[52,47.061]],["parent/53",[]],["name/54",[53,25.806]],["parent/54",[]],["name/55",[54,47.061]],["parent/55",[53,2.373]],["name/56",[55,47.061]],["parent/56",[53,2.373]],["name/57",[56,47.061]],["parent/57",[53,2.373]],["name/58",[57,47.061]],["parent/58",[53,2.373]],["name/59",[58,47.061]],["parent/59",[53,2.373]],["name/60",[59,47.061]],["parent/60",[53,2.373]],["name/61",[60,47.061]],["parent/61",[53,2.373]],["name/62",[61,47.061]],["parent/62",[53,2.373]],["name/63",[21,41.94]],["parent/63",[53,2.373]],["name/64",[62,47.061]],["parent/64",[53,2.373]],["name/65",[22,41.94]],["parent/65",[53,2.373]],["name/66",[63,29.672]],["parent/66",[]],["name/67",[64,41.94]],["parent/67",[63,2.728]],["name/68",[65,33.433,66,33.433]],["parent/68",[63,2.728]],["name/69",[67,47.061]],["parent/69",[63,2.728]],["name/70",[68,47.061]],["parent/70",[63,2.728]],["name/71",[69,47.061]],["parent/71",[63,2.728]],["name/72",[70,47.061]],["parent/72",[63,2.728]],["name/73",[71,41.94]],["parent/73",[63,2.728]],["name/74",[72,47.061]],["parent/74",[]],["name/75",[73,47.061]],["parent/75",[]],["name/76",[74,26.642]],["parent/76",[]],["name/77",[75,47.061]],["parent/77",[74,2.45]],["name/78",[76,47.061]],["parent/78",[74,2.45]],["name/79",[77,47.061]],["parent/79",[74,2.45]],["name/80",[78,47.061]],["parent/80",[74,2.45]],["name/81",[79,47.061]],["parent/81",[74,2.45]],["name/82",[80,47.061]],["parent/82",[74,2.45]],["name/83",[81,47.061]],["parent/83",[74,2.45]],["name/84",[82,47.061]],["parent/84",[74,2.45]],["name/85",[83,47.061]],["parent/85",[74,2.45]],["name/86",[84,47.061]],["parent/86",[74,2.45]],["name/87",[85,41.94]],["parent/87",[]],["name/88",[71,41.94]],["parent/88",[85,3.856]],["name/89",[86,30.927]],["parent/89",[]],["name/90",[87,41.94]],["parent/90",[86,2.844]],["name/91",[88,47.061]],["parent/91",[86,2.844]],["name/92",[89,41.94]],["parent/92",[86,2.844]],["name/93",[90,41.94]],["parent/93",[86,2.844]],["name/94",[91,41.94]],["parent/94",[86,2.844]],["name/95",[92,41.94]],["parent/95",[86,2.844]],["name/96",[93,32.361]],["parent/96",[]],["name/97",[89,41.94]],["parent/97",[93,2.976]],["name/98",[90,41.94]],["parent/98",[93,2.976]],["name/99",[91,41.94]],["parent/99",[93,2.976]],["name/100",[87,41.94]],["parent/100",[93,2.976]],["name/101",[94,47.061]],["parent/101",[93,2.976]],["name/102",[95,22.433]],["parent/102",[]],["name/103",[96,47.061]],["parent/103",[95,2.063]],["name/104",[97,41.94]],["parent/104",[95,2.063]],["name/105",[98,47.061]],["parent/105",[95,2.063]],["name/106",[99,47.061]],["parent/106",[95,2.063]],["name/107",[100,47.061]],["parent/107",[95,2.063]],["name/108",[101,47.061]],["parent/108",[95,2.063]],["name/109",[102,47.061]],["parent/109",[95,2.063]],["name/110",[103,47.061]],["parent/110",[95,2.063]],["name/111",[104,47.061]],["parent/111",[95,2.063]],["name/112",[105,47.061]],["parent/112",[95,2.063]],["name/113",[106,47.061]],["parent/113",[95,2.063]],["name/114",[107,47.061]],["parent/114",[95,2.063]],["name/115",[108,47.061]],["parent/115",[95,2.063]],["name/116",[109,47.061]],["parent/116",[95,2.063]],["name/117",[110,47.061]],["parent/117",[95,2.063]],["name/118",[111,47.061]],["parent/118",[95,2.063]],["name/119",[112,47.061]],["parent/119",[]],["name/120",[113,41.94]],["parent/120",[]],["name/121",[114,41.94]],["parent/121",[113,3.856]],["name/122",[115,41.94]],["parent/122",[116,2.175]],["name/123",[117,47.061]],["parent/123",[116,2.175]],["name/124",[118,47.061]],["parent/124",[116,2.175]],["name/125",[119,47.061]],["parent/125",[116,2.175]],["name/126",[120,47.061]],["parent/126",[116,2.175]],["name/127",[121,47.061]],["parent/127",[116,2.175]],["name/128",[122,47.061]],["parent/128",[116,2.175]],["name/129",[123,47.061]],["parent/129",[116,2.175]],["name/130",[124,47.061]],["parent/130",[116,2.175]],["name/131",[97,41.94]],["parent/131",[116,2.175]],["name/132",[125,47.061]],["parent/132",[116,2.175]],["name/133",[126,41.94]],["parent/133",[116,2.175]],["name/134",[127,47.061]],["parent/134",[116,2.175]],["name/135",[128,41.94]],["parent/135",[116,2.175]],["name/136",[129,47.061]],["parent/136",[116,2.175]],["name/137",[130,47.061]],["parent/137",[]],["name/138",[131,47.061]],["parent/138",[]],["name/139",[132,36.048]],["parent/139",[]],["name/140",[27,41.94]],["parent/140",[132,3.315]],["name/141",[128,41.94]],["parent/141",[132,3.315]],["name/142",[126,41.94]],["parent/142",[132,3.315]],["name/143",[133,38.567]],["parent/143",[]],["name/144",[115,41.94]],["parent/144",[133,3.546]],["name/145",[134,47.061]],["parent/145",[133,3.546]],["name/146",[135,32.361]],["parent/146",[]],["name/147",[64,41.94]],["parent/147",[135,2.976]],["name/148",[136,47.061]],["parent/148",[135,2.976]],["name/149",[137,47.061]],["parent/149",[135,2.976]],["name/150",[138,47.061]],["parent/150",[135,2.976]],["name/151",[139,47.061]],["parent/151",[135,2.976]],["name/152",[140,47.061]],["parent/152",[]],["name/153",[141,41.94]],["parent/153",[]],["name/154",[142,47.061]],["parent/154",[141,3.856]],["name/155",[143,47.061]],["parent/155",[]],["name/156",[144,47.061]],["parent/156",[]],["name/157",[145,41.94]],["parent/157",[]],["name/158",[114,41.94]],["parent/158",[145,3.856]],["name/159",[92,41.94]],["parent/159",[146,4.327]],["name/160",[147,47.061]],["parent/160",[]],["name/161",[148,47.061]],["parent/161",[]],["name/162",[149,47.061]],["parent/162",[]]],"invertedIndex":[["_",{"_index":0,"name":{"0":{}},"parent":{"1":{},"3":{}}}],["_.array",{"_index":3,"name":{},"parent":{"2":{}}}],["_.date",{"_index":6,"name":{},"parent":{"4":{},"5":{}}}],["__type",{"_index":114,"name":{"121":{},"158":{}},"parent":{}}],["actionablestatuses",{"_index":109,"name":{"116":{}},"parent":{}}],["adam_id",{"_index":31,"name":{"29":{}},"parent":{}}],["alltransactions",{"_index":105,"name":{"112":{}},"parent":{}}],["app_item_id",{"_index":32,"name":{"30":{}},"parent":{}}],["apple_internal_error",{"_index":83,"name":{"85":{}},"parent":{}}],["appleerror",{"_index":86,"name":{"89":{}},"parent":{"90":{},"91":{},"92":{},"93":{},"94":{},"95":{}}}],["appleerrorcode",{"_index":88,"name":{"91":{}},"parent":{}}],["appleinapppurchasetransaction",{"_index":8,"name":{"6":{}},"parent":{"7":{},"8":{},"9":{},"10":{},"11":{},"12":{},"13":{},"14":{},"15":{},"16":{},"17":{},"18":{},"19":{},"20":{},"21":{},"22":{},"23":{},"24":{},"25":{},"26":{},"27":{}}}],["applelatestreceiptinfo",{"_index":52,"name":{"53":{}},"parent":{}}],["applependingrenewalinfo",{"_index":53,"name":{"54":{}},"parent":{"55":{},"56":{},"57":{},"58":{},"59":{},"60":{},"61":{},"62":{},"63":{},"64":{},"65":{}}}],["applereceipt",{"_index":30,"name":{"28":{}},"parent":{"29":{},"30":{},"31":{},"32":{},"33":{},"34":{},"35":{},"36":{},"37":{},"38":{},"39":{},"40":{},"41":{},"42":{},"43":{},"44":{},"45":{},"46":{},"47":{},"48":{},"49":{},"50":{},"51":{},"52":{}}}],["appleverifyreceipterrorcode",{"_index":74,"name":{"76":{}},"parent":{"77":{},"78":{},"79":{},"80":{},"81":{},"82":{},"83":{},"84":{},"85":{},"86":{}}}],["appleverifyreceipterrorresponse",{"_index":85,"name":{"87":{}},"parent":{"88":{}}}],["appleverifyreceiptresponse",{"_index":73,"name":{"75":{}},"parent":{}}],["appleverifyreceiptsuccessfulresponse",{"_index":63,"name":{"66":{}},"parent":{"67":{},"68":{},"69":{},"70":{},"71":{},"72":{},"73":{}}}],["appleverifyreceiptsuccessfulstatus",{"_index":72,"name":{"74":{}},"parent":{}}],["application_version",{"_index":33,"name":{"31":{}},"parent":{}}],["array",{"_index":1,"name":{"1":{}},"parent":{}}],["auto_renew_product_id",{"_index":54,"name":{"55":{}},"parent":{}}],["auto_renew_status",{"_index":55,"name":{"56":{}},"parent":{}}],["auto_renewable_subscriptions",{"_index":136,"name":{"148":{}},"parent":{}}],["autorenewablesubscription",{"_index":95,"name":{"102":{}},"parent":{"103":{},"104":{},"105":{},"106":{},"107":{},"108":{},"109":{},"110":{},"111":{},"112":{},"113":{},"114":{},"115":{},"116":{},"117":{},"118":{}}}],["autorenewablesubscriptionactionablestatus",{"_index":130,"name":{"137":{}},"parent":{}}],["autorenewablesubscriptioninactivestatus",{"_index":131,"name":{"138":{}},"parent":{}}],["autorenewablesubscriptionrefundreason",{"_index":112,"name":{"119":{}},"parent":{}}],["autorenewablesubscriptiontransaction",{"_index":113,"name":{"120":{}},"parent":{"121":{}}}],["autorenewablesubscriptiontransaction.__type",{"_index":116,"name":{},"parent":{"122":{},"123":{},"124":{},"125":{},"126":{},"127":{},"128":{},"129":{},"130":{},"131":{},"132":{},"133":{},"134":{},"135":{},"136":{}}}],["bundle_id",{"_index":34,"name":{"32":{}},"parent":{}}],["cancellation_date",{"_index":9,"name":{"7":{}},"parent":{}}],["cancellation_date_ms",{"_index":10,"name":{"8":{}},"parent":{}}],["cancellation_date_pst",{"_index":11,"name":{"9":{}},"parent":{}}],["cancellation_reason",{"_index":12,"name":{"10":{}},"parent":{}}],["constructor",{"_index":87,"name":{"90":{},"100":{}},"parent":{}}],["contains",{"_index":2,"name":{"2":{}},"parent":{}}],["currentproductid",{"_index":96,"name":{"103":{}},"parent":{}}],["currentsubscriptionperiodexpires",{"_index":101,"name":{"108":{}},"parent":{}}],["customer_not_found",{"_index":84,"name":{"86":{}},"parent":{}}],["date",{"_index":4,"name":{"3":{}},"parent":{}}],["download_id",{"_index":35,"name":{"33":{}},"parent":{}}],["environment",{"_index":64,"name":{"67":{},"147":{}},"parent":{}}],["error",{"_index":92,"name":{"95":{},"159":{}},"parent":{}}],["expiration_date",{"_index":36,"name":{"34":{}},"parent":{}}],["expiration_date_ms",{"_index":37,"name":{"35":{}},"parent":{}}],["expiration_date_pst",{"_index":38,"name":{"36":{}},"parent":{}}],["expiration_intent",{"_index":56,"name":{"57":{}},"parent":{}}],["expires_date",{"_index":13,"name":{"11":{}},"parent":{}}],["expires_date_ms",{"_index":14,"name":{"12":{}},"parent":{}}],["expires_date_pst",{"_index":15,"name":{"13":{}},"parent":{}}],["expiresdate",{"_index":119,"name":{"125":{}},"parent":{}}],["fromerror",{"_index":94,"name":{"101":{}},"parent":{}}],["grace_period_expires_date",{"_index":57,"name":{"58":{}},"parent":{}}],["grace_period_expires_date_ms",{"_index":58,"name":{"59":{}},"parent":{}}],["grace_period_expires_date_pst",{"_index":59,"name":{"60":{}},"parent":{}}],["graceperiod",{"_index":111,"name":{"118":{}},"parent":{}}],["in_app",{"_index":39,"name":{"37":{}},"parent":{}}],["inactivestatus",{"_index":102,"name":{"109":{}},"parent":{}}],["inintroperiod",{"_index":120,"name":{"126":{}},"parent":{}}],["internalerror",{"_index":93,"name":{"96":{}},"parent":{"97":{},"98":{},"99":{},"100":{},"101":{}}}],["intrialperiod",{"_index":121,"name":{"127":{}},"parent":{}}],["invalid_receipt_or_down",{"_index":77,"name":{"79":{}},"parent":{}}],["is",{"_index":65,"name":{"68":{}},"parent":{}}],["is_in_billing_retry_period",{"_index":60,"name":{"61":{}},"parent":{}}],["is_in_intro_offer_period",{"_index":16,"name":{"14":{}},"parent":{}}],["is_trial_period",{"_index":17,"name":{"15":{}},"parent":{}}],["iseligibleintroductoryoffer",{"_index":100,"name":{"107":{}},"parent":{}}],["isfailure",{"_index":149,"name":{"162":{}},"parent":{}}],["isinfreetrialperiod",{"_index":99,"name":{"106":{}},"parent":{}}],["isupgradetransaction",{"_index":122,"name":{"128":{}},"parent":{}}],["latest_receipt",{"_index":67,"name":{"69":{}},"parent":{}}],["latest_receipt_info",{"_index":68,"name":{"70":{}},"parent":{}}],["latestreceipt",{"_index":139,"name":{"151":{}},"parent":{}}],["maxnumberhttpattempts",{"_index":143,"name":{"155":{}},"parent":{}}],["message",{"_index":90,"name":{"93":{},"98":{}},"parent":{}}],["mostrecenttransaction",{"_index":106,"name":{"113":{}},"parent":{}}],["name",{"_index":89,"name":{"92":{},"97":{}},"parent":{}}],["not_post",{"_index":75,"name":{"77":{}},"parent":{}}],["offer_code_ref_name",{"_index":61,"name":{"62":{}},"parent":{}}],["offercoderefname",{"_index":123,"name":{"129":{}},"parent":{}}],["original_application_version",{"_index":40,"name":{"38":{}},"parent":{}}],["original_purchase_date",{"_index":18,"name":{"16":{},"39":{}},"parent":{}}],["original_purchase_date_ms",{"_index":19,"name":{"17":{},"40":{}},"parent":{}}],["original_purchase_date_pst",{"_index":20,"name":{"18":{},"41":{}},"parent":{}}],["original_transaction_id",{"_index":21,"name":{"19":{},"63":{}},"parent":{}}],["originalpurchasedate",{"_index":124,"name":{"130":{}},"parent":{}}],["originaltransactionid",{"_index":97,"name":{"104":{},"131":{}},"parent":{}}],["parsedreceipt",{"_index":135,"name":{"146":{}},"parent":{"147":{},"148":{},"149":{},"150":{},"151":{}}}],["parsesuccess",{"_index":140,"name":{"152":{}},"parent":{}}],["pending_renewal_info",{"_index":69,"name":{"71":{}},"parent":{}}],["preorder_date",{"_index":41,"name":{"42":{}},"parent":{}}],["preorder_date_ms",{"_index":42,"name":{"43":{}},"parent":{}}],["preorder_date_pst",{"_index":43,"name":{"44":{}},"parent":{}}],["price_consent_status",{"_index":62,"name":{"64":{}},"parent":{}}],["priceincreaseaccepted",{"_index":110,"name":{"117":{}},"parent":{}}],["product_id",{"_index":22,"name":{"20":{},"65":{}},"parent":{}}],["product_purchases",{"_index":137,"name":{"149":{}},"parent":{}}],["productid",{"_index":115,"name":{"122":{},"144":{}},"parent":{}}],["production",{"_index":142,"name":{"154":{}},"parent":{}}],["productpurchase",{"_index":133,"name":{"143":{}},"parent":{"144":{},"145":{}}}],["productpurchasetransaction",{"_index":132,"name":{"139":{}},"parent":{"140":{},"141":{},"142":{}}}],["promotional_offer_id",{"_index":23,"name":{"21":{}},"parent":{}}],["promotionalofferid",{"_index":125,"name":{"132":{}},"parent":{}}],["purchase_date",{"_index":24,"name":{"22":{}},"parent":{}}],["purchase_date_ms",{"_index":25,"name":{"23":{}},"parent":{}}],["purchase_date_pst",{"_index":26,"name":{"24":{}},"parent":{}}],["purchasedate",{"_index":126,"name":{"133":{},"142":{}},"parent":{}}],["quantity",{"_index":27,"name":{"25":{},"140":{}},"parent":{}}],["rawresponse",{"_index":138,"name":{"150":{}},"parent":{}}],["receipt",{"_index":70,"name":{"72":{}},"parent":{}}],["receipt_creation_date",{"_index":44,"name":{"45":{}},"parent":{}}],["receipt_creation_date_ms",{"_index":45,"name":{"46":{}},"parent":{}}],["receipt_creation_date_pst",{"_index":46,"name":{"47":{}},"parent":{}}],["receipt_type",{"_index":47,"name":{"48":{}},"parent":{}}],["refundeddate",{"_index":117,"name":{"123":{}},"parent":{}}],["refundreason",{"_index":118,"name":{"124":{}},"parent":{}}],["request_date",{"_index":48,"name":{"49":{}},"parent":{}}],["request_date_ms",{"_index":49,"name":{"50":{}},"parent":{}}],["request_date_pst",{"_index":50,"name":{"51":{}},"parent":{}}],["retryable",{"_index":66,"name":{"68":{}},"parent":{}}],["runverifyreceipt",{"_index":144,"name":{"156":{}},"parent":{}}],["service_down",{"_index":80,"name":{"82":{}},"parent":{}}],["should_not_happen",{"_index":76,"name":{"78":{}},"parent":{}}],["sortnewtoold",{"_index":7,"name":{"5":{}},"parent":{}}],["sortoldtonew",{"_index":5,"name":{"4":{}},"parent":{}}],["stack",{"_index":91,"name":{"94":{},"99":{}},"parent":{}}],["status",{"_index":71,"name":{"73":{},"88":{}},"parent":{}}],["subscriptiongroup",{"_index":98,"name":{"105":{}},"parent":{}}],["subscriptiongroupid",{"_index":127,"name":{"134":{}},"parent":{}}],["transaction_id",{"_index":28,"name":{"26":{}},"parent":{}}],["transactionid",{"_index":128,"name":{"135":{},"141":{}},"parent":{}}],["transactions",{"_index":134,"name":{"145":{}},"parent":{}}],["unauthorized",{"_index":78,"name":{"80":{}},"parent":{}}],["use_production_environment",{"_index":82,"name":{"84":{}},"parent":{}}],["use_test_environment",{"_index":81,"name":{"83":{}},"parent":{}}],["usedoffercodes",{"_index":103,"name":{"110":{}},"parent":{}}],["usedpromotionaloffers",{"_index":104,"name":{"111":{}},"parent":{}}],["verifyreceipt",{"_index":148,"name":{"161":{}},"parent":{}}],["verifyreceiptfailure",{"_index":145,"name":{"157":{}},"parent":{"158":{}}}],["verifyreceiptfailure.__type",{"_index":146,"name":{},"parent":{"159":{}}}],["verifyreceiptoptions",{"_index":141,"name":{"153":{}},"parent":{"154":{}}}],["verifyreceiptresponse",{"_index":147,"name":{"160":{}},"parent":{}}],["version_external_identifier",{"_index":51,"name":{"52":{}},"parent":{}}],["web_order_line_item_id",{"_index":29,"name":{"27":{}},"parent":{}}],["weborderlineitemid",{"_index":129,"name":{"136":{}},"parent":{}}],["willautorenew",{"_index":107,"name":{"114":{}},"parent":{}}],["willdowngradetoproductid",{"_index":108,"name":{"115":{}},"parent":{}}],["wrong_shared_secret",{"_index":79,"name":{"81":{}},"parent":{}}]],"pipeline":[]}} \ No newline at end of file diff --git a/docs/api/classes/appleerror.html b/docs/api/classes/appleerror.html new file mode 100644 index 0000000..b49cab5 --- /dev/null +++ b/docs/api/classes/appleerror.html @@ -0,0 +1,364 @@ + + + + + + AppleError | dollabill-apple + + + + + +
+
+
+
+ +
+
+ Options +
+
+ All +
    +
  • Public
  • +
  • Public/Protected
  • +
  • All
  • +
+
+ + + + + + +
+
+ Menu +
+
+
+
+
+ +
+
+
+
+
+

An error came back from Apple.

+
+
+
+
+

Hierarchy

+
    +
  • + Error +
      +
    • + AppleError +
    • +
    +
  • +
+
+
+

Index

+
+
+
+

Constructors

+ +
+
+

Properties

+ +
+
+
+
+
+

Constructors

+
+ +

constructor

+ + +
+
+
+

Properties

+
+ +

appleErrorCode

+ + +
+
+ +

message

+
message: string
+ +
+
+ +

name

+
name: string
+ +
+
+ +

Optional stack

+
stack: undefined | string
+ +
+
+ +

Static Error

+
Error: ErrorConstructor
+ +
+
+
+
+

Legend

+
+
    +
  • Constructor
  • +
  • Property
  • +
+
    +
  • Static property
  • +
  • Static method
  • +
+
    +
  • Property
  • +
+
    +
  • Inherited property
  • +
+
+
+
+
+

Generated using TypeDoc

+
+
+
+ + + \ No newline at end of file diff --git a/docs/api/classes/internalerror.html b/docs/api/classes/internalerror.html new file mode 100644 index 0000000..83bab5a --- /dev/null +++ b/docs/api/classes/internalerror.html @@ -0,0 +1,376 @@ + + + + + + InternalError | dollabill-apple + + + + + +
+
+
+
+ +
+
+ Options +
+
+ All +
    +
  • Public
  • +
  • Public/Protected
  • +
  • All
  • +
+
+ + + + + + +
+
+ Menu +
+
+
+
+
+ +
+
+
+
+
+

Goals of this class:

+
    +
  1. Prompt the developer to file a bug report.
  2. +
  3. Include in this error all of the information from the given error so we can fix the issue.
  4. +
+
+
+
+
+

Hierarchy

+
    +
  • + InternalError +
  • +
+
+
+

Implements

+ +
+
+

Index

+
+
+
+

Constructors

+ +
+
+

Properties

+ +
+
+

Methods

+ +
+
+
+
+
+

Constructors

+
+ +

constructor

+
    +
  • new InternalError(name: string, message: string, stack?: undefined | string): InternalError
  • +
+
    +
  • + +

    Parameters

    +
      +
    • +
      name: string
      +
    • +
    • +
      message: string
      +
    • +
    • +
      Optional stack: undefined | string
      +
    • +
    +

    Returns InternalError

    +
  • +
+
+
+
+

Properties

+
+ +

message

+
message: string
+ +
+
+ +

name

+
name: string
+ +
+
+ +

stack

+
stack: string
+ +
+
+
+

Methods

+
+ +

Static fromError

+ +
    +
  • + +

    Parameters

    + +

    Returns InternalError

    +
  • +
+
+
+
+
+

Legend

+
+
    +
  • Constructor
  • +
  • Property
  • +
+
    +
  • Static property
  • +
  • Static method
  • +
+
    +
  • Property
  • +
+
    +
  • Inherited property
  • +
+
+
+
+
+

Generated using TypeDoc

+
+
+
+ + + \ No newline at end of file diff --git a/docs/api/enums/appleverifyreceipterrorcode.html b/docs/api/enums/appleverifyreceipterrorcode.html new file mode 100644 index 0000000..9f0c5e9 --- /dev/null +++ b/docs/api/enums/appleverifyreceipterrorcode.html @@ -0,0 +1,381 @@ + + + + + + AppleVerifyReceiptErrorCode | dollabill-apple + + + + + +
+
+
+
+ +
+
+ Options +
+
+ All +
    +
  • Public
  • +
  • Public/Protected
  • +
  • All
  • +
+
+ + + + + + +
+
+ Menu +
+
+
+
+
+ +
+
+
+
+
+

All of the statuses that are not successful. Apple was not able to provide back receipt information.

+
+

Official Apple documentation

+
+
+
+

Index

+
+ +
+
+
+

Enumeration members

+
+ +

APPLE_INTERNAL_ERROR

+
APPLE_INTERNAL_ERROR: = 21009
+ +
+
+ +

CUSTOMER_NOT_FOUND

+
CUSTOMER_NOT_FOUND: = 21010
+ +
+
+ +

INVALID_RECEIPT_OR_DOWN

+
INVALID_RECEIPT_OR_DOWN: = 21002
+ +
+
+ +

NOT_POST

+
NOT_POST: = 21000
+ +
+
+ +

SERVICE_DOWN

+
SERVICE_DOWN: = 21005
+ +
+
+ +

SHOULD_NOT_HAPPEN

+
SHOULD_NOT_HAPPEN: = 21001
+ +
+
+ +

UNAUTHORIZED

+
UNAUTHORIZED: = 21003
+ +
+
+ +

USE_PRODUCTION_ENVIRONMENT

+
USE_PRODUCTION_ENVIRONMENT: = 21008
+ +
+
+ +

USE_TEST_ENVIRONMENT

+
USE_TEST_ENVIRONMENT: = 21007
+ +
+
+ +

WRONG_SHARED_SECRET

+
WRONG_SHARED_SECRET: = 21004
+ +
+
+
+
+

Legend

+
+
    +
  • Constructor
  • +
  • Property
  • +
+
    +
  • Static property
  • +
  • Static method
  • +
+
    +
  • Property
  • +
+
    +
  • Inherited property
  • +
+
+
+
+
+

Generated using TypeDoc

+
+
+
+ + + \ No newline at end of file diff --git a/docs/api/globals.html b/docs/api/globals.html index bdb6e0a..0096596 100644 --- a/docs/api/globals.html +++ b/docs/api/globals.html @@ -3,8 +3,8 @@ - npm-module-blanky | npm-module-blanky - + dollabill-apple | dollabill-apple + @@ -15,7 +15,7 @@
@@ -45,43 +45,763 @@
- -

npm-module-blanky

+
+

dolla bill

-

Opinionated boilerplate used to make and deploy npm modules.

- -

Goals of this project

+

Easily work with Apple in-app purchases. So you can more easily collect your "Dolla dolla bills, ya'll"

+
+

Why use dolla bill?

+

When you are accepting in-app purchases in an iOS mobile app, you need to verify the transaction and continuously update your customer's subscription status. This verification and parsing of the responses back from Apple is boilerplate. This small module exists to help you with these tasks.

    -
  • Contain configuration files to setup all tools I tend to use in my development flow.
  • -
  • Clone, rename some files, and get developing!
  • -
  • Start with zero dependencies. I try my best to keep all npm modules as slim as possible.
  • +
  • No dependencies. This module tries to be as small as possible.
  • +
  • Fully tested. Automated testing + QA testing gives you the confidence that the module will work well for you.
  • +
  • Complete API documentation to more easily use it in your code.
  • +
  • Up-to-date If and when Apple changes the way you perform verification or Apple adds more features to verification, update this module and your projects can take advantage of the changes quickly and easily.
  • +
  • Auto retry If there is a HTTP error, dolla bill will try again for you.
  • +
  • Typescript typings This project is written in Typescript for that strict typing goodness.
+
+

Note: Technically there is one dependency but it's a tiny wrapper around a built-in nodejs feature. This module still remains super small with this.

+
+ +

Getting started

+
+
import dollaBill from "dollabill"
+
+const receiptFromStoreKit = // The base64 encoded receipt data that StoreKit gave your app
+
+const appleResponse = await dollaBill.verifyReceipt({
+    appPassword: process.env.APPLE_IN_APP_PURCHASE_PASSWORD,
+    receipt: receiptFromStoreKit
+})
+
+// If an error happens, it's fatal. It means that there is a bug with this library (create a GitHub issue with stacktrace and other info, please) or you the developer made a mistake when using this library.
+
+// It's recommended that you, the developer, views the error message and stacktrace to fix the issue. Note: The error here is not meant to be shown to your users.
+
+if (!appleResponse.isValid) {
+    // There was a problem. The request was valid and can be retried.
+
+    // look at the error and act on it however you wish.
+    // It's recommended that you log the error and then return back a message to your users saying there was a problem. The error here is meant for the developer to see, not a good error to return back to the user.
+    appleResponse.error
+} else {
+    // Time for you to update your database with the status of your customer and their subscription.
+    // This is easy because dolla bill parses the response from Apple to be easily readable.
+    // Check out the API documentation to learn about what `appleResponse` properties there are.
+}
+
type Response = AppleResponse | AppleError
+
+enum AppleErrorType {}
+// TODO
+
+type AppleError = {
+  code: number
+  type: AppleErrorType
+}
+
+type AppleResponse = {
+  isValid: boolean
+  autoRenewableSubscription?: AutoRenewableSubscription // If your in-app purchase is for auto renewable subscriptions, go here.
+  decodedReceipt: Receipt
+  raw: any // just in case you need it, here is the raw JSON response from Apple. https://developer.apple.com/documentation/appstorereceipts/responsebody
+}
+
+type AutoRenewableSubscription = {
+  transactions: Array<Transaction>
+  pendingRenewals: Array<PendingRenewal>
+  isGracePeriod: bool
+  gracePeriod: GracePeriodInfo
+}
+
+type GracePeriodInfo = {
+  // TODO
+  // make this up. it comes from: https://developer.apple.com/documentation/appstorereceipts/responsebody/pending_renewal_info
+}
+
+type PendingRenewal = {
+  // TODO
+  // https://developer.apple.com/documentation/appstorereceipts/responsebody/pending_renewal_info
+}
+
+type Transaction = {
+  // TODO
+  // https://developer.apple.com/documentation/appstorereceipts/responsebody/latest_receipt_info
+}
+
+type Receipt = {
+  // TODO
+  // https://developer.apple.com/documentation/appstorereceipts/responsebody/receipt
+}
+ +

Documentation

+
+

// TODO

+ +

Contributors

+
+

Thanks goes to these wonderful people (emoji key)

+ + + + + + + +

Levi Bostian

💻 📖 🚧
+ + +

Index

+
+

Type aliases

+
+ +

AppleLatestReceiptInfo

+
AppleLatestReceiptInfo: AppleInAppPurchaseTransaction & { is_upgraded?: undefined | "true"; offer_code_ref_name?: undefined | string; subscription_group_identifier?: undefined | string }
+ + +
+
+ +

AppleVerifyReceiptResponse

+ + +
+
+ +

AppleVerifyReceiptSuccessfulStatus

+
AppleVerifyReceiptSuccessfulStatus: 0 | 21006
+ +
+
+

All of the statuses that are successful. Apple is able to return back receipt information when any of these status codes occur.

+
+
+

Note: 0 means the receipt is valid. 21006 means it's valid, but the subscription is expired. You will still receive back a receipt decoded in the response with this status code.

+
+

Official Apple documentation

+
+
+
+ +

AutoRenewableSubscriptionActionableStatus

+
AutoRenewableSubscriptionActionableStatus: "not_yet_accepting_price_increase" | "cancelled" | "billing_issue" | "will_voluntary_cancel"
+ +
+
+

Options:

+
    +
  1. not_yet_accepting_price_increase - You have introduced a price increase. The customer has been notified about it but has not yet accepted it. This means the customer's subscription may be expired if they do not accept it.
  2. +
  3. cancelled - recently cancelled through Apple Support. good to message them to check out an alternative subscription we offer
  4. +
  5. billing_issue - the subscription expired because of a billing issue.
  6. +
  7. will_voluntary_cancel // time to send them UI saying to downgrade to try and keep them
  8. +
+
+
+
+
+ +

AutoRenewableSubscriptionInactiveStatus

+
AutoRenewableSubscriptionInactiveStatus: "voluntary_cancelled" | "in_grace_period" | "involuntary_cancelled" | "refunded"
+ +
+
+

Options: + 2. voluntary_cancel - they cancelled the subscription. You should not give the customer any access to the paid content. + 3. grace_period - There is a billing issue and Apple is attempting to automatically renew the subscription. Grace period means the customer has not cancelled or refunded. You have not yet lost them as a customer. Learn more about Grace Periods in the main docs for this project.

+
+
+

Tip: You can open the link https://apps.apple.com/account/billing to send the user to their payment details page in the App Store to update their payment information.

+
+
    +
  1. involuntary_cancel - the customer had a billing issue and their subscription is no longer active. Maybe you (1) did not enable the Grace Period feature in App Store Connect and the customer has encountered a billing issue or (2) you did enable the Grace Period feature but Apple has since given up on attempting to renew the subscription for the customer. You should no longer give the customer access to the paid content.
  2. +
  3. refunded - the customer contacted Apple support and asked for a refund of their purchase and received the partial or full refund. You should not give the customer any access to the paid content. You may optionally offer another subscription plan to them to switch to a different offer that meets their needs better.
  4. +
+
+
+
+ +

AutoRenewableSubscriptionRefundReason

+
AutoRenewableSubscriptionRefundReason: "upgrade" | "app_issue" | "other"
+ +
+
+ +

AutoRenewableSubscriptionTransaction

+
AutoRenewableSubscriptionTransaction: { expiresDate: Date; inIntroPeriod: boolean; inTrialPeriod: boolean; isUpgradeTransaction: boolean; offerCodeRefName?: undefined | string; originalPurchaseDate: Date; originalTransactionId: string; productId: string; promotionalOfferId?: undefined | string; purchaseDate: Date; refundReason?: AutoRenewableSubscriptionRefundReason; refundedDate?: Date; subscriptionGroupId: string; transactionId: string; webOrderLineItemId?: undefined | string }
+ +
+
+

Receipt for an auto-renewable subscription.

+
+
+
+

Type declaration

+
    +
  • +
    expiresDate: Date
    +
  • +
  • +
    inIntroPeriod: boolean
    +
  • +
  • +
    inTrialPeriod: boolean
    +
  • +
  • +
    isUpgradeTransaction: boolean
    +
  • +
  • +
    Optional offerCodeRefName?: undefined | string
    +
  • +
  • +
    originalPurchaseDate: Date
    +
  • +
  • +
    originalTransactionId: string
    +
  • +
  • +
    productId: string
    +
  • +
  • +
    Optional promotionalOfferId?: undefined | string
    +
  • +
  • +
    purchaseDate: Date
    +
  • +
  • +
    Optional refundReason?: AutoRenewableSubscriptionRefundReason
    +
  • +
  • +
    Optional refundedDate?: Date
    +
  • +
  • +
    subscriptionGroupId: string
    +
  • +
  • +
    transactionId: string
    +
  • +
  • +
    Optional webOrderLineItemId?: undefined | string
    +
  • +
+
+
+
+ +

VerifyReceiptFailure

+
VerifyReceiptFailure: { error: Error }
+ +
+

Type declaration

+ +
+
+
+ +

VerifyReceiptResponse

+
VerifyReceiptResponse: ParsedReceipt | VerifyReceiptFailure
+ +
+
+
+

Variables

+
+ +

Const maxNumberHttpAttempts

+
maxNumberHttpAttempts: 2 = 2
+ +
+
+
+

Functions

+
+ +

Const isFailure

+ + +
+
+ +

Const parseSuccess

+ + +
+
+ +

Const runVerifyReceipt

+ + +
+
+ +

Const verifyReceipt

+ + +
+
+
+

Object literals

+
+ +

Const _

+
_: object
+ +
+ +

array

+
array: object
+ +
+ +

contains

+
    +
  • contains<T>(array: T[], doesContain: (element: T) => boolean): boolean
  • +
+
    +
  • + +

    Type parameters

    +
      +
    • +

      T

      +
    • +
    +

    Parameters

    +
      +
    • +
      array: T[]
      +
    • +
    • +
      doesContain: (element: T) => boolean
      +
        +
      • +
          +
        • (element: T): boolean
        • +
        +
          +
        • +

          Parameters

          +
            +
          • +
            element: T
            +
          • +
          +

          Returns boolean

          +
        • +
        +
      • +
      +
    • +
    +

    Returns boolean

    +
  • +
+
+
+
+ +

date

+
date: object
+ +
+ +

sortNewToOld

+
    +
  • sortNewToOld(first: Date, second: Date): number
  • +
+
    +
  • + +
    +
    +

    Sort date array in order: [newer date, older date].

    +
    +

    Use:

    +
    const dates: Date[] = []
    +dates.sort(_.date.sortNewToOld)
    +
    +const nestedDates: {date: Date}[] = []
    +nestedDates.sort((first, second) => _.date.sortNewToOld(first.date, second.date))
    +
    +

    Parameters

    +
      +
    • +
      first: Date
      +
    • +
    • +
      second: Date
      +
    • +
    +

    Returns number

    +
  • +
+
+
+ +

sortOldToNew

+
    +
  • sortOldToNew(first: Date, second: Date): number
  • +
+
    +
  • + +
    +
    +

    Sort date array in order: [older date, newer date].

    +
    +

    Use:

    +
    const dates: Date[] = []
    +dates.sort(_.date.sortOldToNew)
    +
    +const nestedDates: {date: Date}[] = []
    +nestedDates.sort((first, second) => _.date.sortOldToNew(first.date, second.date))
    +
    +

    Parameters

    +
      +
    • +
      first: Date
      +
    • +
    • +
      second: Date
      +
    • +
    +

    Returns number

    +
  • +
+
+
+
+