diff --git a/.github/workflows/ci-hog.yml b/.github/workflows/ci-hog.yml index 6f4c326506da6..5733393905e49 100644 --- a/.github/workflows/ci-hog.yml +++ b/.github/workflows/ci-hog.yml @@ -118,9 +118,9 @@ jobs: antlr | grep "Version" npm run grammar:build && git diff --exit-code env: - # Installing a version of ANTLR compatible with what's in Homebrew as of October 2023 (version 4.13), + # Installing a version of ANTLR compatible with what's in Homebrew as of August 2024 (version 4.13.2), # as apt-get is quite out of date. The same version must be set in hogql_parser/pyproject.toml - ANTLR_VERSION: '4.13.1' + ANTLR_VERSION: '4.13.2' - name: Run HogVM Python tests if: needs.changes.outputs.hog == 'true' diff --git a/.github/workflows/rust-docker-build.yml b/.github/workflows/rust-docker-build.yml index 078fac107a400..352df97d2d5d7 100644 --- a/.github/workflows/rust-docker-build.yml +++ b/.github/workflows/rust-docker-build.yml @@ -1,4 +1,4 @@ -name: Build rust container images +name: Build and deploy rust container images on: workflow_dispatch: @@ -28,6 +28,9 @@ jobs: contents: read # allow reading the repo contents packages: write # allow push to ghcr.io + outputs: + digest: ${{ steps.docker_build.outputs.digest }} + defaults: run: working-directory: rust @@ -91,3 +94,36 @@ jobs: - name: Container image digest run: echo ${{ steps.docker_build.outputs.digest }} + + + deploy: + name: Deploy capture-replay + runs-on: ubuntu-latest + needs: build + if: github.ref == 'refs/heads/master' + steps: + - name: get deployer token + id: deployer + uses: getsentry/action-github-app-token@v3 + with: + app_id: ${{ secrets.DEPLOYER_APP_ID }} + private_key: ${{ secrets.DEPLOYER_APP_PRIVATE_KEY }} + + - name: Trigger livestream deployment + uses: peter-evans/repository-dispatch@v3 + with: + token: ${{ steps.deployer.outputs.token }} + repository: PostHog/charts + event-type: commit_state_update + client-payload: | + { + "values": { + "image": { + "sha": "${{ needs.build.outputs.digest }}" + } + }, + "release": "capture-replay", + "commit": ${{ toJson(github.event.head_commit) }}, + "repository": ${{ toJson(github.repository) }}, + "labels": [] + } diff --git a/cypress/e2e/alerts.cy.ts b/cypress/e2e/alerts.cy.ts index ea2dd44380ab1..110647969e6a7 100644 --- a/cypress/e2e/alerts.cy.ts +++ b/cypress/e2e/alerts.cy.ts @@ -15,19 +15,19 @@ describe('Alerts', () => { const createAlert = ( name: string = 'Alert name', - email: string = 'a@b.c', lowerThreshold: string = '100', upperThreshold: string = '200' ): void => { cy.get('[data-attr=more-button]').click() - cy.contains('Alerts').click() + cy.contains('Manage alerts').click() cy.contains('New alert').click() cy.get('[data-attr=alert-name]').clear().type(name) - cy.get('[data-attr=alert-notification-targets').clear().type(email) + cy.get('[data-attr=subscribed-users').click().type('{downarrow}{enter}') cy.get('[data-attr=alert-lower-threshold').clear().type(lowerThreshold) cy.get('[data-attr=alert-upper-threshold').clear().type(upperThreshold) cy.contains('Create alert').click() + cy.get('.Toastify__toast-body').should('contain', 'Alert saved') cy.url().should('not.include', '/new') cy.get('[aria-label="close"]').click() @@ -38,6 +38,7 @@ describe('Alerts', () => { cy.get('[data-attr=insight-edit-button]').click() cy.get('[data-attr=chart-filter]').click() cy.contains(displayType).click() + cy.get('.insight-empty-state').should('not.exist') cy.get('[data-attr=insight-save-button]').contains('Save').click() cy.url().should('not.include', '/edit') } @@ -45,7 +46,7 @@ describe('Alerts', () => { it('Should allow create and delete an alert', () => { cy.get('[data-attr=more-button]').click() // Alerts should be disabled for trends represented with graphs - cy.get('[data-attr=disabled-alerts-button]').should('exist') + cy.get('[data-attr=manage-alerts-button]').should('have.attr', 'aria-disabled', 'true') setInsightDisplayTypeAndSave('Number') @@ -54,10 +55,8 @@ describe('Alerts', () => { // Check the alert has the same values as when it was created cy.get('[data-attr=more-button]').click() - cy.contains('Alerts').click() cy.contains('Manage alerts').click() cy.get('[data-attr=alert-list-item]').contains('Alert name').click() - cy.get('[data-attr=alert-notification-targets]').should('have.value', 'a@b.c') cy.get('[data-attr=alert-name]').should('have.value', 'Alert name') cy.get('[data-attr=alert-lower-threshold').should('have.value', '100') cy.get('[data-attr=alert-upper-threshold').should('have.value', '200') @@ -90,7 +89,6 @@ describe('Alerts', () => { // Assert that saving an insight in an incompatible state removes alerts cy.get('[data-attr=more-button]').click() - cy.contains('Alerts').click() cy.contains('Manage alerts').click() cy.contains('Alert to be deleted because of a changed insight').should('not.exist') }) diff --git a/cypress/e2e/trends.cy.ts b/cypress/e2e/trends.cy.ts index b9b1fbba24680..ce8a6e8574b30 100644 --- a/cypress/e2e/trends.cy.ts +++ b/cypress/e2e/trends.cy.ts @@ -170,4 +170,8 @@ describe('Trends', () => { cy.get('[data-attr=math-monthly_active-0]').trigger('mouseenter') // Activate warning tooltip cy.get('.Tooltip').contains('we recommend using "Unique users" here instead').should('exist') }) + + it('Does not show delete button on single series insight', () => { + cy.get('[data-attr=delete-prop-filter-0]').should('not.exist') + }) }) diff --git a/frontend/__snapshots__/exporter-exporter--user-paths-insight--light.png b/frontend/__snapshots__/exporter-exporter--user-paths-insight--light.png index 25984c46d9502..333981c508581 100644 Binary files a/frontend/__snapshots__/exporter-exporter--user-paths-insight--light.png and b/frontend/__snapshots__/exporter-exporter--user-paths-insight--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--stickiness-edit--light--webkit.png b/frontend/__snapshots__/scenes-app-insights--stickiness-edit--light--webkit.png index 61dad0d0766e4..1940933f16bce 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--stickiness-edit--light--webkit.png and b/frontend/__snapshots__/scenes-app-insights--stickiness-edit--light--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--stickiness-edit--light.png b/frontend/__snapshots__/scenes-app-insights--stickiness-edit--light.png index 243c1cea204e1..ebf51524e67c1 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--stickiness-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--stickiness-edit--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--dark--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--dark--webkit.png index 64d99b712d842..335b16fdfc3ce 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--dark--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--dark--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--dark.png b/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--dark.png index 39fab033ecbe3..f2ab6d8018611 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--dark.png and b/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--dark.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--light--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--light--webkit.png index 2f87102a89afe..b5ef01afd3144 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--light--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--light--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--light.png b/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--light.png index 57a0cc2eaca3c..1ecf743c0a675 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--trends-bar-breakdown-edit--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--dark--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--dark--webkit.png index 55bf81ce188f4..0b30896b49b77 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--dark--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--dark--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--light--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--light--webkit.png index 4dcd20b4b0385..b11f8da52af3e 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--light--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--light--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--light.png b/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--light.png index fab546748b274..4494a987d7629 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--trends-bar-edit--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--dark--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--dark--webkit.png index e5b89d0605c8e..ea5bb505ae65a 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--dark--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--dark--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--dark.png b/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--dark.png index 0fd320603c08f..73ff3eb4b06d2 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--dark.png and b/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--dark.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--light--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--light--webkit.png index 7679dbac61976..f73c42037b141 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--light--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--light--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--light.png b/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--light.png index c386f35d6f590..68556c8a2300d 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--trends-line-breakdown-edit--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-line-edit--dark.png b/frontend/__snapshots__/scenes-app-insights--trends-line-edit--dark.png index 972045ad6f932..086476ba716c0 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-line-edit--dark.png and b/frontend/__snapshots__/scenes-app-insights--trends-line-edit--dark.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-line-edit--light.png b/frontend/__snapshots__/scenes-app-insights--trends-line-edit--light.png index 97cafb52341a0..1072316bf304a 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-line-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--trends-line-edit--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--dark--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--dark--webkit.png index f84ffcfeca674..b6fd3facddbab 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--dark--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--dark--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--dark.png b/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--dark.png index 0683b6ec5b810..d536ad9eb5cac 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--dark.png and b/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--dark.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--light--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--light--webkit.png index 2b3e77629b9ff..e8f32791910e1 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--light--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--light--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--light.png b/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--light.png index 560130a8d43fa..399f128c91764 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--trends-pie-breakdown-edit--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--dark--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--dark--webkit.png index bde049dababeb..756951deb6614 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--dark--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--dark--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--light--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--light--webkit.png index 329801d59fc36..1097be4f08f67 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--light--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--light--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--light.png b/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--light.png index 165d63d09a6b5..dc241e8086113 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--trends-pie-edit--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-table-breakdown-edit--light.png b/frontend/__snapshots__/scenes-app-insights--trends-table-breakdown-edit--light.png index 5095d619040db..3e06506b96a9f 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-table-breakdown-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--trends-table-breakdown-edit--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-table-edit--dark--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-table-edit--dark--webkit.png index 4fce62a2b668a..dd77d958263fb 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-table-edit--dark--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-table-edit--dark--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-table-edit--dark.png b/frontend/__snapshots__/scenes-app-insights--trends-table-edit--dark.png index 81bce266164f8..d40098ec1d9ec 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-table-edit--dark.png and b/frontend/__snapshots__/scenes-app-insights--trends-table-edit--dark.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-table-edit--light--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-table-edit--light--webkit.png index 32eecac9a1425..9ea4836236b93 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-table-edit--light--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-table-edit--light--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-table-edit--light.png b/frontend/__snapshots__/scenes-app-insights--trends-table-edit--light.png index 39e4a572c854a..2cdbc00422d69 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-table-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--trends-table-edit--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-value-breakdown-edit--light--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-value-breakdown-edit--light--webkit.png index c5a68841ec463..3ee3f6bedc6be 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-value-breakdown-edit--light--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-value-breakdown-edit--light--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-value-breakdown-edit--light.png b/frontend/__snapshots__/scenes-app-insights--trends-value-breakdown-edit--light.png index a8178333d0764..e58a0ca460bc8 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-value-breakdown-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--trends-value-breakdown-edit--light.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-value-edit--light--webkit.png b/frontend/__snapshots__/scenes-app-insights--trends-value-edit--light--webkit.png index 56160ec991fc9..6802705a2606f 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-value-edit--light--webkit.png and b/frontend/__snapshots__/scenes-app-insights--trends-value-edit--light--webkit.png differ diff --git a/frontend/__snapshots__/scenes-app-insights--trends-value-edit--light.png b/frontend/__snapshots__/scenes-app-insights--trends-value-edit--light.png index aba24e56a0b36..a7c96c46245bd 100644 Binary files a/frontend/__snapshots__/scenes-app-insights--trends-value-edit--light.png and b/frontend/__snapshots__/scenes-app-insights--trends-value-edit--light.png differ diff --git a/frontend/src/lib/api.ts b/frontend/src/lib/api.ts index 7b668a45bc434..99cd61b959ef4 100644 --- a/frontend/src/lib/api.ts +++ b/frontend/src/lib/api.ts @@ -10,16 +10,19 @@ import { SavedSessionRecordingPlaylistsResult } from 'scenes/session-recordings/ import { getCurrentExporterData } from '~/exporter/exporterViewLogic' import { + AlertType, + AlertTypeWrite, DatabaseSerializedFieldType, ErrorTrackingGroup, QuerySchema, QueryStatusResponse, + RecordingsQuery, + RecordingsQueryResponse, RefreshType, } from '~/queries/schema' import { ActionType, ActivityScope, - AlertType, AppMetricsTotalsV2Response, AppMetricsV2RequestParams, AppMetricsV2Response, @@ -93,7 +96,6 @@ import { SessionRecordingPlaylistType, SessionRecordingSnapshotParams, SessionRecordingSnapshotResponse, - SessionRecordingsResponse, SessionRecordingType, SharingConfigurationType, SlackChannelType, @@ -730,12 +732,12 @@ class ApiRequest { } // # Alerts - public alerts(teamId?: TeamType['id']): ApiRequest { - return this.projectsDetail(teamId).addPathComponent('alerts') + public alerts(id: InsightModel['id'], teamId?: TeamType['id']): ApiRequest { + return this.insight(id, teamId).addPathComponent('alerts') } - public alert(id: AlertType['id'], teamId?: TeamType['id']): ApiRequest { - return this.alerts(teamId).addPathComponent(id) + public alert(id: AlertType['id'], insightId: InsightModel['id'], teamId?: TeamType['id']): ApiRequest { + return this.alerts(insightId, teamId).addPathComponent(id) } // Resource Access Permissions @@ -1750,7 +1752,7 @@ const api = { }, recordings: { - async list(params: Record): Promise { + async list(params: RecordingsQuery): Promise { return await new ApiRequest().recordings().withQueryString(toParams(params)).get() }, async getMatchingEvents(params: string): Promise<{ results: string[] }> { @@ -1838,7 +1840,7 @@ const api = { async listPlaylistRecordings( playlistId: SessionRecordingPlaylistType['short_id'], params: Record = {} - ): Promise { + ): Promise { return await new ApiRequest() .recordingPlaylist(playlistId) .withAction('recordings') @@ -2264,20 +2266,20 @@ const api = { }, alerts: { - async get(alertId: AlertType['id']): Promise { - return await new ApiRequest().alert(alertId).get() + async get(insightId: number, alertId: AlertType['id']): Promise { + return await new ApiRequest().alert(alertId, insightId).get() }, - async create(data: Partial): Promise { - return await new ApiRequest().alerts().create({ data }) + async create(insightId: number, data: Partial): Promise { + return await new ApiRequest().alerts(insightId).create({ data }) }, - async update(alertId: AlertType['id'], data: Partial): Promise { - return await new ApiRequest().alert(alertId).update({ data }) + async update(insightId: number, alertId: AlertType['id'], data: Partial): Promise { + return await new ApiRequest().alert(alertId, insightId).update({ data }) }, async list(insightId: number): Promise> { - return await new ApiRequest().alerts().withQueryString(`insight=${insightId}`).get() + return await new ApiRequest().alerts(insightId).get() }, - async delete(alertId: AlertType['id']): Promise { - return await new ApiRequest().alert(alertId).delete() + async delete(insightId: number, alertId: AlertType['id']): Promise { + return await new ApiRequest().alert(alertId, insightId).delete() }, }, diff --git a/frontend/src/lib/components/Alerts/AlertDeletionWarning.tsx b/frontend/src/lib/components/Alerts/AlertDeletionWarning.tsx index cf1e5d0ad492c..d3c798462e792 100644 --- a/frontend/src/lib/components/Alerts/AlertDeletionWarning.tsx +++ b/frontend/src/lib/components/Alerts/AlertDeletionWarning.tsx @@ -7,8 +7,16 @@ import { alertsLogic } from './alertsLogic' export function AlertDeletionWarning(): JSX.Element | null { const { insightProps, insight } = useValues(insightLogic) + if (!insight?.short_id) { + return null + } + const { shouldShowAlertDeletionWarning } = useValues( - alertsLogic({ insightShortId: insight.short_id!, insightLogicProps: insightProps }) + alertsLogic({ + insightShortId: insight.short_id, + insightId: insight.id as number, + insightLogicProps: insightProps, + }) ) if (!shouldShowAlertDeletionWarning || !insight.short_id) { diff --git a/frontend/src/lib/components/Alerts/AlertsModal.tsx b/frontend/src/lib/components/Alerts/AlertsModal.tsx index d1ad9d9baf722..22e57b1112b82 100644 --- a/frontend/src/lib/components/Alerts/AlertsModal.tsx +++ b/frontend/src/lib/components/Alerts/AlertsModal.tsx @@ -1,4 +1,4 @@ -import { LemonButton, LemonButtonWithDropdown } from '@posthog/lemon-ui' +import { LemonButton } from '@posthog/lemon-ui' import { useActions, useValues } from 'kea' import { router } from 'kea-router' import { FEATURE_FLAGS } from 'lib/constants' @@ -17,11 +17,11 @@ import { ManageAlerts } from './views/ManageAlerts' export interface AlertsModalProps extends AlertsLogicProps { isOpen: boolean closeModal: () => void - alertId: number | 'new' | null + alertId?: string | null } export function AlertsModal(props: AlertsModalProps): JSX.Element { - const { closeModal, insightShortId, insightLogicProps, alertId, isOpen } = props + const { closeModal, insightId, insightShortId, insightLogicProps, alertId, isOpen } = props const { push } = useActions(router) const { userLoading } = useValues(userLogic) @@ -32,14 +32,16 @@ export function AlertsModal(props: AlertsModalProps): JSX.Element { {!alertId ? ( push(urls.alert(insightShortId, id.toString()))} + onSelect={(id) => push(urls.alert(insightShortId, id ?? 'new'))} /> ) : ( push(urls.alerts(insightShortId))} @@ -62,36 +64,19 @@ export function AlertsButton({ insight }: AlertsButtonProps): JSX.Element { if (!showAlerts) { return <> } - if (!areAlertsSupportedForInsight(insight.query)) { - return ( - - Alerts - - ) - } + return ( - push(urls.alerts(insight.short_id!))} fullWidth - dropdown={{ - actionable: true, - closeParentPopoverOnClickInside: true, - placement: 'right-start', - overlay: ( - <> - push(urls.alert(insight.short_id!, 'new'))} fullWidth> - New alert - - push(urls.alerts(insight.short_id!))} fullWidth> - Manage alerts - - - ), - }} + disabledReason={ + !areAlertsSupportedForInsight(insight.query) + ? 'Insights are only available for trends represented as a number. Change the insight representation to add alerts.' + : undefined + } > - Alerts - + Manage alerts + ) } diff --git a/frontend/src/lib/components/Alerts/alertLogic.ts b/frontend/src/lib/components/Alerts/alertLogic.ts index 5887329a815ec..66b3fe483c5cd 100644 --- a/frontend/src/lib/components/Alerts/alertLogic.ts +++ b/frontend/src/lib/components/Alerts/alertLogic.ts @@ -1,38 +1,38 @@ -import { connect, kea, key, path, props } from 'kea' +import { connect, kea, key, listeners, path, props } from 'kea' import { forms } from 'kea-forms' import { loaders } from 'kea-loaders' import { router, urlToAction } from 'kea-router' import api from 'lib/api' import { lemonToast } from 'lib/lemon-ui/LemonToast/LemonToast' -import { isEmail } from 'lib/utils' -import { getInsightId } from 'scenes/insights/utils' import { urls } from 'scenes/urls' -import { AlertType } from '~/types' +import { AlertType, AlertTypeWrite } from '~/queries/schema' import type { alertLogicType } from './alertLogicType' import { alertsLogic, AlertsLogicProps } from './alertsLogic' export interface AlertLogicProps extends AlertsLogicProps { - id: number | 'new' + id?: string } export const alertLogic = kea([ path(['lib', 'components', 'Alerts', 'alertLogic']), props({} as AlertLogicProps), - key(({ id, insightShortId }) => `${insightShortId}-${id ?? 'new'}`), + key(({ id, insightId }) => `${insightId}-${id ?? 'new'}`), connect(() => ({ - actions: [alertsLogic, ['loadAlerts']], + actions: [alertsLogic, ['loadAlerts'], router, ['push']], })), loaders(({ props }) => ({ alert: { __default: undefined as unknown as AlertType, loadAlert: async () => { - if (props.id && props.id !== 'new') { - return await api.alerts.get(props.id) + if (props.id) { + return await api.alerts.get(props.insightId, props.id) + } + return { + enabled: true, } - return { anomaly_condition: { absoluteThreshold: {} } } }, }, })), @@ -40,40 +40,43 @@ export const alertLogic = kea([ forms(({ props, actions }) => ({ alert: { defaults: {} as unknown as AlertType, - errors: ({ name, target_value }) => ({ + errors: ({ name }) => ({ name: !name ? 'You need to give your alert a name' : undefined, - target_value: !target_value - ? 'This field is required.' - : !target_value.split(',').every((email) => isEmail(email)) - ? 'All emails must be valid' - : undefined, }), submit: async (alert) => { - const insightId = await getInsightId(props.insightShortId) - - const payload = { + const payload: AlertTypeWrite = { ...alert, - insight: insightId, + subscribed_users: alert.subscribed_users?.map(({ id }) => id), + insight: props.insightId, } - const updatedAlert: AlertType = - props.id === 'new' ? await api.alerts.create(payload) : await api.alerts.update(props.id, payload) + try { + const updatedAlert: AlertType = !props.id + ? await api.alerts.create(props.insightId, payload) + : await api.alerts.update(props.insightId, props.id, payload) - actions.resetAlert() - - if (updatedAlert.id !== props.id) { - router.actions.replace(urls.alerts(props.insightShortId)) - } + actions.resetAlert() - actions.loadAlerts() - actions.loadAlertSuccess(updatedAlert) - lemonToast.success(`Alert saved.`) + actions.loadAlerts() + actions.loadAlertSuccess(updatedAlert) + lemonToast.success(`Alert saved.`) - return updatedAlert + return updatedAlert + } catch (error: any) { + const field = error.data?.attr?.replace(/_/g, ' ') + lemonToast.error(`Error saving alert: ${field}: ${error.detail}`) + throw error + } }, }, })), + listeners(({ props }) => ({ + submitAlertSuccess: () => { + router.actions.push(urls.alerts(props.insightShortId)) + }, + })), + urlToAction(({ actions }) => ({ '/*/*/alerts/:id': () => { actions.loadAlert() diff --git a/frontend/src/lib/components/Alerts/alertsLogic.ts b/frontend/src/lib/components/Alerts/alertsLogic.ts index 3e24f5daa07d9..467d7c616b42a 100644 --- a/frontend/src/lib/components/Alerts/alertsLogic.ts +++ b/frontend/src/lib/components/Alerts/alertsLogic.ts @@ -2,14 +2,15 @@ import { actions, afterMount, connect, kea, key, listeners, path, props, reducer import { loaders } from 'kea-loaders' import api from 'lib/api' import { insightVizDataLogic } from 'scenes/insights/insightVizDataLogic' -import { getInsightId } from 'scenes/insights/utils' +import { AlertType } from '~/queries/schema' import { isInsightVizNode, isTrendsQuery } from '~/queries/utils' -import { AlertType, ChartDisplayType, InsightLogicProps, InsightShortId } from '~/types' +import { ChartDisplayType, InsightLogicProps, InsightShortId } from '~/types' import type { alertsLogicType } from './alertsLogicType' export interface AlertsLogicProps { + insightId: number insightShortId: InsightShortId insightLogicProps: InsightLogicProps } @@ -27,9 +28,9 @@ export const areAlertsSupportedForInsight = (query?: Record | null) export const alertsLogic = kea([ path(['lib', 'components', 'Alerts', 'alertsLogic']), props({} as AlertsLogicProps), - key(({ insightShortId }) => `insight-${insightShortId}`), + key(({ insightId }) => `insight-${insightId}`), actions({ - deleteAlert: (id: number) => ({ id }), + deleteAlert: (id: string) => ({ id }), setShouldShowAlertDeletionWarning: (show: boolean) => ({ show }), }), @@ -41,11 +42,7 @@ export const alertsLogic = kea([ alerts: { __default: [] as AlertType[], loadAlerts: async () => { - const insightId = await getInsightId(props.insightShortId) - if (!insightId) { - return [] - } - const response = await api.alerts.list(insightId) + const response = await api.alerts.list(props.insightId) return response.results }, }, @@ -63,9 +60,9 @@ export const alertsLogic = kea([ ], }), - listeners(({ actions, values }) => ({ + listeners(({ actions, values, props }) => ({ deleteAlert: async ({ id }) => { - await api.alerts.delete(id) + await api.alerts.delete(props.insightId, id) }, setQuery: ({ query }) => { if (values.alerts.length === 0 || areAlertsSupportedForInsight(query)) { diff --git a/frontend/src/lib/components/Alerts/views/EditAlert.tsx b/frontend/src/lib/components/Alerts/views/EditAlert.tsx index 46d0dfd627cc1..eac20a8703086 100644 --- a/frontend/src/lib/components/Alerts/views/EditAlert.tsx +++ b/frontend/src/lib/components/Alerts/views/EditAlert.tsx @@ -1,11 +1,17 @@ -import { LemonInput } from '@posthog/lemon-ui' +import { LemonCheckbox, LemonInput } from '@posthog/lemon-ui' import { useActions, useValues } from 'kea' import { Form, Group } from 'kea-forms' +import { AlertStateIndicator } from 'lib/components/Alerts/views/ManageAlerts' +import { MemberSelectMultiple } from 'lib/components/MemberSelectMultiple' +import { TZLabel } from 'lib/components/TZLabel' +import { UserActivityIndicator } from 'lib/components/UserActivityIndicator/UserActivityIndicator' import { IconChevronLeft } from 'lib/lemon-ui/icons' import { LemonButton } from 'lib/lemon-ui/LemonButton' import { LemonField } from 'lib/lemon-ui/LemonField' import { LemonModal } from 'lib/lemon-ui/LemonModal' +import { AlertType } from '~/queries/schema' + import { alertLogic, AlertLogicProps } from '../alertLogic' import { alertsLogic } from '../alertsLogic' @@ -14,16 +20,54 @@ interface EditAlertProps extends AlertLogicProps { onDelete: () => void } +export function AlertState({ alert }: { alert: AlertType }): JSX.Element | null { + if (!alert.checks || alert.checks.length === 0) { + return null + } + + return ( +
+

+ Current status {alert.state === 'firing' ? 'firing' : 'not met'} + +

+ + + + + + + + + + + {alert.checks.map((check) => ( + + + + + + + ))} + +
StatusTimeValueTargets notified
{check.state === 'firing' ? 'Firing' : 'Not met'} + + {check.calculated_value}{check.targets_notified ? 'Yes' : 'No'}
+
+ ) +} + export function EditAlert(props: EditAlertProps): JSX.Element { const logic = alertLogic(props) const alertslogic = alertsLogic(props) const { alert, isAlertSubmitting, alertChanged } = useValues(logic) const { deleteAlert } = useActions(alertslogic) + const { setAlertValue } = useActions(logic) const id = props.id const _onDelete = (): void => { - if (id !== 'new') { + if (id) { deleteAlert(id) props.onDelete() } @@ -35,11 +79,11 @@ export function EditAlert(props: EditAlertProps): JSX.Element {
} onClick={props.onCancel} size="xsmall" /> -

{id === 'new' ? 'New' : 'Edit '} Alert

+

{!id ? 'New' : 'Edit '} Alert

- + {!alert ? (

Not found

@@ -47,42 +91,62 @@ export function EditAlert(props: EditAlertProps): JSX.Element {
) : ( <> - - - - - - - - - - - - - - - - - +
+ {alert?.created_by ? ( + + ) : null} + + + + + + + + + + u.id) ?? []} + idKey="id" + onChange={(value) => setAlertValue('subscribed_users', value)} + /> + + + + + + + + + + + +
+ )}
- {alert && id !== 'new' && ( + {alert && id && ( Delete alert @@ -92,7 +156,7 @@ export function EditAlert(props: EditAlertProps): JSX.Element { Cancel - {id === 'new' ? 'Create alert' : 'Save'} + {!id ? 'Create alert' : 'Save'} diff --git a/frontend/src/lib/components/Alerts/views/ManageAlerts.tsx b/frontend/src/lib/components/Alerts/views/ManageAlerts.tsx index 6f7c8f953ffa3..6d904f0133105 100644 --- a/frontend/src/lib/components/Alerts/views/ManageAlerts.tsx +++ b/frontend/src/lib/components/Alerts/views/ManageAlerts.tsx @@ -1,14 +1,26 @@ -import { IconEllipsis } from '@posthog/icons' +import { IconEllipsis, IconPause } from '@posthog/icons' import { useActions, useValues } from 'kea' +import { router } from 'kea-router' +import { IconPlayCircle } from 'lib/lemon-ui/icons' import { LemonButton } from 'lib/lemon-ui/LemonButton' import { LemonModal } from 'lib/lemon-ui/LemonModal' +import { LemonTag } from 'lib/lemon-ui/LemonTag' import { ProfileBubbles } from 'lib/lemon-ui/ProfilePicture' import { pluralize } from 'lib/utils' +import { urls } from 'scenes/urls' -import { AlertType } from '~/types' +import { AlertType } from '~/queries/schema' import { alertsLogic, AlertsLogicProps } from '../alertsLogic' +export function AlertStateIndicator({ alert }: { alert: AlertType }): JSX.Element { + return alert.state === 'firing' ? ( + + ) : ( + + ) +} + interface AlertListItemProps { alert: AlertType onClick: () => void @@ -16,15 +28,16 @@ interface AlertListItemProps { } export function AlertListItem({ alert, onClick, onDelete }: AlertListItemProps): JSX.Element { + const absoluteThreshold = alert.threshold?.configuration?.absoluteThreshold return ( : } sideAction={{ icon: , - dropdown: { overlay: ( <> @@ -45,9 +58,23 @@ export function AlertListItem({ alert, onClick, onDelete }: AlertListItemProps): >
-
{alert.name}
+
+ {alert.name} + {alert.enabled ? ( + <> + +
+ {absoluteThreshold?.lower && `Low ${absoluteThreshold.lower}`} + {absoluteThreshold?.lower && absoluteThreshold?.upper ? ' · ' : ''} + {absoluteThreshold?.upper && `High ${absoluteThreshold.upper}`} +
+ + ) : ( +
Disabled
+ )} +
- ({ email }))} /> + ({ email }))} />
) @@ -55,10 +82,11 @@ export function AlertListItem({ alert, onClick, onDelete }: AlertListItemProps): interface ManageAlertsProps extends AlertsLogicProps { onCancel: () => void - onSelect: (value: number | 'new') => void + onSelect: (value?: string) => void } export function ManageAlerts(props: ManageAlertsProps): JSX.Element { + const { push } = useActions(router) const logic = alertsLogic(props) const { alerts } = useValues(logic) @@ -67,15 +95,20 @@ export function ManageAlerts(props: ManageAlertsProps): JSX.Element { return ( <> -

Manage Alerts

+

+ Manage Alerts ALPHA +

+
+ With alerts, PostHog will monitor your insight and notify you when certain conditions are met. We do + not evaluate alerts in real-time, but rather on a schedule of once every hour. Please note that + alerts are in alpha and may not be fully reliable. +
{alerts.length ? (
- {alerts?.length} - {' active '} - {pluralize(alerts.length || 0, 'alert', 'alerts', false)} + {alerts?.length} {pluralize(alerts.length || 0, 'alert', 'alerts', false)}
{alerts.map((alert) => ( @@ -97,6 +130,9 @@ export function ManageAlerts(props: ManageAlertsProps): JSX.Element { + push(urls.alert(props.insightShortId, 'new'))}> + New alert + Close diff --git a/frontend/src/lib/components/MemberSelectMultiple.tsx b/frontend/src/lib/components/MemberSelectMultiple.tsx new file mode 100644 index 0000000000000..97900f97947b1 --- /dev/null +++ b/frontend/src/lib/components/MemberSelectMultiple.tsx @@ -0,0 +1,48 @@ +import { LemonInputSelect, ProfilePicture } from '@posthog/lemon-ui' +import { useActions, useValues } from 'kea' +import { fullName } from 'lib/utils' +import { useEffect } from 'react' +import { membersLogic } from 'scenes/organization/membersLogic' + +import { UserBasicType } from '~/types' + +type UserIdType = string | number + +export type MemberSelectMultipleProps = { + idKey: 'email' | 'uuid' | 'id' + value: UserIdType[] + onChange: (values: UserBasicType[]) => void +} + +export function MemberSelectMultiple({ idKey, value, onChange }: MemberSelectMultipleProps): JSX.Element { + const { filteredMembers, membersLoading } = useValues(membersLogic) + const { ensureAllMembersLoaded } = useActions(membersLogic) + + useEffect(() => { + ensureAllMembersLoaded() + }, []) + + const options = filteredMembers.map((member) => ({ + key: member.user[idKey].toString(), + label: fullName(member.user), + value: member.user[idKey], + icon: , + })) + + return ( + v.toString())} + loading={membersLoading} + onChange={(newValues: UserIdType[]) => { + const selectedUsers = filteredMembers.filter((member) => + newValues.includes(member.user[idKey].toString()) + ) + onChange(selectedUsers.map((member) => member.user)) + }} + mode="multiple" + options={options} + data-attr="subscribed-users" + /> + ) +} diff --git a/frontend/src/lib/components/PropertyFilters/utils.ts b/frontend/src/lib/components/PropertyFilters/utils.ts index f048469305add..2c022d570e9ed 100644 --- a/frontend/src/lib/components/PropertyFilters/utils.ts +++ b/frontend/src/lib/components/PropertyFilters/utils.ts @@ -355,10 +355,6 @@ export function taxonomicFilterTypeToPropertyFilterType( return PropertyFilterType.DataWarehousePersonProperty } - if (filterType == TaxonomicFilterGroupType.Replay) { - return PropertyFilterType.Recording - } - return Object.entries(propertyFilterMapping).find(([, v]) => v === filterType)?.[0] as | PropertyFilterType | undefined diff --git a/frontend/src/lib/components/UniversalFilters/utils.ts b/frontend/src/lib/components/UniversalFilters/utils.ts index 923ca44767385..f8b63af80ce5e 100644 --- a/frontend/src/lib/components/UniversalFilters/utils.ts +++ b/frontend/src/lib/components/UniversalFilters/utils.ts @@ -1,4 +1,4 @@ -import { ActionFilter, FilterLogicalOperator, RecordingPropertyFilter } from '~/types' +import { ActionFilter, FilterLogicalOperator, LogEntryPropertyFilter, RecordingPropertyFilter } from '~/types' import { isCohortPropertyFilter } from '../PropertyFilters/utils' import { UniversalFiltersGroup, UniversalFiltersGroupValue, UniversalFilterValue } from './UniversalFilters' @@ -6,23 +6,21 @@ import { UniversalFiltersGroup, UniversalFiltersGroupValue, UniversalFilterValue export function isUniversalGroupFilterLike(filter?: UniversalFiltersGroupValue): filter is UniversalFiltersGroup { return filter?.type === FilterLogicalOperator.And || filter?.type === FilterLogicalOperator.Or } - export function isEntityFilter(filter: UniversalFilterValue): filter is ActionFilter { return isEventFilter(filter) || isActionFilter(filter) } - export function isEventFilter(filter: UniversalFilterValue): filter is ActionFilter { return filter.type === 'events' } - export function isActionFilter(filter: UniversalFilterValue): filter is ActionFilter { return filter.type === 'actions' } - export function isRecordingPropertyFilter(filter: UniversalFilterValue): filter is RecordingPropertyFilter { return filter.type === 'recording' } - +export function isLogEntryPropertyFilter(filter: UniversalFilterValue): filter is LogEntryPropertyFilter { + return filter.type === 'log_entry' +} export function isEditableFilter(filter: UniversalFilterValue): boolean { return isEntityFilter(filter) ? false : !isCohortPropertyFilter(filter) } diff --git a/frontend/src/queries/nodes/InsightViz/TrendsSeries.tsx b/frontend/src/queries/nodes/InsightViz/TrendsSeries.tsx index 3c5748cba6dd6..f4516279a254c 100644 --- a/frontend/src/queries/nodes/InsightViz/TrendsSeries.tsx +++ b/frontend/src/queries/nodes/InsightViz/TrendsSeries.tsx @@ -19,7 +19,9 @@ import { queryNodeToFilter } from '../InsightQuery/utils/queryNodeToFilter' export function TrendsSeries(): JSX.Element | null { const { insightProps } = useValues(insightLogic) - const { querySource, isLifecycle, isStickiness, display, hasFormula } = useValues(insightVizDataLogic(insightProps)) + const { querySource, isLifecycle, isStickiness, display, hasFormula, series } = useValues( + insightVizDataLogic(insightProps) + ) const { updateQuerySource } = useActions(insightVizDataLogic(insightProps)) const { showGroupsOptions, groupsTaxonomicTypes } = useValues(groupsModel) @@ -88,6 +90,7 @@ export function TrendsSeries(): JSX.Element | null { TaxonomicFilterGroupType.Actions, TaxonomicFilterGroupType.DataWarehouse, ]} + hideDeleteBtn={series?.length === 1} /> ) diff --git a/frontend/src/queries/schema.json b/frontend/src/queries/schema.json index 58d64587f477c..3584da8f25fd7 100644 --- a/frontend/src/queries/schema.json +++ b/frontend/src/queries/schema.json @@ -1,18 +1,6 @@ { "$schema": "http://json-schema.org/draft-07/schema#", "definitions": { - "AbsoluteThreshold": { - "additionalProperties": false, - "properties": { - "lower": { - "type": ["number", "null"] - }, - "upper": { - "type": ["number", "null"] - } - }, - "type": "object" - }, "ActionsNode": { "additionalProperties": false, "properties": { @@ -195,14 +183,143 @@ "enum": ["numeric", "duration", "duration_ms", "percentage", "percentage_scaled"], "type": "string" }, - "AnomalyCondition": { + "AlertCheck": { "additionalProperties": false, "properties": { - "absoluteThreshold": { - "$ref": "#/definitions/AbsoluteThreshold" + "calculated_value": { + "type": "number" + }, + "created_at": { + "type": "string" + }, + "id": { + "type": "string" + }, + "state": { + "type": "string" + }, + "targets_notified": { + "type": "boolean" } }, - "required": ["absoluteThreshold"], + "required": ["id", "created_at", "calculated_value", "state", "targets_notified"], + "type": "object" + }, + "AlertCondition": { + "additionalProperties": false, + "type": "object" + }, + "AlertType": { + "additionalProperties": false, + "properties": { + "checks": { + "items": { + "$ref": "#/definitions/AlertCheck" + }, + "type": "array" + }, + "condition": { + "$ref": "#/definitions/AlertCondition" + }, + "created_at": { + "type": "string" + }, + "created_by": { + "$ref": "#/definitions/UserBasicType" + }, + "enabled": { + "type": "boolean" + }, + "id": { + "type": "string" + }, + "insight": { + "type": "number" + }, + "last_notified_at": { + "type": "string" + }, + "name": { + "type": "string" + }, + "state": { + "type": "string" + }, + "subscribed_users": { + "items": { + "$ref": "#/definitions/UserBasicType" + }, + "type": "array" + }, + "threshold": { + "additionalProperties": false, + "properties": { + "configuration": { + "$ref": "#/definitions/InsightThreshold" + } + }, + "required": ["configuration"], + "type": "object" + } + }, + "required": [ + "checks", + "condition", + "created_at", + "created_by", + "enabled", + "id", + "insight", + "last_notified_at", + "name", + "state", + "subscribed_users", + "threshold" + ], + "type": "object" + }, + "AlertTypeBase": { + "additionalProperties": false, + "properties": { + "condition": { + "$ref": "#/definitions/AlertCondition" + }, + "enabled": { + "type": "boolean" + }, + "insight": { + "type": "number" + }, + "name": { + "type": "string" + } + }, + "required": ["name", "condition", "enabled", "insight"], + "type": "object" + }, + "AlertTypeWrite": { + "additionalProperties": false, + "properties": { + "condition": { + "$ref": "#/definitions/AlertCondition" + }, + "enabled": { + "type": "boolean" + }, + "insight": { + "type": "number" + }, + "name": { + "type": "string" + }, + "subscribed_users": { + "items": { + "type": "integer" + }, + "type": "array" + } + }, + "required": ["condition", "enabled", "insight", "name", "subscribed_users"], "type": "object" }, "AnyDataNode": { @@ -4745,6 +4862,10 @@ } ] }, + "HedgehogColorOptions": { + "enum": ["green", "red", "blue", "purple", "dark", "light", "sepia", "invert", "invert-hue", "greyscale"], + "type": "string" + }, "HogLanguage": { "enum": ["hog", "hogJson", "hogQL", "hogQLExpr", "hogTemplate"], "type": "string" @@ -5478,6 +5599,15 @@ "InsightShortId": { "type": "string" }, + "InsightThreshold": { + "additionalProperties": false, + "properties": { + "absoluteThreshold": { + "$ref": "#/definitions/InsightsThresholdAbsolute" + } + }, + "type": "object" + }, "InsightVizNode": { "additionalProperties": false, "properties": { @@ -5780,6 +5910,18 @@ "required": ["kind"], "type": "object" }, + "InsightsThresholdAbsolute": { + "additionalProperties": false, + "properties": { + "lower": { + "type": "number" + }, + "upper": { + "type": "number" + } + }, + "type": "object" + }, "IntervalType": { "enum": ["minute", "hour", "day", "week", "month"], "type": "string" @@ -5951,6 +6093,32 @@ "required": ["key", "operator", "type"], "type": "object" }, + "MatchedRecording": { + "additionalProperties": false, + "properties": { + "events": { + "items": { + "$ref": "#/definitions/MatchedRecordingEvent" + }, + "type": "array" + }, + "session_id": { + "type": "string" + } + }, + "required": ["events"], + "type": "object" + }, + "MatchedRecordingEvent": { + "additionalProperties": false, + "properties": { + "uuid": { + "type": "string" + } + }, + "required": ["uuid"], + "type": "object" + }, "MathType": { "anyOf": [ { @@ -5970,6 +6138,32 @@ } ] }, + "MinimalHedgehogConfig": { + "additionalProperties": false, + "properties": { + "accessories": { + "items": { + "type": "string" + }, + "type": "array" + }, + "color": { + "anyOf": [ + { + "$ref": "#/definitions/HedgehogColorOptions" + }, + { + "type": "null" + } + ] + }, + "use_as_profile": { + "type": "boolean" + } + }, + "required": ["use_as_profile", "color", "accessories"], + "type": "object" + }, "MultipleBreakdownOptions": { "additionalProperties": false, "properties": { @@ -6003,6 +6197,7 @@ "FunnelsActorsQuery", "FunnelCorrelationActorsQuery", "SessionsTimelineQuery", + "RecordingsQuery", "SessionAttributionExplorerQuery", "ErrorTrackingQuery", "DataTableNode", @@ -6292,6 +6487,37 @@ "required": ["key", "operator", "type"], "type": "object" }, + "PersonType": { + "additionalProperties": false, + "properties": { + "created_at": { + "type": "string" + }, + "distinct_ids": { + "items": { + "type": "string" + }, + "type": "array" + }, + "id": { + "type": "string" + }, + "is_identified": { + "type": "boolean" + }, + "name": { + "type": "string" + }, + "properties": { + "type": "object" + }, + "uuid": { + "type": "string" + } + }, + "required": ["distinct_ids", "properties"], + "type": "object" + }, "PersonsNode": { "additionalProperties": false, "deprecated": "Use `ActorsQuery` instead.", @@ -8067,14 +8293,6 @@ { "$ref": "#/definitions/DurationType" }, - { - "const": "console_log_level", - "type": "string" - }, - { - "const": "console_log_query", - "type": "string" - }, { "const": "snapshot_source", "type": "string" @@ -8102,6 +8320,112 @@ "required": ["key", "operator", "type"], "type": "object" }, + "RecordingsQuery": { + "additionalProperties": false, + "properties": { + "actions": { + "items": { + "type": "object" + }, + "type": "array" + }, + "console_log_filters": { + "items": { + "$ref": "#/definitions/LogEntryPropertyFilter" + }, + "type": "array" + }, + "date_from": { + "type": ["string", "null"] + }, + "date_to": { + "type": ["string", "null"] + }, + "events": { + "items": { + "type": "object" + }, + "type": "array" + }, + "filter_test_accounts": { + "type": "boolean" + }, + "having_predicates": { + "items": { + "$ref": "#/definitions/AnyPropertyFilter" + }, + "type": "array" + }, + "kind": { + "const": "RecordingsQuery", + "type": "string" + }, + "limit": { + "type": "integer" + }, + "modifiers": { + "$ref": "#/definitions/HogQLQueryModifiers", + "description": "Modifiers used when performing the query" + }, + "offset": { + "type": "integer" + }, + "operand": { + "$ref": "#/definitions/FilterLogicalOperator" + }, + "order": { + "anyOf": [ + { + "$ref": "#/definitions/DurationType" + }, + { + "const": "start_time", + "type": "string" + }, + { + "const": "console_error_count", + "type": "string" + } + ] + }, + "person_uuid": { + "type": "string" + }, + "properties": { + "items": { + "$ref": "#/definitions/AnyPropertyFilter" + }, + "type": "array" + }, + "response": { + "$ref": "#/definitions/RecordingsQueryResponse" + }, + "session_ids": { + "items": { + "type": "string" + }, + "type": "array" + } + }, + "required": ["kind", "order"], + "type": "object" + }, + "RecordingsQueryResponse": { + "additionalProperties": false, + "properties": { + "has_next": { + "type": "boolean" + }, + "results": { + "items": { + "$ref": "#/definitions/SessionRecordingType" + }, + "type": "array" + } + }, + "required": ["results", "has_next"], + "type": "object" + }, "RefreshType": { "anyOf": [ { @@ -8625,6 +8949,88 @@ "required": ["key", "operator", "type"], "type": "object" }, + "SessionRecordingType": { + "additionalProperties": false, + "properties": { + "active_seconds": { + "type": "number" + }, + "click_count": { + "type": "number" + }, + "console_error_count": { + "type": "number" + }, + "console_log_count": { + "type": "number" + }, + "console_warn_count": { + "type": "number" + }, + "distinct_id": { + "type": "string" + }, + "email": { + "type": "string" + }, + "end_time": { + "description": "When the recording ends in ISO format.", + "type": "string" + }, + "id": { + "type": "string" + }, + "inactive_seconds": { + "type": "number" + }, + "keypress_count": { + "type": "number" + }, + "matching_events": { + "description": "List of matching events. *", + "items": { + "$ref": "#/definitions/MatchedRecording" + }, + "type": "array" + }, + "mouse_activity_count": { + "description": "count of all mouse activity in the recording, not just clicks", + "type": "number" + }, + "person": { + "$ref": "#/definitions/PersonType" + }, + "recording_duration": { + "description": "Length of recording in seconds.", + "type": "number" + }, + "snapshot_source": { + "enum": ["web", "mobile", "unknown"], + "type": "string" + }, + "start_time": { + "description": "When the recording starts in ISO format.", + "type": "string" + }, + "start_url": { + "type": "string" + }, + "storage": { + "description": "Where this recording information was loaded from", + "enum": ["object_storage_lts", "object_storage"], + "type": "string" + }, + "summary": { + "type": "string" + }, + "viewed": { + "description": "Whether this recording has been viewed already.", + "type": "boolean" + } + }, + "required": ["id", "viewed", "recording_duration", "start_time", "end_time", "snapshot_source"], + "type": "object" + }, "SessionsTimelineQuery": { "additionalProperties": false, "properties": { @@ -9264,6 +9670,35 @@ "required": ["results"], "type": "object" }, + "UserBasicType": { + "additionalProperties": false, + "properties": { + "distinct_id": { + "type": "string" + }, + "email": { + "type": "string" + }, + "first_name": { + "type": "string" + }, + "hedgehog_config": { + "$ref": "#/definitions/MinimalHedgehogConfig" + }, + "id": { + "type": "number" + }, + "is_email_verified": {}, + "last_name": { + "type": "string" + }, + "uuid": { + "type": "string" + } + }, + "required": ["distinct_id", "email", "first_name", "id", "uuid"], + "type": "object" + }, "VizSpecificOptions": { "additionalProperties": false, "description": "Chart specific rendering options. Use ChartRenderingMetadata for non-serializable values, e.g. onClick handlers", diff --git a/frontend/src/queries/schema.ts b/frontend/src/queries/schema.ts index 0429cfc1bea93..97994e2cf9a1f 100644 --- a/frontend/src/queries/schema.ts +++ b/frontend/src/queries/schema.ts @@ -9,8 +9,10 @@ import { ChartDisplayCategory, ChartDisplayType, CountPerActorMathType, + DurationType, EventPropertyFilter, EventType, + FilterLogicalOperator, FilterType, FunnelsFilterType, GroupMathType, @@ -20,14 +22,17 @@ import { IntervalType, LifecycleFilterType, LifecycleToggle, + LogEntryPropertyFilter, PathsFilterType, PersonPropertyFilter, PropertyGroupFilter, PropertyMathType, RetentionFilterType, SessionPropertyFilter, + SessionRecordingType, StickinessFilterType, TrendsFilterType, + UserBasicType, } from '~/types' export { ChartDisplayCategory } @@ -63,6 +68,7 @@ export enum NodeKind { FunnelsActorsQuery = 'FunnelsActorsQuery', FunnelCorrelationActorsQuery = 'FunnelCorrelationActorsQuery', SessionsTimelineQuery = 'SessionsTimelineQuery', + RecordingsQuery = 'RecordingsQuery', SessionAttributionExplorerQuery = 'SessionAttributionExplorerQuery', ErrorTrackingQuery = 'ErrorTrackingQuery', @@ -264,6 +270,29 @@ export interface HogQuery extends DataNode { code?: string } +export interface RecordingsQueryResponse { + results: SessionRecordingType[] + has_next: boolean +} + +export interface RecordingsQuery extends DataNode { + kind: NodeKind.RecordingsQuery + date_from?: string | null + date_to?: string | null + events?: FilterType['events'] + actions?: FilterType['actions'] + properties?: AnyPropertyFilter[] + console_log_filters?: LogEntryPropertyFilter[] + having_predicates?: AnyPropertyFilter[] // duration and snapshot_source filters + filter_test_accounts?: boolean + operand?: FilterLogicalOperator + session_ids?: string[] + person_uuid?: string + order: DurationType | 'start_time' | 'console_error_count' + limit?: integer + offset?: integer +} + export interface HogQLNotice { start?: integer end?: integer @@ -1688,11 +1717,47 @@ export interface DashboardFilter { properties?: AnyPropertyFilter[] | null } -export interface AbsoluteThreshold { - lower?: number | null - upper?: number | null +export interface InsightsThresholdAbsolute { + lower?: number + upper?: number +} + +export interface InsightThreshold { + absoluteThreshold?: InsightsThresholdAbsolute + // More types of thresholds or conditions can be added here +} + +export interface AlertCondition { + // Conditions in addition to the separate threshold + // TODO: Think about things like relative thresholds, rate of change, etc. +} + +export interface AlertCheck { + id: string + created_at: string + calculated_value: number + state: string + targets_notified: boolean +} + +export interface AlertTypeBase { + name: string + condition: AlertCondition + enabled: boolean + insight: number +} + +export interface AlertTypeWrite extends AlertTypeBase { + subscribed_users: integer[] } -export interface AnomalyCondition { - absoluteThreshold: AbsoluteThreshold +export interface AlertType extends AlertTypeBase { + id: string + subscribed_users: UserBasicType[] + threshold: { configuration: InsightThreshold } + created_by: UserBasicType + created_at: string + state: string + last_notified_at: string + checks: AlertCheck[] } diff --git a/frontend/src/scenes/data-warehouse/new/dataWarehouseTableLogic.tsx b/frontend/src/scenes/data-warehouse/new/dataWarehouseTableLogic.tsx index 7f989a84f2428..92165933b107f 100644 --- a/frontend/src/scenes/data-warehouse/new/dataWarehouseTableLogic.tsx +++ b/frontend/src/scenes/data-warehouse/new/dataWarehouseTableLogic.tsx @@ -92,6 +92,13 @@ export const dataWarehouseTableLogic = kea([ table: { defaults: { ...NEW_WAREHOUSE_TABLE } as DataWarehouseTable, errors: ({ name, url_pattern, credential, format }) => { + if (url_pattern?.startsWith('s3://')) { + return { + url_pattern: + 'Please use the https version of your bucket url e.g. https://your-org.s3.amazonaws.com/airbyte/stripe/invoices/*.pqt', + } + } + return { name: !name && 'Please enter a name.', url_pattern: !url_pattern && 'Please enter a url pattern.', diff --git a/frontend/src/scenes/insights/InsightPageHeader.tsx b/frontend/src/scenes/insights/InsightPageHeader.tsx index a2fd6491eacd7..86e730efafe5b 100644 --- a/frontend/src/scenes/insights/InsightPageHeader.tsx +++ b/frontend/src/scenes/insights/InsightPageHeader.tsx @@ -78,7 +78,7 @@ export function InsightPageHeader({ insightLogicProps }: { insightLogicProps: In isOpen={insightMode === ItemMode.Subscriptions} closeModal={() => push(urls.insightView(insight.short_id as InsightShortId))} insightShortId={insight.short_id} - subscriptionId={itemId} + subscriptionId={typeof itemId === 'number' || itemId === 'new' ? itemId : null} /> push(urls.insightView(insight.short_id as InsightShortId))} isOpen={insightMode === ItemMode.Alerts} insightLogicProps={insightLogicProps} + insightId={insight.id as number} insightShortId={insight.short_id as InsightShortId} - alertId={itemId} + alertId={typeof itemId === 'string' ? itemId : null} /> @@ -150,7 +151,6 @@ export function InsightPageHeader({ insightLogicProps }: { insightLogicProps: In Share or embed - {exportContext ? ( ) : null} + )} diff --git a/frontend/src/scenes/insights/filters/ActionFilter/ActionFilterRow/ActionFilterRow.tsx b/frontend/src/scenes/insights/filters/ActionFilter/ActionFilterRow/ActionFilterRow.tsx index a1ecdd39c48c4..8480c57d1f4e5 100644 --- a/frontend/src/scenes/insights/filters/ActionFilter/ActionFilterRow/ActionFilterRow.tsx +++ b/frontend/src/scenes/insights/filters/ActionFilter/ActionFilterRow/ActionFilterRow.tsx @@ -346,7 +346,7 @@ export function ActionFilterRow({ } - // title="Delete graph series" + title="Delete graph series" data-attr={`delete-prop-filter-${index}`} noPadding={!enablePopup} onClick={() => { diff --git a/frontend/src/scenes/insights/insightSceneLogic.tsx b/frontend/src/scenes/insights/insightSceneLogic.tsx index 23b8ddb6e8c0c..cf1c390cd0138 100644 --- a/frontend/src/scenes/insights/insightSceneLogic.tsx +++ b/frontend/src/scenes/insights/insightSceneLogic.tsx @@ -71,10 +71,16 @@ export const insightSceneLogic = kea([ }, ], itemId: [ - null as null | number | 'new', + null as null | string | number, { setSceneState: (_, { itemId }) => - itemId !== undefined ? (itemId === 'new' ? 'new' : parseInt(itemId, 10)) : null, + itemId !== undefined + ? itemId === 'new' + ? 'new' + : Number.isInteger(+itemId) + ? parseInt(itemId, 10) + : itemId + : null, }, ], insightLogicRef: [ diff --git a/frontend/src/scenes/notebooks/Notebook/migrations/migrate.ts b/frontend/src/scenes/notebooks/Notebook/migrations/migrate.ts index b90653f251135..dcf01df8b0d2a 100644 --- a/frontend/src/scenes/notebooks/Notebook/migrations/migrate.ts +++ b/frontend/src/scenes/notebooks/Notebook/migrations/migrate.ts @@ -33,7 +33,7 @@ import { TrendsFilter, TrendsFilterLegacy, } from '~/queries/schema' -import { FunnelExclusionLegacy, NotebookNodeType, NotebookType, RecordingFilters } from '~/types' +import { FunnelExclusionLegacy, LegacyRecordingFilters, NotebookNodeType, NotebookType } from '~/types' // NOTE: Increment this number when you add a new content migration // It will bust the cache on the localContent in the notebookLogic @@ -62,8 +62,8 @@ function convertPlaylistFiltersToUniversalFilters(content: JSONContent[]): JSONC } // Legacy attrs on Notebook playlist nodes - const simpleFilters = node.attrs?.simpleFilters as RecordingFilters - const filters = node.attrs?.filters as RecordingFilters + const simpleFilters = node.attrs?.simpleFilters as LegacyRecordingFilters + const filters = node.attrs?.filters as LegacyRecordingFilters const { universalFilters } = node.attrs as NotebookNodePlaylistAttributes diff --git a/frontend/src/scenes/saved-insights/SavedInsights.tsx b/frontend/src/scenes/saved-insights/SavedInsights.tsx index bc1c73c036c55..ed0b4fc5110a7 100644 --- a/frontend/src/scenes/saved-insights/SavedInsights.tsx +++ b/frontend/src/scenes/saved-insights/SavedInsights.tsx @@ -17,6 +17,7 @@ import { IconStickiness, IconTrends, IconUserPaths, + IconVideoCamera, IconWarning, } from '@posthog/icons' import { LemonSelectOptions } from '@posthog/lemon-ui' @@ -315,6 +316,12 @@ export const QUERY_TYPES_METADATA: Record = { icon: IconWarning, inMenu: false, }, + [NodeKind.RecordingsQuery]: { + name: 'Session Recordings', + description: 'View available recordings', + icon: IconVideoCamera, + inMenu: false, + }, } export const INSIGHT_TYPE_OPTIONS: LemonSelectOptions = [ diff --git a/frontend/src/scenes/session-recordings/filters/ReplayTaxonomicFilters.tsx b/frontend/src/scenes/session-recordings/filters/ReplayTaxonomicFilters.tsx index a87d5bc1e25b2..0379373bed975 100644 --- a/frontend/src/scenes/session-recordings/filters/ReplayTaxonomicFilters.tsx +++ b/frontend/src/scenes/session-recordings/filters/ReplayTaxonomicFilters.tsx @@ -24,22 +24,26 @@ export function ReplayTaxonomicFilters({ onChange }: ReplayTaxonomicFiltersProps return !!filters.find((f) => f.type === PropertyFilterType.Recording && f.key === key) } - const recordingProperties = [ + const properties = [ { label: 'Visited page', key: 'visited_page', + propertyFilterType: PropertyFilterType.Recording, }, { label: 'Platform', key: 'snapshot_source', + propertyFilterType: PropertyFilterType.Recording, }, { label: 'Console log level', key: 'console_log_level', + propertyFilterType: PropertyFilterType.LogEntry, }, { label: 'Console log text', key: 'console_log_query', + propertyFilterType: PropertyFilterType.LogEntry, }, ] @@ -48,12 +52,12 @@ export function ReplayTaxonomicFilters({ onChange }: ReplayTaxonomicFiltersProps
Replay properties
    - {recordingProperties.map(({ key, label }) => ( + {properties.map(({ key, label, propertyFilterType }) => ( onChange(key, {})} + onClick={() => onChange(key, { propertyFilterType: propertyFilterType })} disabledReason={hasFilter(key) ? `${label} filter already added` : undefined} > {label} diff --git a/frontend/src/scenes/session-recordings/player/inspector/inspectorListFiltering.test.ts b/frontend/src/scenes/session-recordings/player/inspector/inspectorListFiltering.test.ts new file mode 100644 index 0000000000000..ae854da7b1330 --- /dev/null +++ b/frontend/src/scenes/session-recordings/player/inspector/inspectorListFiltering.test.ts @@ -0,0 +1,182 @@ +import { filterInspectorListItems } from 'scenes/session-recordings/player/inspector/inspectorListFiltering' +import { + InspectorListBrowserVisibility, + InspectorListItemDoctor, + InspectorListItemEvent, + InspectorListOfflineStatusChange, +} from 'scenes/session-recordings/player/inspector/playerInspectorLogic' +import { SharedListMiniFilter } from 'scenes/session-recordings/player/playerSettingsLogic' + +import { PerformanceEvent, SessionRecordingPlayerTab } from '~/types' + +describe('filtering inspector list items', () => { + describe('the all tab', () => { + it('includes browser visibility', () => { + expect( + filterInspectorListItems({ + allItems: [ + { + type: 'browser-visibility', + } as InspectorListBrowserVisibility, + ], + tab: SessionRecordingPlayerTab.ALL, + miniFiltersByKey: { 'all-everything': { enabled: true } as unknown as SharedListMiniFilter }, + showOnlyMatching: false, + showMatchingEventsFilter: false, + windowIdFilter: null, + }) + ).toHaveLength(1) + }) + + it('hides doctor items in everything mode', () => { + const filteredItems = filterInspectorListItems({ + allItems: [ + { + type: 'browser-visibility', + } as InspectorListBrowserVisibility, + { + type: 'doctor', + } as InspectorListItemDoctor, + ], + tab: SessionRecordingPlayerTab.ALL, + miniFiltersByKey: { 'all-everything': { enabled: true } as unknown as SharedListMiniFilter }, + showOnlyMatching: false, + showMatchingEventsFilter: false, + windowIdFilter: null, + }) + expect(filteredItems.map((item) => item.type)).toEqual(['browser-visibility']) + }) + }) + + describe('the events tab', () => { + it('filters by window id', () => { + expect( + filterInspectorListItems({ + allItems: [ + { + type: SessionRecordingPlayerTab.EVENTS, + windowId: 'this window', + data: { event: '$exception' } as unknown as PerformanceEvent, + } as unknown as InspectorListItemEvent, + { + type: SessionRecordingPlayerTab.EVENTS, + windowId: 'a different window', + data: { event: '$exception' } as unknown as PerformanceEvent, + } as unknown as InspectorListItemEvent, + ], + tab: SessionRecordingPlayerTab.EVENTS, + miniFiltersByKey: { 'events-all': { enabled: true } as unknown as SharedListMiniFilter }, + showOnlyMatching: false, + showMatchingEventsFilter: false, + windowIdFilter: 'a different window', + }) + ).toHaveLength(1) + }) + + it('excludes browser visibility on console filter', () => { + expect( + filterInspectorListItems({ + allItems: [ + { + type: 'browser-visibility', + } as InspectorListBrowserVisibility, + ], + tab: SessionRecordingPlayerTab.EVENTS, + miniFiltersByKey: { 'all-everything': { enabled: false } as unknown as SharedListMiniFilter }, + showOnlyMatching: false, + showMatchingEventsFilter: false, + windowIdFilter: null, + }) + ).toHaveLength(0) + }) + + it('excludes browser visibility when show only matching', () => { + expect( + filterInspectorListItems({ + allItems: [ + { + type: 'browser-visibility', + } as InspectorListBrowserVisibility, + ], + tab: SessionRecordingPlayerTab.EVENTS, + miniFiltersByKey: { 'all-everything': { enabled: true } as unknown as SharedListMiniFilter }, + showOnlyMatching: true, + showMatchingEventsFilter: true, + windowIdFilter: null, + }) + ).toHaveLength(0) + }) + }) + + describe('the doctor tab', () => { + it('ignores events that are not exceptions', () => { + expect( + filterInspectorListItems({ + allItems: [ + { + type: SessionRecordingPlayerTab.EVENTS, + data: { event: 'an event' } as unknown as PerformanceEvent, + } as unknown as InspectorListItemEvent, + ], + tab: SessionRecordingPlayerTab.DOCTOR, + miniFiltersByKey: {}, + showOnlyMatching: false, + showMatchingEventsFilter: false, + windowIdFilter: null, + }) + ).toHaveLength(0) + }) + + it('includes events that are exceptions', () => { + expect( + filterInspectorListItems({ + allItems: [ + { + type: SessionRecordingPlayerTab.EVENTS, + data: { event: '$exception' } as unknown as PerformanceEvent, + } as unknown as InspectorListItemEvent, + ], + tab: SessionRecordingPlayerTab.DOCTOR, + miniFiltersByKey: {}, + showOnlyMatching: false, + showMatchingEventsFilter: false, + windowIdFilter: null, + }) + ).toHaveLength(1) + }) + + it('includes browser offline status', () => { + expect( + filterInspectorListItems({ + allItems: [ + { + type: 'offline-status', + } as unknown as InspectorListOfflineStatusChange, + ], + tab: SessionRecordingPlayerTab.DOCTOR, + miniFiltersByKey: {}, + showOnlyMatching: false, + showMatchingEventsFilter: false, + windowIdFilter: null, + }) + ).toHaveLength(1) + }) + + it('includes browser visibility status', () => { + expect( + filterInspectorListItems({ + allItems: [ + { + type: 'browser-visibility', + } as InspectorListBrowserVisibility, + ], + tab: SessionRecordingPlayerTab.DOCTOR, + miniFiltersByKey: {}, + showOnlyMatching: false, + showMatchingEventsFilter: false, + windowIdFilter: null, + }) + ).toHaveLength(1) + }) + }) +}) diff --git a/frontend/src/scenes/session-recordings/player/inspector/inspectorListFiltering.ts b/frontend/src/scenes/session-recordings/player/inspector/inspectorListFiltering.ts index fe262dbef0ad1..481170e46f659 100644 --- a/frontend/src/scenes/session-recordings/player/inspector/inspectorListFiltering.ts +++ b/frontend/src/scenes/session-recordings/player/inspector/inspectorListFiltering.ts @@ -109,7 +109,8 @@ export function filterInspectorListItems({ const inspectorTabFilters: Record boolean> = { [SessionRecordingPlayerTab.ALL]: (item: InspectorListItem) => { - const isAllEverything = miniFiltersByKey['all-everything']?.enabled === true + // even in everything mode we don't show doctor events + const isAllEverything = miniFiltersByKey['all-everything']?.enabled === true && !isDoctorEvent(item) const isAllAutomatic = !!miniFiltersByKey['all-automatic']?.enabled && (isOfflineStatusChange(item) || diff --git a/frontend/src/scenes/session-recordings/player/inspector/playerInspectorLogic.test.ts b/frontend/src/scenes/session-recordings/player/inspector/playerInspectorLogic.test.ts index 37b5ee585fdce..4f0bf12fc81cd 100644 --- a/frontend/src/scenes/session-recordings/player/inspector/playerInspectorLogic.test.ts +++ b/frontend/src/scenes/session-recordings/player/inspector/playerInspectorLogic.test.ts @@ -1,17 +1,9 @@ import { expectLogic } from 'kea-test-utils' import { featureFlagLogic } from 'lib/logic/featureFlagLogic' -import { filterInspectorListItems } from 'scenes/session-recordings/player/inspector/inspectorListFiltering' -import { - InspectorListBrowserVisibility, - InspectorListItemEvent, - InspectorListOfflineStatusChange, - playerInspectorLogic, -} from 'scenes/session-recordings/player/inspector/playerInspectorLogic' -import { SharedListMiniFilter } from 'scenes/session-recordings/player/playerSettingsLogic' +import { playerInspectorLogic } from 'scenes/session-recordings/player/inspector/playerInspectorLogic' import { useMocks } from '~/mocks/jest' import { initKeaTests } from '~/test/init' -import { PerformanceEvent, SessionRecordingPlayerTab } from '~/types' const playerLogicProps = { sessionRecordingId: '1', playerKey: 'playlist' } @@ -30,157 +22,6 @@ describe('playerInspectorLogic', () => { logic.mount() }) - describe('filtering inspector list items', () => { - describe('the events tab', () => { - it('filters by window id', () => { - expect( - filterInspectorListItems({ - allItems: [ - { - type: SessionRecordingPlayerTab.EVENTS, - windowId: 'this window', - data: { event: '$exception' } as unknown as PerformanceEvent, - } as unknown as InspectorListItemEvent, - { - type: SessionRecordingPlayerTab.EVENTS, - windowId: 'a different window', - data: { event: '$exception' } as unknown as PerformanceEvent, - } as unknown as InspectorListItemEvent, - ], - tab: SessionRecordingPlayerTab.EVENTS, - miniFiltersByKey: { 'events-all': { enabled: true } as unknown as SharedListMiniFilter }, - showOnlyMatching: false, - showMatchingEventsFilter: false, - windowIdFilter: 'a different window', - }) - ).toHaveLength(1) - }) - - it('includes browser visibility', () => { - expect( - filterInspectorListItems({ - allItems: [ - { - type: 'browser-visibility', - } as InspectorListBrowserVisibility, - ], - tab: SessionRecordingPlayerTab.ALL, - miniFiltersByKey: { 'all-everything': { enabled: true } as unknown as SharedListMiniFilter }, - showOnlyMatching: false, - showMatchingEventsFilter: false, - windowIdFilter: null, - }) - ).toHaveLength(1) - }) - - it('excludes browser visibility on console filter', () => { - expect( - filterInspectorListItems({ - allItems: [ - { - type: 'browser-visibility', - } as InspectorListBrowserVisibility, - ], - tab: SessionRecordingPlayerTab.EVENTS, - miniFiltersByKey: { 'all-everything': { enabled: false } as unknown as SharedListMiniFilter }, - showOnlyMatching: false, - showMatchingEventsFilter: false, - windowIdFilter: null, - }) - ).toHaveLength(0) - }) - - it('excludes browser visibility when show only matching', () => { - expect( - filterInspectorListItems({ - allItems: [ - { - type: 'browser-visibility', - } as InspectorListBrowserVisibility, - ], - tab: SessionRecordingPlayerTab.EVENTS, - miniFiltersByKey: { 'all-everything': { enabled: true } as unknown as SharedListMiniFilter }, - showOnlyMatching: true, - showMatchingEventsFilter: true, - windowIdFilter: null, - }) - ).toHaveLength(0) - }) - }) - - describe('the doctor tab', () => { - it('ignores events that are not exceptions', () => { - expect( - filterInspectorListItems({ - allItems: [ - { - type: SessionRecordingPlayerTab.EVENTS, - data: { event: 'an event' } as unknown as PerformanceEvent, - } as unknown as InspectorListItemEvent, - ], - tab: SessionRecordingPlayerTab.DOCTOR, - miniFiltersByKey: {}, - showOnlyMatching: false, - showMatchingEventsFilter: false, - windowIdFilter: null, - }) - ).toHaveLength(0) - }) - - it('includes events that are exceptions', () => { - expect( - filterInspectorListItems({ - allItems: [ - { - type: SessionRecordingPlayerTab.EVENTS, - data: { event: '$exception' } as unknown as PerformanceEvent, - } as unknown as InspectorListItemEvent, - ], - tab: SessionRecordingPlayerTab.DOCTOR, - miniFiltersByKey: {}, - showOnlyMatching: false, - showMatchingEventsFilter: false, - windowIdFilter: null, - }) - ).toHaveLength(1) - }) - - it('includes browser offline status', () => { - expect( - filterInspectorListItems({ - allItems: [ - { - type: 'offline-status', - } as unknown as InspectorListOfflineStatusChange, - ], - tab: SessionRecordingPlayerTab.DOCTOR, - miniFiltersByKey: {}, - showOnlyMatching: false, - showMatchingEventsFilter: false, - windowIdFilter: null, - }) - ).toHaveLength(1) - }) - - it('includes browser visibility status', () => { - expect( - filterInspectorListItems({ - allItems: [ - { - type: 'browser-visibility', - } as InspectorListBrowserVisibility, - ], - tab: SessionRecordingPlayerTab.DOCTOR, - miniFiltersByKey: {}, - showOnlyMatching: false, - showMatchingEventsFilter: false, - windowIdFilter: null, - }) - ).toHaveLength(1) - }) - }) - }) - describe('setWindowIdFilter', () => { it('happy case', async () => { await expectLogic(logic).toMatchValues({ diff --git a/frontend/src/scenes/session-recordings/player/inspector/playerInspectorLogic.ts b/frontend/src/scenes/session-recordings/player/inspector/playerInspectorLogic.ts index 35c0a0c7b8d9c..a212523f397e5 100644 --- a/frontend/src/scenes/session-recordings/player/inspector/playerInspectorLogic.ts +++ b/frontend/src/scenes/session-recordings/player/inspector/playerInspectorLogic.ts @@ -15,7 +15,7 @@ import { import { filterInspectorListItems } from 'scenes/session-recordings/player/inspector/inspectorListFiltering' import { playerSettingsLogic } from 'scenes/session-recordings/player/playerSettingsLogic' import { - convertUniversalFiltersToLegacyFilters, + convertUniversalFiltersToRecordingsQuery, MatchingEventsMatchType, } from 'scenes/session-recordings/playlist/sessionRecordingsPlaylistLogic' @@ -229,7 +229,7 @@ export const playerInspectorLogic = kea([ throw new Error('Backend matching events type must include its filters') } const params = toParams({ - ...convertUniversalFiltersToLegacyFilters(filters), + ...convertUniversalFiltersToRecordingsQuery(filters), session_ids: [props.sessionRecordingId], }) const response = await api.recordings.getMatchingEvents(params) diff --git a/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistLogic.test.ts b/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistLogic.test.ts index d0104691cfa88..695db20866757 100644 --- a/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistLogic.test.ts +++ b/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistLogic.test.ts @@ -8,7 +8,7 @@ import { FilterLogicalOperator, PropertyFilterType, PropertyOperator } from '~/t import { sessionRecordingDataLogic } from '../player/sessionRecordingDataLogic' import { convertLegacyFiltersToUniversalFilters, - convertUniversalFiltersToLegacyFilters, + convertUniversalFiltersToRecordingsQuery, DEFAULT_RECORDING_FILTERS, sessionRecordingsPlaylistLogic, } from './sessionRecordingsPlaylistLogic' @@ -67,7 +67,10 @@ describe('sessionRecordingsPlaylistLogic', () => { results: ['Recordings filtered by date'], }, ] - } else if (JSON.parse(searchParams.get('session_recording_duration') ?? '{}')['value'] === 600) { + } else if ( + (searchParams.get('having_predicates')?.length || 0) > 0 && + JSON.parse(searchParams.get('having_predicates') || '[]')[0]['value'] === 600 + ) { return [ 200, { @@ -611,8 +614,8 @@ describe('sessionRecordingsPlaylistLogic', () => { type: FilterLogicalOperator.And, values: [ { - type: PropertyFilterType.Recording, - key: 'console_log_level', + type: PropertyFilterType.LogEntry, + key: 'level', operator: PropertyOperator.IContains, value: ['warn', 'error'], }, @@ -634,8 +637,8 @@ describe('sessionRecordingsPlaylistLogic', () => { type: FilterLogicalOperator.And, values: [ { - type: PropertyFilterType.Recording, - key: 'console_log_query', + type: PropertyFilterType.LogEntry, + key: 'message', operator: PropertyOperator.Exact, value: 'this is a test', }, @@ -668,8 +671,8 @@ describe('sessionRecordingsPlaylistLogic', () => { type: FilterLogicalOperator.And, values: [ { - type: PropertyFilterType.Recording, - key: 'console_log_level', + type: PropertyFilterType.LogEntry, + key: 'level', operator: PropertyOperator.IContains, value: ['warn', 'error'], }, @@ -707,9 +710,9 @@ describe('sessionRecordingsPlaylistLogic', () => { }) }) - describe('convertUniversalFiltersToLegacyFilters', () => { + describe('convertUniversalFiltersToRecordingsQuery', () => { it('expands the visited_page filter to a pageview with $current_url property', () => { - const result = convertUniversalFiltersToLegacyFilters({ + const result = convertUniversalFiltersToRecordingsQuery({ ...DEFAULT_RECORDING_FILTERS, filter_group: { type: FilterLogicalOperator.And, @@ -808,15 +811,15 @@ describe('sessionRecordingsPlaylistLogic', () => { { key: 'email', value: ['email@posthog.com'], operator: 'exact', type: 'person' }, { key: 'email', value: ['test@posthog.com'], operator: 'exact', type: 'person' }, { - key: 'console_log_level', + key: 'level', operator: 'exact', - type: 'recording', + type: 'log_entry', value: ['info', 'warn'], }, { - key: 'console_log_query', + key: 'message', operator: 'exact', - type: 'recording', + type: 'log_entry', value: ['this is a query log'], }, ], diff --git a/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistLogic.ts b/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistLogic.ts index 644fa16ab97c0..12336da783084 100644 --- a/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistLogic.ts +++ b/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistLogic.ts @@ -6,27 +6,29 @@ import { subscriptions } from 'kea-subscriptions' import api from 'lib/api' import { isAnyPropertyfilter } from 'lib/components/PropertyFilters/utils' import { DEFAULT_UNIVERSAL_GROUP_FILTER } from 'lib/components/UniversalFilters/universalFiltersLogic' -import { isActionFilter, isEventFilter, isRecordingPropertyFilter } from 'lib/components/UniversalFilters/utils' +import { + isActionFilter, + isEventFilter, + isLogEntryPropertyFilter, + isRecordingPropertyFilter, +} from 'lib/components/UniversalFilters/utils' import { featureFlagLogic } from 'lib/logic/featureFlagLogic' import { objectClean } from 'lib/utils' import { eventUsageLogic } from 'lib/utils/eventUsageLogic' import posthog from 'posthog-js' +import { NodeKind, RecordingsQuery, RecordingsQueryResponse } from '~/queries/schema' import { - AnyPropertyFilter, - DurationType, EntityTypes, - FilterableLogLevel, FilterLogicalOperator, FilterType, + LegacyRecordingFilters, + LogEntryPropertyFilter, PropertyFilterType, PropertyOperator, RecordingDurationFilter, - RecordingFilters, - RecordingPropertyFilter, RecordingUniversalFilters, SessionRecordingId, - SessionRecordingsResponse, SessionRecordingType, } from '~/types' @@ -36,12 +38,11 @@ import { sessionRecordingsListPropertiesLogic } from './sessionRecordingsListPro import type { sessionRecordingsPlaylistLogicType } from './sessionRecordingsPlaylistLogicType' export type PersonUUID = string -export type SessionOrderingType = DurationType | 'start_time' | 'console_error_count' interface Params { filters?: RecordingUniversalFilters - simpleFilters?: RecordingFilters - advancedFilters?: RecordingFilters + simpleFilters?: LegacyRecordingFilters + advancedFilters?: LegacyRecordingFilters sessionRecordingId?: SessionRecordingId } @@ -65,7 +66,7 @@ interface BackendEventsMatching { } export type MatchingEventsMatchType = NoEventsToMatch | EventNamesMatching | EventUUIDsMatching | BackendEventsMatching -export type SimpleFiltersType = Pick +export type SimpleFiltersType = Pick export const RECORDINGS_LIMIT = 20 export const PINNED_RECORDINGS_LIMIT = 100 // NOTE: This is high but avoids the need for pagination for now... @@ -94,7 +95,7 @@ export const getDefaultFilters = (personUUID?: PersonUUID): RecordingUniversalFi return personUUID ? DEFAULT_PERSON_RECORDING_FILTERS : DEFAULT_RECORDING_FILTERS } -const capturePartialFilters = (filters: Partial): void => { +const capturePartialFilters = (filters: Partial): void => { // capture only the partial filters applied (not the full filters object) // take each key from the filter and change it to `partial_filter_chosen_${key}` const partialFilters = Object.keys(filters).reduce((acc, key) => { @@ -107,33 +108,31 @@ const capturePartialFilters = (filters: Partial): void => { }) } -export function convertUniversalFiltersToLegacyFilters(universalFilters: RecordingUniversalFilters): RecordingFilters { +export function convertUniversalFiltersToRecordingsQuery(universalFilters: RecordingUniversalFilters): RecordingsQuery { const filters = filtersFromUniversalFilterGroups(universalFilters) - const properties: AnyPropertyFilter[] = [] - const events: FilterType['events'] = [] - const actions: FilterType['actions'] = [] - let console_logs: FilterableLogLevel[] = [] - let snapshot_source: AnyPropertyFilter | null = null - let console_search_query = '' + const events: RecordingsQuery['events'] = [] + const actions: RecordingsQuery['actions'] = [] + const properties: RecordingsQuery['properties'] = [] + const console_log_filters: RecordingsQuery['console_log_filters'] = [] + const having_predicates: RecordingsQuery['having_predicates'] = [] + + const durationFilter = universalFilters.duration[0] + + if (durationFilter) { + having_predicates.push(durationFilter) + } filters.forEach((f) => { if (isEventFilter(f)) { events.push(f) } else if (isActionFilter(f)) { actions.push(f) + } else if (isLogEntryPropertyFilter(f)) { + console_log_filters.push(f) } else if (isAnyPropertyfilter(f)) { - if (f.type === PropertyFilterType.Recording) { - if (f.key === 'console_log_level') { - console_logs = f.value as FilterableLogLevel[] - } else if (f.key === 'console_log_query') { - console_search_query = (f.value || '') as string - } else if (f.key === 'snapshot_source') { - const value = f.value as string[] | null - if (value) { - snapshot_source = f - } - } else if (f.key === 'visited_page') { + if (isRecordingPropertyFilter(f)) { + if (f.key === 'visited_page') { events.push({ id: '$pageview', name: '$pageview', @@ -147,6 +146,8 @@ export function convertUniversalFiltersToLegacyFilters(universalFilters: Recordi }, ], }) + } else if (f.key === 'snapshot_source' && f.value) { + having_predicates.push(f) } } else { properties.push(f) @@ -154,49 +155,48 @@ export function convertUniversalFiltersToLegacyFilters(universalFilters: Recordi } }) - const durationFilter = universalFilters.duration[0] - return { - ...universalFilters, + kind: NodeKind.RecordingsQuery, + order: 'start_time', + date_from: universalFilters.date_from, + date_to: universalFilters.date_to, properties, events, actions, - session_recording_duration: { ...durationFilter, key: 'duration' }, - duration_type_filter: durationFilter.key, - console_search_query, - console_logs, - snapshot_source, + console_log_filters, + having_predicates, + filter_test_accounts: universalFilters.filter_test_accounts, operand: universalFilters.filter_group.type, } } export function convertLegacyFiltersToUniversalFilters( - simpleFilters?: RecordingFilters, - advancedFilters?: RecordingFilters + simpleFilters?: LegacyRecordingFilters, + advancedFilters?: LegacyRecordingFilters ): RecordingUniversalFilters { - const filters = combineRecordingFilters(simpleFilters || {}, advancedFilters || {}) + const filters = combineLegacyRecordingFilters(simpleFilters || {}, advancedFilters || {}) const events = filters.events ?? [] const actions = filters.actions ?? [] const properties = filters.properties ?? [] - const logLevelFilters: RecordingPropertyFilter[] = + const logLevelFilters: LogEntryPropertyFilter[] = filters.console_logs && filters.console_logs.length > 0 ? [ { - key: 'console_log_level', + key: 'level', value: filters.console_logs, operator: PropertyOperator.Exact, - type: PropertyFilterType.Recording, + type: PropertyFilterType.LogEntry, }, ] : [] - const logQueryFilters: RecordingPropertyFilter[] = filters.console_search_query + const logQueryFilters: LogEntryPropertyFilter[] = filters.console_search_query ? [ { - key: 'console_log_query', + key: 'message', value: [filters.console_search_query], operator: PropertyOperator.Exact, - type: PropertyFilterType.Recording, + type: PropertyFilterType.LogEntry, }, ] : [] @@ -223,7 +223,10 @@ export function convertLegacyFiltersToUniversalFilters( } } -function combineRecordingFilters(simpleFilters: RecordingFilters, advancedFilters: RecordingFilters): RecordingFilters { +function combineLegacyRecordingFilters( + simpleFilters: LegacyRecordingFilters, + advancedFilters: LegacyRecordingFilters +): LegacyRecordingFilters { return { ...advancedFilters, events: [...(simpleFilters?.events || []), ...(advancedFilters?.events || [])], @@ -274,7 +277,7 @@ export const sessionRecordingsPlaylistLogic = kea) => ({ filters }), setShowFilters: (showFilters: boolean) => ({ showFilters }), setShowSettings: (showSettings: boolean) => ({ showSettings }), - setOrderBy: (orderBy: SessionOrderingType) => ({ orderBy }), + setOrderBy: (orderBy: RecordingsQuery['order']) => ({ orderBy }), resetFilters: true, setSelectedRecordingId: (id: SessionRecordingType['id'] | null) => ({ id, @@ -312,7 +315,8 @@ export const sessionRecordingsPlaylistLogic = kea, { loadEventsHaveSessionId: async () => { - const events: FilterType['events'] = convertUniversalFiltersToLegacyFilters(values.filters).events + const filters = filtersFromUniversalFilterGroups(values.filters) + const events: FilterType['events'] = filters.filter(isEventFilter) if (events === undefined || events.length === 0) { return {} @@ -329,33 +333,31 @@ export const sessionRecordingsPlaylistLogic = kea { - const params = { - // TODO: requires a backend change so will include in a separate PR - ...convertUniversalFiltersToLegacyFilters(values.filters), + const params: RecordingsQuery = { + ...convertUniversalFiltersToRecordingsQuery(values.filters), person_uuid: props.personUUID ?? '', - target_entity_order: values.orderBy, + order: values.orderBy, limit: RECORDINGS_LIMIT, } if (values.orderBy === 'start_time') { if (direction === 'older') { - params['date_to'] = - values.sessionRecordings[values.sessionRecordings.length - 1]?.start_time + params.date_to = values.sessionRecordings[values.sessionRecordings.length - 1]?.start_time } if (direction === 'newer') { - params['date_from'] = values.sessionRecordings[0]?.start_time + params.date_from = values.sessionRecordings[0]?.start_time } } else { if (direction === 'older') { - params['offset'] = values.sessionRecordings.length + params.offset = values.sessionRecordings.length } if (direction === 'newer') { - params['offset'] = 0 + params.offset = 0 } } @@ -396,7 +398,9 @@ export const sessionRecordingsPlaylistLogic = kea ({ orderBy: [ - 'start_time' as SessionOrderingType, + 'start_time' as RecordingsQuery['order'], { persist: true }, { setOrderBy: (_, { orderBy }) => orderBy, diff --git a/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistSceneLogic.ts b/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistSceneLogic.ts index a40d50e1f49b9..7aea380bae5d6 100644 --- a/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistSceneLogic.ts +++ b/frontend/src/scenes/session-recordings/playlist/sessionRecordingsPlaylistSceneLogic.ts @@ -18,7 +18,7 @@ import { urls } from 'scenes/urls' import { cohortsModel } from '~/models/cohortsModel' import { Breadcrumb, - RecordingFilters, + LegacyRecordingFilters, RecordingUniversalFilters, ReplayTabs, SessionRecordingPlaylistType, @@ -46,7 +46,7 @@ export const sessionRecordingsPlaylistSceneLogic = kea ({ filters }), + setFilters: (filters: LegacyRecordingFilters | RecordingUniversalFilters | null) => ({ filters }), loadPinnedRecordings: true, onPinnedChange: (recording: SessionRecordingType, pinned: boolean) => ({ pinned, recording }), }), @@ -116,7 +116,7 @@ export const sessionRecordingsPlaylistSceneLogic = kea ({ filters: [ - null as RecordingFilters | RecordingUniversalFilters | null, + null as LegacyRecordingFilters | RecordingUniversalFilters | null, { getPlaylistSuccess: (_, { playlist }) => playlist?.filters || null, updatePlaylistSuccess: (_, { playlist }) => playlist?.filters || null, diff --git a/frontend/src/scenes/session-recordings/utils.ts b/frontend/src/scenes/session-recordings/utils.ts index 9dfb2eab13664..273152331b5a4 100644 --- a/frontend/src/scenes/session-recordings/utils.ts +++ b/frontend/src/scenes/session-recordings/utils.ts @@ -1,9 +1,9 @@ import { UniversalFiltersGroup, UniversalFilterValue } from 'lib/components/UniversalFilters/UniversalFilters' -import { RecordingFilters, RecordingUniversalFilters } from '~/types' +import { LegacyRecordingFilters, RecordingUniversalFilters } from '~/types' export const isUniversalFilters = ( - filters: RecordingUniversalFilters | RecordingFilters + filters: RecordingUniversalFilters | LegacyRecordingFilters ): filters is RecordingUniversalFilters => { return 'filter_group' in filters } diff --git a/frontend/src/scenes/trends/viz/ActionsHorizontalBar.tsx b/frontend/src/scenes/trends/viz/ActionsHorizontalBar.tsx index 0fbdecb949d35..9866e42c4cf56 100644 --- a/frontend/src/scenes/trends/viz/ActionsHorizontalBar.tsx +++ b/frontend/src/scenes/trends/viz/ActionsHorizontalBar.tsx @@ -48,12 +48,19 @@ export function ActionsHorizontalBar({ showPersonsModal = true }: ChartParams): personsValues: _data.map((item) => item.persons), breakdownValues: _data.map((item) => item.breakdown_value), breakdownLabels: _data.map((item) => { - return formatBreakdownLabel( + const itemLabel = item.action.custom_name ?? item.action.name ?? item.action.id + if (!item.breakdown_value) { + return itemLabel + } + + const breakdownLabel = formatBreakdownLabel( item.breakdown_value, breakdownFilter, cohorts, formatPropertyValueForDisplay ) + + return `${itemLabel} - ${breakdownLabel}` }), compareLabels: _data.map((item) => item.compare_label), backgroundColor: colorList, diff --git a/frontend/src/types.ts b/frontend/src/types.ts index 66d7f2cf51c7e..7bdb1a9b8b306 100644 --- a/frontend/src/types.ts +++ b/frontend/src/types.ts @@ -33,7 +33,6 @@ import { Scene } from 'scenes/sceneTypes' import { QueryContext } from '~/queries/types' import type { - AnomalyCondition, DashboardFilter, DatabaseSchemaField, HogQLQuery, @@ -961,7 +960,7 @@ export type ActionStepProperties = export interface RecordingPropertyFilter extends BasePropertyFilter { type: PropertyFilterType.Recording - key: DurationType | 'console_log_level' | 'console_log_query' | 'snapshot_source' | 'visited_page' + key: DurationType | 'snapshot_source' | 'visited_page' operator: PropertyOperator } @@ -970,7 +969,10 @@ export interface RecordingDurationFilter extends RecordingPropertyFilter { value: number } -export type DurationType = 'duration' | 'active_seconds' | 'inactive_seconds' +export interface LogEntryPropertyFilter extends BasePropertyFilter { + type: PropertyFilterType.LogEntry + operator: PropertyOperator +} export interface LogEntryPropertyFilter extends BasePropertyFilter { type: PropertyFilterType.LogEntry @@ -985,9 +987,11 @@ export interface LogEntryMessageFilter extends LogEntryPropertyFilter { key: 'message' value: string } + +export type DurationType = 'duration' | 'active_seconds' | 'inactive_seconds' export type FilterableLogLevel = 'info' | 'warn' | 'error' -export interface RecordingFilters { +export interface LegacyRecordingFilters { date_from?: string | null date_to?: string | null events?: FilterType['events'] @@ -1010,11 +1014,6 @@ export interface RecordingUniversalFilters { filter_group: UniversalFiltersGroup } -export interface SessionRecordingsResponse { - results: SessionRecordingType[] - has_next: boolean -} - export type ErrorCluster = { cluster: number sample: string @@ -1324,7 +1323,7 @@ export interface SessionRecordingPlaylistType { created_by: UserBasicType | null last_modified_at: string last_modified_by: UserBasicType | null - filters?: RecordingFilters + filters?: LegacyRecordingFilters } export interface SessionRecordingSegmentType { @@ -4452,14 +4451,6 @@ export type HogFunctionInvocationGlobals = { > } -export interface AlertType { - id: number - name: string - insight?: number - target_value: string - anomaly_condition: AnomalyCondition -} - export type AppMetricsV2Response = { labels: string[] series: { diff --git a/hogql_parser/HogQLLexer.cpp b/hogql_parser/HogQLLexer.cpp index f5583b7fa4ac8..8c8c7c5a2b643 100644 --- a/hogql_parser/HogQLLexer.cpp +++ b/hogql_parser/HogQLLexer.cpp @@ -1,5 +1,5 @@ -// Generated from HogQLLexer.g4 by ANTLR 4.13.1 +// Generated from HogQLLexer.g4 by ANTLR 4.13.2 #include "HogQLLexer.h" @@ -45,7 +45,7 @@ ::antlr4::internal::OnceFlag hogqllexerLexerOnceFlag; #if ANTLR4_USE_THREAD_LOCAL_CACHE static thread_local #endif -HogQLLexerStaticData *hogqllexerLexerStaticData = nullptr; +std::unique_ptr hogqllexerLexerStaticData = nullptr; void hogqllexerLexerInitialize() { #if ANTLR4_USE_THREAD_LOCAL_CACHE @@ -673,7 +673,7 @@ void hogqllexerLexerInitialize() { for (size_t i = 0; i < count; i++) { staticData->decisionToDFA.emplace_back(staticData->atn->getDecisionState(i), i); } - hogqllexerLexerStaticData = staticData.release(); + hogqllexerLexerStaticData = std::move(staticData); } } diff --git a/hogql_parser/HogQLLexer.h b/hogql_parser/HogQLLexer.h index b54914ebc913e..d537cee2ba0cb 100644 --- a/hogql_parser/HogQLLexer.h +++ b/hogql_parser/HogQLLexer.h @@ -1,5 +1,5 @@ -// Generated from HogQLLexer.g4 by ANTLR 4.13.1 +// Generated from HogQLLexer.g4 by ANTLR 4.13.2 #pragma once diff --git a/hogql_parser/HogQLParser.cpp b/hogql_parser/HogQLParser.cpp index e2021ef8c06ae..55a2fca242632 100644 --- a/hogql_parser/HogQLParser.cpp +++ b/hogql_parser/HogQLParser.cpp @@ -1,5 +1,5 @@ -// Generated from HogQLParser.g4 by ANTLR 4.13.1 +// Generated from HogQLParser.g4 by ANTLR 4.13.2 #include "HogQLParserVisitor.h" @@ -40,7 +40,7 @@ ::antlr4::internal::OnceFlag hogqlparserParserOnceFlag; #if ANTLR4_USE_THREAD_LOCAL_CACHE static thread_local #endif -HogQLParserStaticData *hogqlparserParserStaticData = nullptr; +std::unique_ptr hogqlparserParserStaticData = nullptr; void hogqlparserParserInitialize() { #if ANTLR4_USE_THREAD_LOCAL_CACHE @@ -64,13 +64,12 @@ void hogqlparserParserInitialize() { "orderExpr", "ratioExpr", "settingExprList", "settingExpr", "windowExpr", "winPartitionByClause", "winOrderByClause", "winFrameClause", "winFrameExtend", "winFrameBound", "expr", "columnTypeExpr", "columnExprList", "columnExpr", - "columnArgList", "columnArgExpr", "columnLambdaExpr", "hogqlxTagElement", - "hogqlxTagAttribute", "withExprList", "withExpr", "columnIdentifier", - "nestedIdentifier", "tableExpr", "tableFunctionExpr", "tableIdentifier", - "tableArgList", "databaseIdentifier", "floatingLiteral", "numberLiteral", - "literal", "interval", "keyword", "keywordForAlias", "alias", "identifier", - "enumValue", "placeholder", "string", "templateString", "stringContents", - "fullTemplateString", "stringContentsFull" + "columnLambdaExpr", "hogqlxTagElement", "hogqlxTagAttribute", "withExprList", + "withExpr", "columnIdentifier", "nestedIdentifier", "tableExpr", "tableFunctionExpr", + "tableIdentifier", "tableArgList", "databaseIdentifier", "floatingLiteral", + "numberLiteral", "literal", "interval", "keyword", "keywordForAlias", + "alias", "identifier", "enumValue", "placeholder", "string", "templateString", + "stringContents", "fullTemplateString", "stringContentsFull" }, std::vector{ "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", "", @@ -114,7 +113,7 @@ void hogqlparserParserInitialize() { } ); static const int32_t serializedATNSegment[] = { - 4,1,159,1311,2,0,7,0,2,1,7,1,2,2,7,2,2,3,7,3,2,4,7,4,2,5,7,5,2,6,7,6, + 4,1,159,1303,2,0,7,0,2,1,7,1,2,2,7,2,2,3,7,3,2,4,7,4,2,5,7,5,2,6,7,6, 2,7,7,7,2,8,7,8,2,9,7,9,2,10,7,10,2,11,7,11,2,12,7,12,2,13,7,13,2,14, 7,14,2,15,7,15,2,16,7,16,2,17,7,17,2,18,7,18,2,19,7,19,2,20,7,20,2,21, 7,21,2,22,7,22,2,23,7,23,2,24,7,24,2,25,7,25,2,26,7,26,2,27,7,27,2,28, @@ -126,484 +125,481 @@ void hogqlparserParserInitialize() { 7,63,2,64,7,64,2,65,7,65,2,66,7,66,2,67,7,67,2,68,7,68,2,69,7,69,2,70, 7,70,2,71,7,71,2,72,7,72,2,73,7,73,2,74,7,74,2,75,7,75,2,76,7,76,2,77, 7,77,2,78,7,78,2,79,7,79,2,80,7,80,2,81,7,81,2,82,7,82,2,83,7,83,2,84, - 7,84,2,85,7,85,2,86,7,86,2,87,7,87,1,0,5,0,178,8,0,10,0,12,0,181,9,0, - 1,0,1,0,1,1,1,1,3,1,187,8,1,1,2,1,2,1,3,1,3,1,3,1,3,1,3,3,3,196,8,3,1, - 4,1,4,1,4,5,4,201,8,4,10,4,12,4,204,9,4,1,4,3,4,207,8,4,1,5,1,5,1,5,1, - 5,1,5,1,5,1,5,1,5,1,5,1,5,1,5,1,5,3,5,221,8,5,1,6,1,6,3,6,225,8,6,1,6, - 3,6,228,8,6,1,7,1,7,3,7,232,8,7,1,7,3,7,235,8,7,1,8,1,8,1,8,1,8,1,8,3, - 8,242,8,8,1,8,1,8,3,8,246,8,8,1,8,1,8,1,9,1,9,1,9,5,9,253,8,9,10,9,12, - 9,256,9,9,1,9,1,9,3,9,260,8,9,1,10,1,10,1,10,1,10,1,10,1,10,1,10,3,10, - 269,8,10,1,11,1,11,1,11,1,11,1,11,1,11,3,11,277,8,11,1,12,1,12,1,12,1, - 12,1,12,3,12,284,8,12,1,12,1,12,3,12,288,8,12,1,12,1,12,1,12,1,12,3,12, - 294,8,12,1,12,1,12,1,12,3,12,299,8,12,1,13,1,13,1,13,1,13,1,13,1,13,3, - 13,307,8,13,1,13,1,13,1,13,1,13,1,13,3,13,314,8,13,1,14,1,14,1,14,1,14, - 3,14,320,8,14,1,14,1,14,1,14,1,15,1,15,1,15,1,15,1,15,1,16,1,16,3,16, - 332,8,16,1,17,1,17,1,18,1,18,5,18,338,8,18,10,18,12,18,341,9,18,1,18, - 1,18,1,19,1,19,1,19,1,19,1,20,1,20,1,20,5,20,352,8,20,10,20,12,20,355, - 9,20,1,20,3,20,358,8,20,1,21,1,21,1,21,3,21,363,8,21,1,21,1,21,1,22,1, - 22,1,22,1,22,5,22,371,8,22,10,22,12,22,374,9,22,1,23,1,23,1,23,1,23,1, - 23,1,23,3,23,382,8,23,1,24,3,24,385,8,24,1,24,1,24,3,24,389,8,24,1,24, - 3,24,392,8,24,1,24,1,24,3,24,396,8,24,1,24,3,24,399,8,24,1,24,3,24,402, - 8,24,1,24,3,24,405,8,24,1,24,3,24,408,8,24,1,24,1,24,3,24,412,8,24,1, - 24,1,24,3,24,416,8,24,1,24,3,24,419,8,24,1,24,3,24,422,8,24,1,24,3,24, - 425,8,24,1,24,1,24,3,24,429,8,24,1,24,3,24,432,8,24,1,25,1,25,1,25,1, - 26,1,26,1,26,1,26,3,26,441,8,26,1,27,1,27,1,27,1,28,3,28,447,8,28,1,28, - 1,28,1,28,1,28,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29, - 1,29,1,29,5,29,466,8,29,10,29,12,29,469,9,29,1,30,1,30,1,30,1,31,1,31, - 1,31,1,32,1,32,1,32,1,32,1,32,1,32,1,32,1,32,3,32,485,8,32,1,33,1,33, - 1,33,1,34,1,34,1,34,1,34,1,35,1,35,1,35,1,35,1,36,1,36,1,36,1,36,3,36, - 502,8,36,1,36,1,36,1,36,1,36,3,36,508,8,36,1,36,1,36,1,36,1,36,3,36,514, - 8,36,1,36,1,36,1,36,1,36,1,36,1,36,1,36,1,36,1,36,3,36,525,8,36,3,36, - 527,8,36,1,37,1,37,1,37,1,38,1,38,1,38,1,39,1,39,1,39,3,39,538,8,39,1, - 39,3,39,541,8,39,1,39,1,39,1,39,1,39,3,39,547,8,39,1,39,1,39,1,39,1,39, - 1,39,1,39,3,39,555,8,39,1,39,1,39,1,39,1,39,5,39,561,8,39,10,39,12,39, - 564,9,39,1,40,3,40,567,8,40,1,40,1,40,1,40,3,40,572,8,40,1,40,3,40,575, - 8,40,1,40,3,40,578,8,40,1,40,1,40,3,40,582,8,40,1,40,1,40,3,40,586,8, - 40,1,40,3,40,589,8,40,3,40,591,8,40,1,40,3,40,594,8,40,1,40,1,40,3,40, - 598,8,40,1,40,1,40,3,40,602,8,40,1,40,3,40,605,8,40,3,40,607,8,40,3,40, - 609,8,40,1,41,1,41,1,41,3,41,614,8,41,1,42,1,42,1,42,1,42,1,42,1,42,1, - 42,1,42,1,42,3,42,625,8,42,1,43,1,43,1,43,1,43,3,43,631,8,43,1,44,1,44, - 1,44,5,44,636,8,44,10,44,12,44,639,9,44,1,45,1,45,3,45,643,8,45,1,45, - 1,45,3,45,647,8,45,1,45,1,45,3,45,651,8,45,1,46,1,46,1,46,1,46,3,46,657, - 8,46,3,46,659,8,46,1,47,1,47,1,47,5,47,664,8,47,10,47,12,47,667,9,47, - 1,48,1,48,1,48,1,48,1,49,3,49,674,8,49,1,49,3,49,677,8,49,1,49,3,49,680, - 8,49,1,50,1,50,1,50,1,50,1,51,1,51,1,51,1,51,1,52,1,52,1,52,1,53,1,53, - 1,53,1,53,1,53,1,53,3,53,699,8,53,1,54,1,54,1,54,1,54,1,54,1,54,1,54, - 1,54,1,54,1,54,1,54,1,54,3,54,713,8,54,1,55,1,55,1,55,1,56,1,56,1,56, - 1,56,1,56,1,56,1,56,1,56,1,56,5,56,727,8,56,10,56,12,56,730,9,56,1,56, - 3,56,733,8,56,1,56,1,56,1,56,1,56,1,56,1,56,1,56,5,56,742,8,56,10,56, - 12,56,745,9,56,1,56,3,56,748,8,56,1,56,1,56,1,56,1,56,1,56,1,56,1,56, - 5,56,757,8,56,10,56,12,56,760,9,56,1,56,3,56,763,8,56,1,56,1,56,1,56, - 1,56,1,56,3,56,770,8,56,1,56,1,56,3,56,774,8,56,1,57,1,57,1,57,5,57,779, - 8,57,10,57,12,57,782,9,57,1,57,3,57,785,8,57,1,58,1,58,1,58,3,58,790, - 8,58,1,58,1,58,1,58,1,58,1,58,4,58,797,8,58,11,58,12,58,798,1,58,1,58, - 3,58,803,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, - 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,827,8,58, + 7,84,2,85,7,85,1,0,5,0,174,8,0,10,0,12,0,177,9,0,1,0,1,0,1,1,1,1,3,1, + 183,8,1,1,2,1,2,1,3,1,3,1,3,1,3,1,3,3,3,192,8,3,1,4,1,4,1,4,5,4,197,8, + 4,10,4,12,4,200,9,4,1,4,3,4,203,8,4,1,5,1,5,1,5,1,5,1,5,1,5,1,5,1,5,1, + 5,1,5,1,5,1,5,3,5,217,8,5,1,6,1,6,3,6,221,8,6,1,6,3,6,224,8,6,1,7,1,7, + 3,7,228,8,7,1,7,3,7,231,8,7,1,8,1,8,1,8,1,8,1,8,3,8,238,8,8,1,8,1,8,3, + 8,242,8,8,1,8,1,8,1,9,1,9,1,9,5,9,249,8,9,10,9,12,9,252,9,9,1,9,1,9,3, + 9,256,8,9,1,10,1,10,1,10,1,10,1,10,1,10,1,10,3,10,265,8,10,1,11,1,11, + 1,11,1,11,1,11,1,11,3,11,273,8,11,1,12,1,12,1,12,1,12,1,12,3,12,280,8, + 12,1,12,1,12,3,12,284,8,12,1,12,1,12,1,12,1,12,3,12,290,8,12,1,12,1,12, + 1,12,3,12,295,8,12,1,13,1,13,1,13,1,13,1,13,1,13,3,13,303,8,13,1,13,1, + 13,1,13,1,13,1,13,3,13,310,8,13,1,14,1,14,1,14,1,14,3,14,316,8,14,1,14, + 1,14,1,14,1,15,1,15,1,15,1,15,1,15,1,16,1,16,3,16,328,8,16,1,17,1,17, + 1,18,1,18,5,18,334,8,18,10,18,12,18,337,9,18,1,18,1,18,1,19,1,19,1,19, + 1,19,1,20,1,20,1,20,5,20,348,8,20,10,20,12,20,351,9,20,1,20,3,20,354, + 8,20,1,21,1,21,1,21,3,21,359,8,21,1,21,1,21,1,22,1,22,1,22,1,22,5,22, + 367,8,22,10,22,12,22,370,9,22,1,23,1,23,1,23,1,23,1,23,1,23,3,23,378, + 8,23,1,24,3,24,381,8,24,1,24,1,24,3,24,385,8,24,1,24,3,24,388,8,24,1, + 24,1,24,3,24,392,8,24,1,24,3,24,395,8,24,1,24,3,24,398,8,24,1,24,3,24, + 401,8,24,1,24,3,24,404,8,24,1,24,1,24,3,24,408,8,24,1,24,1,24,3,24,412, + 8,24,1,24,3,24,415,8,24,1,24,3,24,418,8,24,1,24,3,24,421,8,24,1,24,1, + 24,3,24,425,8,24,1,24,3,24,428,8,24,1,25,1,25,1,25,1,26,1,26,1,26,1,26, + 3,26,437,8,26,1,27,1,27,1,27,1,28,3,28,443,8,28,1,28,1,28,1,28,1,28,1, + 29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,5,29,462, + 8,29,10,29,12,29,465,9,29,1,30,1,30,1,30,1,31,1,31,1,31,1,32,1,32,1,32, + 1,32,1,32,1,32,1,32,1,32,3,32,481,8,32,1,33,1,33,1,33,1,34,1,34,1,34, + 1,34,1,35,1,35,1,35,1,35,1,36,1,36,1,36,1,36,3,36,498,8,36,1,36,1,36, + 1,36,1,36,3,36,504,8,36,1,36,1,36,1,36,1,36,3,36,510,8,36,1,36,1,36,1, + 36,1,36,1,36,1,36,1,36,1,36,1,36,3,36,521,8,36,3,36,523,8,36,1,37,1,37, + 1,37,1,38,1,38,1,38,1,39,1,39,1,39,3,39,534,8,39,1,39,3,39,537,8,39,1, + 39,1,39,1,39,1,39,3,39,543,8,39,1,39,1,39,1,39,1,39,1,39,1,39,3,39,551, + 8,39,1,39,1,39,1,39,1,39,5,39,557,8,39,10,39,12,39,560,9,39,1,40,3,40, + 563,8,40,1,40,1,40,1,40,3,40,568,8,40,1,40,3,40,571,8,40,1,40,3,40,574, + 8,40,1,40,1,40,3,40,578,8,40,1,40,1,40,3,40,582,8,40,1,40,3,40,585,8, + 40,3,40,587,8,40,1,40,3,40,590,8,40,1,40,1,40,3,40,594,8,40,1,40,1,40, + 3,40,598,8,40,1,40,3,40,601,8,40,3,40,603,8,40,3,40,605,8,40,1,41,1,41, + 1,41,3,41,610,8,41,1,42,1,42,1,42,1,42,1,42,1,42,1,42,1,42,1,42,3,42, + 621,8,42,1,43,1,43,1,43,1,43,3,43,627,8,43,1,44,1,44,1,44,5,44,632,8, + 44,10,44,12,44,635,9,44,1,45,1,45,3,45,639,8,45,1,45,1,45,3,45,643,8, + 45,1,45,1,45,3,45,647,8,45,1,46,1,46,1,46,1,46,3,46,653,8,46,3,46,655, + 8,46,1,47,1,47,1,47,5,47,660,8,47,10,47,12,47,663,9,47,1,48,1,48,1,48, + 1,48,1,49,3,49,670,8,49,1,49,3,49,673,8,49,1,49,3,49,676,8,49,1,50,1, + 50,1,50,1,50,1,51,1,51,1,51,1,51,1,52,1,52,1,52,1,53,1,53,1,53,1,53,1, + 53,1,53,3,53,695,8,53,1,54,1,54,1,54,1,54,1,54,1,54,1,54,1,54,1,54,1, + 54,1,54,1,54,3,54,709,8,54,1,55,1,55,1,55,1,56,1,56,1,56,1,56,1,56,1, + 56,1,56,1,56,1,56,5,56,723,8,56,10,56,12,56,726,9,56,1,56,3,56,729,8, + 56,1,56,1,56,1,56,1,56,1,56,1,56,1,56,5,56,738,8,56,10,56,12,56,741,9, + 56,1,56,3,56,744,8,56,1,56,1,56,1,56,1,56,1,56,1,56,1,56,5,56,753,8,56, + 10,56,12,56,756,9,56,1,56,3,56,759,8,56,1,56,1,56,1,56,1,56,1,56,3,56, + 766,8,56,1,56,1,56,3,56,770,8,56,1,57,1,57,1,57,5,57,775,8,57,10,57,12, + 57,778,9,57,1,57,3,57,781,8,57,1,58,1,58,1,58,3,58,786,8,58,1,58,1,58, + 1,58,1,58,1,58,4,58,793,8,58,11,58,12,58,794,1,58,1,58,3,58,799,8,58, 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, - 1,58,3,58,844,8,58,1,58,1,58,1,58,1,58,3,58,850,8,58,1,58,3,58,853,8, - 58,1,58,3,58,856,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,866, - 8,58,1,58,1,58,1,58,1,58,3,58,872,8,58,1,58,3,58,875,8,58,1,58,3,58,878, - 8,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,886,8,58,1,58,3,58,889,8,58,1, - 58,1,58,3,58,893,8,58,1,58,3,58,896,8,58,1,58,1,58,1,58,1,58,1,58,1,58, - 1,58,1,58,1,58,1,58,1,58,1,58,3,58,910,8,58,1,58,1,58,1,58,1,58,1,58, - 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,927,8,58,1,58, - 1,58,1,58,3,58,932,8,58,1,58,1,58,3,58,936,8,58,1,58,1,58,1,58,1,58,3, - 58,942,8,58,1,58,1,58,1,58,1,58,1,58,3,58,949,8,58,1,58,1,58,1,58,1,58, - 1,58,1,58,1,58,1,58,1,58,1,58,3,58,961,8,58,1,58,1,58,3,58,965,8,58,1, - 58,3,58,968,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,977,8,58,1,58, - 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,991,8,58, + 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,823,8,58,1,58,1,58,1,58, + 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,840, + 8,58,1,58,1,58,1,58,1,58,3,58,846,8,58,1,58,3,58,849,8,58,1,58,3,58,852, + 8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,862,8,58,1,58,1,58, + 1,58,1,58,3,58,868,8,58,1,58,3,58,871,8,58,1,58,3,58,874,8,58,1,58,1, + 58,1,58,1,58,1,58,1,58,3,58,882,8,58,1,58,3,58,885,8,58,1,58,1,58,3,58, + 889,8,58,1,58,3,58,892,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1, + 58,1,58,1,58,1,58,3,58,906,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1, + 58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,923,8,58,1,58,1,58,1,58,3, + 58,928,8,58,1,58,1,58,1,58,3,58,933,8,58,1,58,1,58,1,58,1,58,3,58,939, + 8,58,1,58,1,58,1,58,1,58,1,58,3,58,946,8,58,1,58,1,58,1,58,1,58,1,58, + 1,58,1,58,1,58,1,58,1,58,3,58,958,8,58,1,58,1,58,3,58,962,8,58,1,58,3, + 58,965,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,974,8,58,1,58,1,58, + 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,988,8,58,1,58, + 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58, + 1004,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, - 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, - 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,1030,8,58,1,58,1,58, - 1,58,1,58,1,58,1,58,3,58,1038,8,58,5,58,1040,8,58,10,58,12,58,1043,9, - 58,1,59,1,59,1,59,5,59,1048,8,59,10,59,12,59,1051,9,59,1,59,3,59,1054, - 8,59,1,60,1,60,3,60,1058,8,60,1,61,1,61,1,61,1,61,5,61,1064,8,61,10,61, - 12,61,1067,9,61,1,61,3,61,1070,8,61,1,61,1,61,1,61,1,61,1,61,5,61,1077, - 8,61,10,61,12,61,1080,9,61,1,61,3,61,1083,8,61,3,61,1085,8,61,1,61,1, - 61,1,61,1,62,1,62,1,62,5,62,1093,8,62,10,62,12,62,1096,9,62,1,62,1,62, - 1,62,1,62,1,62,1,62,5,62,1104,8,62,10,62,12,62,1107,9,62,1,62,1,62,1, - 62,1,62,1,62,1,62,3,62,1115,8,62,1,62,1,62,1,62,1,62,1,62,3,62,1122,8, - 62,1,63,1,63,1,63,1,63,1,63,1,63,1,63,1,63,1,63,1,63,1,63,3,63,1135,8, - 63,1,64,1,64,1,64,5,64,1140,8,64,10,64,12,64,1143,9,64,1,64,3,64,1146, - 8,64,1,65,1,65,1,65,1,65,1,65,1,65,1,65,1,65,1,65,1,65,3,65,1158,8,65, - 1,66,1,66,1,66,1,66,3,66,1164,8,66,1,66,3,66,1167,8,66,1,67,1,67,1,67, - 5,67,1172,8,67,10,67,12,67,1175,9,67,1,68,1,68,1,68,1,68,1,68,1,68,1, - 68,1,68,1,68,3,68,1186,8,68,1,68,1,68,1,68,1,68,3,68,1192,8,68,5,68,1194, - 8,68,10,68,12,68,1197,9,68,1,69,1,69,1,69,3,69,1202,8,69,1,69,1,69,1, - 70,1,70,1,70,3,70,1209,8,70,1,70,1,70,1,71,1,71,1,71,5,71,1216,8,71,10, - 71,12,71,1219,9,71,1,71,3,71,1222,8,71,1,72,1,72,1,73,1,73,1,73,1,73, - 1,73,1,73,3,73,1232,8,73,3,73,1234,8,73,1,74,3,74,1237,8,74,1,74,1,74, - 1,74,1,74,1,74,1,74,3,74,1245,8,74,1,75,1,75,1,75,3,75,1250,8,75,1,76, - 1,76,1,77,1,77,1,78,1,78,1,79,1,79,3,79,1260,8,79,1,80,1,80,1,80,3,80, - 1265,8,80,1,81,1,81,1,81,1,81,1,82,1,82,1,82,1,82,1,83,1,83,3,83,1277, - 8,83,1,84,1,84,5,84,1281,8,84,10,84,12,84,1284,9,84,1,84,1,84,1,85,1, - 85,1,85,1,85,1,85,3,85,1293,8,85,1,86,1,86,5,86,1297,8,86,10,86,12,86, - 1300,9,86,1,86,1,86,1,87,1,87,1,87,1,87,1,87,3,87,1309,8,87,1,87,0,3, - 78,116,136,88,0,2,4,6,8,10,12,14,16,18,20,22,24,26,28,30,32,34,36,38, - 40,42,44,46,48,50,52,54,56,58,60,62,64,66,68,70,72,74,76,78,80,82,84, - 86,88,90,92,94,96,98,100,102,104,106,108,110,112,114,116,118,120,122, - 124,126,128,130,132,134,136,138,140,142,144,146,148,150,152,154,156,158, - 160,162,164,166,168,170,172,174,0,16,2,0,18,18,74,74,2,0,44,44,51,51, - 3,0,1,1,4,4,8,8,4,0,1,1,3,4,8,8,80,80,2,0,51,51,73,73,2,0,1,1,4,4,2,0, - 7,7,22,23,2,0,30,30,49,49,2,0,71,71,76,76,3,0,10,10,50,50,90,90,2,0,41, - 41,53,53,1,0,107,108,2,0,118,118,139,139,7,0,21,21,38,38,55,56,70,70, - 78,78,97,97,103,103,16,0,1,13,15,20,22,28,30,30,32,37,39,42,44,51,53, - 54,58,58,60,69,71,77,79,83,85,92,94,96,98,99,101,102,4,0,20,20,30,30, - 39,39,48,48,1479,0,179,1,0,0,0,2,186,1,0,0,0,4,188,1,0,0,0,6,190,1,0, - 0,0,8,197,1,0,0,0,10,220,1,0,0,0,12,222,1,0,0,0,14,229,1,0,0,0,16,236, - 1,0,0,0,18,249,1,0,0,0,20,261,1,0,0,0,22,270,1,0,0,0,24,278,1,0,0,0,26, - 300,1,0,0,0,28,315,1,0,0,0,30,324,1,0,0,0,32,329,1,0,0,0,34,333,1,0,0, - 0,36,335,1,0,0,0,38,344,1,0,0,0,40,348,1,0,0,0,42,362,1,0,0,0,44,366, - 1,0,0,0,46,381,1,0,0,0,48,384,1,0,0,0,50,433,1,0,0,0,52,436,1,0,0,0,54, - 442,1,0,0,0,56,446,1,0,0,0,58,452,1,0,0,0,60,470,1,0,0,0,62,473,1,0,0, - 0,64,476,1,0,0,0,66,486,1,0,0,0,68,489,1,0,0,0,70,493,1,0,0,0,72,526, - 1,0,0,0,74,528,1,0,0,0,76,531,1,0,0,0,78,546,1,0,0,0,80,608,1,0,0,0,82, - 613,1,0,0,0,84,624,1,0,0,0,86,626,1,0,0,0,88,632,1,0,0,0,90,640,1,0,0, - 0,92,658,1,0,0,0,94,660,1,0,0,0,96,668,1,0,0,0,98,673,1,0,0,0,100,681, - 1,0,0,0,102,685,1,0,0,0,104,689,1,0,0,0,106,698,1,0,0,0,108,712,1,0,0, - 0,110,714,1,0,0,0,112,773,1,0,0,0,114,775,1,0,0,0,116,935,1,0,0,0,118, - 1044,1,0,0,0,120,1057,1,0,0,0,122,1084,1,0,0,0,124,1121,1,0,0,0,126,1134, - 1,0,0,0,128,1136,1,0,0,0,130,1157,1,0,0,0,132,1166,1,0,0,0,134,1168,1, - 0,0,0,136,1185,1,0,0,0,138,1198,1,0,0,0,140,1208,1,0,0,0,142,1212,1,0, - 0,0,144,1223,1,0,0,0,146,1233,1,0,0,0,148,1236,1,0,0,0,150,1249,1,0,0, - 0,152,1251,1,0,0,0,154,1253,1,0,0,0,156,1255,1,0,0,0,158,1259,1,0,0,0, - 160,1264,1,0,0,0,162,1266,1,0,0,0,164,1270,1,0,0,0,166,1276,1,0,0,0,168, - 1278,1,0,0,0,170,1292,1,0,0,0,172,1294,1,0,0,0,174,1308,1,0,0,0,176,178, - 3,2,1,0,177,176,1,0,0,0,178,181,1,0,0,0,179,177,1,0,0,0,179,180,1,0,0, - 0,180,182,1,0,0,0,181,179,1,0,0,0,182,183,5,0,0,1,183,1,1,0,0,0,184,187, - 3,6,3,0,185,187,3,10,5,0,186,184,1,0,0,0,186,185,1,0,0,0,187,3,1,0,0, - 0,188,189,3,116,58,0,189,5,1,0,0,0,190,191,5,52,0,0,191,195,3,160,80, - 0,192,193,5,115,0,0,193,194,5,122,0,0,194,196,3,4,2,0,195,192,1,0,0,0, - 195,196,1,0,0,0,196,7,1,0,0,0,197,202,3,160,80,0,198,199,5,116,0,0,199, - 201,3,160,80,0,200,198,1,0,0,0,201,204,1,0,0,0,202,200,1,0,0,0,202,203, - 1,0,0,0,203,206,1,0,0,0,204,202,1,0,0,0,205,207,5,116,0,0,206,205,1,0, - 0,0,206,207,1,0,0,0,207,9,1,0,0,0,208,221,3,12,6,0,209,221,3,14,7,0,210, - 221,3,18,9,0,211,221,3,20,10,0,212,221,3,22,11,0,213,221,3,26,13,0,214, - 221,3,24,12,0,215,221,3,28,14,0,216,221,3,30,15,0,217,221,3,36,18,0,218, - 221,3,32,16,0,219,221,3,34,17,0,220,208,1,0,0,0,220,209,1,0,0,0,220,210, - 1,0,0,0,220,211,1,0,0,0,220,212,1,0,0,0,220,213,1,0,0,0,220,214,1,0,0, - 0,220,215,1,0,0,0,220,216,1,0,0,0,220,217,1,0,0,0,220,218,1,0,0,0,220, - 219,1,0,0,0,221,11,1,0,0,0,222,224,5,72,0,0,223,225,3,4,2,0,224,223,1, - 0,0,0,224,225,1,0,0,0,225,227,1,0,0,0,226,228,5,150,0,0,227,226,1,0,0, - 0,227,228,1,0,0,0,228,13,1,0,0,0,229,231,5,84,0,0,230,232,3,4,2,0,231, - 230,1,0,0,0,231,232,1,0,0,0,232,234,1,0,0,0,233,235,5,150,0,0,234,233, - 1,0,0,0,234,235,1,0,0,0,235,15,1,0,0,0,236,245,5,14,0,0,237,238,5,130, - 0,0,238,241,3,160,80,0,239,240,5,115,0,0,240,242,3,160,80,0,241,239,1, - 0,0,0,241,242,1,0,0,0,242,243,1,0,0,0,243,244,5,149,0,0,244,246,1,0,0, - 0,245,237,1,0,0,0,245,246,1,0,0,0,246,247,1,0,0,0,247,248,3,36,18,0,248, - 17,1,0,0,0,249,250,5,93,0,0,250,254,3,36,18,0,251,253,3,16,8,0,252,251, - 1,0,0,0,253,256,1,0,0,0,254,252,1,0,0,0,254,255,1,0,0,0,255,259,1,0,0, - 0,256,254,1,0,0,0,257,258,5,29,0,0,258,260,3,36,18,0,259,257,1,0,0,0, - 259,260,1,0,0,0,260,19,1,0,0,0,261,262,5,40,0,0,262,263,5,130,0,0,263, - 264,3,4,2,0,264,265,5,149,0,0,265,268,3,10,5,0,266,267,5,25,0,0,267,269, - 3,10,5,0,268,266,1,0,0,0,268,269,1,0,0,0,269,21,1,0,0,0,270,271,5,100, - 0,0,271,272,5,130,0,0,272,273,3,4,2,0,273,274,5,149,0,0,274,276,3,10, - 5,0,275,277,5,150,0,0,276,275,1,0,0,0,276,277,1,0,0,0,277,23,1,0,0,0, - 278,279,5,33,0,0,279,283,5,130,0,0,280,284,3,6,3,0,281,284,3,30,15,0, - 282,284,3,4,2,0,283,280,1,0,0,0,283,281,1,0,0,0,283,282,1,0,0,0,283,284, - 1,0,0,0,284,285,1,0,0,0,285,287,5,150,0,0,286,288,3,4,2,0,287,286,1,0, - 0,0,287,288,1,0,0,0,288,289,1,0,0,0,289,293,5,150,0,0,290,294,3,6,3,0, - 291,294,3,30,15,0,292,294,3,4,2,0,293,290,1,0,0,0,293,291,1,0,0,0,293, - 292,1,0,0,0,293,294,1,0,0,0,294,295,1,0,0,0,295,296,5,149,0,0,296,298, - 3,10,5,0,297,299,5,150,0,0,298,297,1,0,0,0,298,299,1,0,0,0,299,25,1,0, - 0,0,300,301,5,33,0,0,301,302,5,130,0,0,302,303,5,52,0,0,303,306,3,160, - 80,0,304,305,5,116,0,0,305,307,3,160,80,0,306,304,1,0,0,0,306,307,1,0, - 0,0,307,308,1,0,0,0,308,309,5,42,0,0,309,310,3,4,2,0,310,311,5,149,0, - 0,311,313,3,10,5,0,312,314,5,150,0,0,313,312,1,0,0,0,313,314,1,0,0,0, - 314,27,1,0,0,0,315,316,5,31,0,0,316,317,3,160,80,0,317,319,5,130,0,0, - 318,320,3,8,4,0,319,318,1,0,0,0,319,320,1,0,0,0,320,321,1,0,0,0,321,322, - 5,149,0,0,322,323,3,36,18,0,323,29,1,0,0,0,324,325,3,4,2,0,325,326,5, - 115,0,0,326,327,5,122,0,0,327,328,3,4,2,0,328,31,1,0,0,0,329,331,3,4, - 2,0,330,332,5,150,0,0,331,330,1,0,0,0,331,332,1,0,0,0,332,33,1,0,0,0, - 333,334,5,150,0,0,334,35,1,0,0,0,335,339,5,128,0,0,336,338,3,2,1,0,337, - 336,1,0,0,0,338,341,1,0,0,0,339,337,1,0,0,0,339,340,1,0,0,0,340,342,1, - 0,0,0,341,339,1,0,0,0,342,343,5,147,0,0,343,37,1,0,0,0,344,345,3,4,2, - 0,345,346,5,115,0,0,346,347,3,4,2,0,347,39,1,0,0,0,348,353,3,38,19,0, - 349,350,5,116,0,0,350,352,3,38,19,0,351,349,1,0,0,0,352,355,1,0,0,0,353, - 351,1,0,0,0,353,354,1,0,0,0,354,357,1,0,0,0,355,353,1,0,0,0,356,358,5, - 116,0,0,357,356,1,0,0,0,357,358,1,0,0,0,358,41,1,0,0,0,359,363,3,44,22, - 0,360,363,3,48,24,0,361,363,3,124,62,0,362,359,1,0,0,0,362,360,1,0,0, - 0,362,361,1,0,0,0,363,364,1,0,0,0,364,365,5,0,0,1,365,43,1,0,0,0,366, - 372,3,46,23,0,367,368,5,95,0,0,368,369,5,1,0,0,369,371,3,46,23,0,370, - 367,1,0,0,0,371,374,1,0,0,0,372,370,1,0,0,0,372,373,1,0,0,0,373,45,1, - 0,0,0,374,372,1,0,0,0,375,382,3,48,24,0,376,377,5,130,0,0,377,378,3,44, - 22,0,378,379,5,149,0,0,379,382,1,0,0,0,380,382,3,164,82,0,381,375,1,0, - 0,0,381,376,1,0,0,0,381,380,1,0,0,0,382,47,1,0,0,0,383,385,3,50,25,0, - 384,383,1,0,0,0,384,385,1,0,0,0,385,386,1,0,0,0,386,388,5,79,0,0,387, - 389,5,24,0,0,388,387,1,0,0,0,388,389,1,0,0,0,389,391,1,0,0,0,390,392, - 3,52,26,0,391,390,1,0,0,0,391,392,1,0,0,0,392,393,1,0,0,0,393,395,3,114, - 57,0,394,396,3,54,27,0,395,394,1,0,0,0,395,396,1,0,0,0,396,398,1,0,0, - 0,397,399,3,56,28,0,398,397,1,0,0,0,398,399,1,0,0,0,399,401,1,0,0,0,400, - 402,3,60,30,0,401,400,1,0,0,0,401,402,1,0,0,0,402,404,1,0,0,0,403,405, - 3,62,31,0,404,403,1,0,0,0,404,405,1,0,0,0,405,407,1,0,0,0,406,408,3,64, - 32,0,407,406,1,0,0,0,407,408,1,0,0,0,408,411,1,0,0,0,409,410,5,102,0, - 0,410,412,7,0,0,0,411,409,1,0,0,0,411,412,1,0,0,0,412,415,1,0,0,0,413, - 414,5,102,0,0,414,416,5,89,0,0,415,413,1,0,0,0,415,416,1,0,0,0,416,418, - 1,0,0,0,417,419,3,66,33,0,418,417,1,0,0,0,418,419,1,0,0,0,419,421,1,0, - 0,0,420,422,3,58,29,0,421,420,1,0,0,0,421,422,1,0,0,0,422,424,1,0,0,0, - 423,425,3,68,34,0,424,423,1,0,0,0,424,425,1,0,0,0,425,428,1,0,0,0,426, - 429,3,72,36,0,427,429,3,74,37,0,428,426,1,0,0,0,428,427,1,0,0,0,428,429, - 1,0,0,0,429,431,1,0,0,0,430,432,3,76,38,0,431,430,1,0,0,0,431,432,1,0, - 0,0,432,49,1,0,0,0,433,434,5,102,0,0,434,435,3,128,64,0,435,51,1,0,0, - 0,436,437,5,88,0,0,437,440,5,108,0,0,438,439,5,102,0,0,439,441,5,85,0, - 0,440,438,1,0,0,0,440,441,1,0,0,0,441,53,1,0,0,0,442,443,5,34,0,0,443, - 444,3,78,39,0,444,55,1,0,0,0,445,447,7,1,0,0,446,445,1,0,0,0,446,447, - 1,0,0,0,447,448,1,0,0,0,448,449,5,5,0,0,449,450,5,47,0,0,450,451,3,114, - 57,0,451,57,1,0,0,0,452,453,5,101,0,0,453,454,3,160,80,0,454,455,5,6, - 0,0,455,456,5,130,0,0,456,457,3,98,49,0,457,467,5,149,0,0,458,459,5,116, - 0,0,459,460,3,160,80,0,460,461,5,6,0,0,461,462,5,130,0,0,462,463,3,98, - 49,0,463,464,5,149,0,0,464,466,1,0,0,0,465,458,1,0,0,0,466,469,1,0,0, - 0,467,465,1,0,0,0,467,468,1,0,0,0,468,59,1,0,0,0,469,467,1,0,0,0,470, - 471,5,69,0,0,471,472,3,116,58,0,472,61,1,0,0,0,473,474,5,99,0,0,474,475, - 3,116,58,0,475,63,1,0,0,0,476,477,5,36,0,0,477,484,5,11,0,0,478,479,7, - 0,0,0,479,480,5,130,0,0,480,481,3,114,57,0,481,482,5,149,0,0,482,485, - 1,0,0,0,483,485,3,114,57,0,484,478,1,0,0,0,484,483,1,0,0,0,485,65,1,0, - 0,0,486,487,5,37,0,0,487,488,3,116,58,0,488,67,1,0,0,0,489,490,5,64,0, - 0,490,491,5,11,0,0,491,492,3,88,44,0,492,69,1,0,0,0,493,494,5,64,0,0, - 494,495,5,11,0,0,495,496,3,114,57,0,496,71,1,0,0,0,497,498,5,54,0,0,498, - 501,3,116,58,0,499,500,5,116,0,0,500,502,3,116,58,0,501,499,1,0,0,0,501, - 502,1,0,0,0,502,507,1,0,0,0,503,504,5,102,0,0,504,508,5,85,0,0,505,506, - 5,11,0,0,506,508,3,114,57,0,507,503,1,0,0,0,507,505,1,0,0,0,507,508,1, - 0,0,0,508,527,1,0,0,0,509,510,5,54,0,0,510,513,3,116,58,0,511,512,5,102, - 0,0,512,514,5,85,0,0,513,511,1,0,0,0,513,514,1,0,0,0,514,515,1,0,0,0, - 515,516,5,61,0,0,516,517,3,116,58,0,517,527,1,0,0,0,518,519,5,54,0,0, - 519,520,3,116,58,0,520,521,5,61,0,0,521,524,3,116,58,0,522,523,5,11,0, - 0,523,525,3,114,57,0,524,522,1,0,0,0,524,525,1,0,0,0,525,527,1,0,0,0, - 526,497,1,0,0,0,526,509,1,0,0,0,526,518,1,0,0,0,527,73,1,0,0,0,528,529, - 5,61,0,0,529,530,3,116,58,0,530,75,1,0,0,0,531,532,5,81,0,0,532,533,3, - 94,47,0,533,77,1,0,0,0,534,535,6,39,-1,0,535,537,3,136,68,0,536,538,5, - 28,0,0,537,536,1,0,0,0,537,538,1,0,0,0,538,540,1,0,0,0,539,541,3,86,43, - 0,540,539,1,0,0,0,540,541,1,0,0,0,541,547,1,0,0,0,542,543,5,130,0,0,543, - 544,3,78,39,0,544,545,5,149,0,0,545,547,1,0,0,0,546,534,1,0,0,0,546,542, - 1,0,0,0,547,562,1,0,0,0,548,549,10,3,0,0,549,550,3,82,41,0,550,551,3, - 78,39,4,551,561,1,0,0,0,552,554,10,4,0,0,553,555,3,80,40,0,554,553,1, - 0,0,0,554,555,1,0,0,0,555,556,1,0,0,0,556,557,5,47,0,0,557,558,3,78,39, - 0,558,559,3,84,42,0,559,561,1,0,0,0,560,548,1,0,0,0,560,552,1,0,0,0,561, - 564,1,0,0,0,562,560,1,0,0,0,562,563,1,0,0,0,563,79,1,0,0,0,564,562,1, - 0,0,0,565,567,7,2,0,0,566,565,1,0,0,0,566,567,1,0,0,0,567,568,1,0,0,0, - 568,575,5,44,0,0,569,571,5,44,0,0,570,572,7,2,0,0,571,570,1,0,0,0,571, - 572,1,0,0,0,572,575,1,0,0,0,573,575,7,2,0,0,574,566,1,0,0,0,574,569,1, - 0,0,0,574,573,1,0,0,0,575,609,1,0,0,0,576,578,7,3,0,0,577,576,1,0,0,0, - 577,578,1,0,0,0,578,579,1,0,0,0,579,581,7,4,0,0,580,582,5,65,0,0,581, - 580,1,0,0,0,581,582,1,0,0,0,582,591,1,0,0,0,583,585,7,4,0,0,584,586,5, - 65,0,0,585,584,1,0,0,0,585,586,1,0,0,0,586,588,1,0,0,0,587,589,7,3,0, - 0,588,587,1,0,0,0,588,589,1,0,0,0,589,591,1,0,0,0,590,577,1,0,0,0,590, - 583,1,0,0,0,591,609,1,0,0,0,592,594,7,5,0,0,593,592,1,0,0,0,593,594,1, - 0,0,0,594,595,1,0,0,0,595,597,5,35,0,0,596,598,5,65,0,0,597,596,1,0,0, - 0,597,598,1,0,0,0,598,607,1,0,0,0,599,601,5,35,0,0,600,602,5,65,0,0,601, - 600,1,0,0,0,601,602,1,0,0,0,602,604,1,0,0,0,603,605,7,5,0,0,604,603,1, - 0,0,0,604,605,1,0,0,0,605,607,1,0,0,0,606,593,1,0,0,0,606,599,1,0,0,0, - 607,609,1,0,0,0,608,574,1,0,0,0,608,590,1,0,0,0,608,606,1,0,0,0,609,81, - 1,0,0,0,610,611,5,17,0,0,611,614,5,47,0,0,612,614,5,116,0,0,613,610,1, - 0,0,0,613,612,1,0,0,0,614,83,1,0,0,0,615,616,5,62,0,0,616,625,3,114,57, - 0,617,618,5,96,0,0,618,619,5,130,0,0,619,620,3,114,57,0,620,621,5,149, - 0,0,621,625,1,0,0,0,622,623,5,96,0,0,623,625,3,114,57,0,624,615,1,0,0, - 0,624,617,1,0,0,0,624,622,1,0,0,0,625,85,1,0,0,0,626,627,5,77,0,0,627, - 630,3,92,46,0,628,629,5,61,0,0,629,631,3,92,46,0,630,628,1,0,0,0,630, - 631,1,0,0,0,631,87,1,0,0,0,632,637,3,90,45,0,633,634,5,116,0,0,634,636, - 3,90,45,0,635,633,1,0,0,0,636,639,1,0,0,0,637,635,1,0,0,0,637,638,1,0, - 0,0,638,89,1,0,0,0,639,637,1,0,0,0,640,642,3,116,58,0,641,643,7,6,0,0, - 642,641,1,0,0,0,642,643,1,0,0,0,643,646,1,0,0,0,644,645,5,60,0,0,645, - 647,7,7,0,0,646,644,1,0,0,0,646,647,1,0,0,0,647,650,1,0,0,0,648,649,5, - 16,0,0,649,651,5,110,0,0,650,648,1,0,0,0,650,651,1,0,0,0,651,91,1,0,0, - 0,652,659,3,164,82,0,653,656,3,148,74,0,654,655,5,151,0,0,655,657,3,148, - 74,0,656,654,1,0,0,0,656,657,1,0,0,0,657,659,1,0,0,0,658,652,1,0,0,0, - 658,653,1,0,0,0,659,93,1,0,0,0,660,665,3,96,48,0,661,662,5,116,0,0,662, - 664,3,96,48,0,663,661,1,0,0,0,664,667,1,0,0,0,665,663,1,0,0,0,665,666, - 1,0,0,0,666,95,1,0,0,0,667,665,1,0,0,0,668,669,3,160,80,0,669,670,5,122, - 0,0,670,671,3,150,75,0,671,97,1,0,0,0,672,674,3,100,50,0,673,672,1,0, - 0,0,673,674,1,0,0,0,674,676,1,0,0,0,675,677,3,102,51,0,676,675,1,0,0, - 0,676,677,1,0,0,0,677,679,1,0,0,0,678,680,3,104,52,0,679,678,1,0,0,0, - 679,680,1,0,0,0,680,99,1,0,0,0,681,682,5,67,0,0,682,683,5,11,0,0,683, - 684,3,114,57,0,684,101,1,0,0,0,685,686,5,64,0,0,686,687,5,11,0,0,687, - 688,3,88,44,0,688,103,1,0,0,0,689,690,7,8,0,0,690,691,3,106,53,0,691, - 105,1,0,0,0,692,699,3,108,54,0,693,694,5,9,0,0,694,695,3,108,54,0,695, - 696,5,2,0,0,696,697,3,108,54,0,697,699,1,0,0,0,698,692,1,0,0,0,698,693, - 1,0,0,0,699,107,1,0,0,0,700,701,5,19,0,0,701,713,5,75,0,0,702,703,5,94, - 0,0,703,713,5,68,0,0,704,705,5,94,0,0,705,713,5,32,0,0,706,707,3,148, - 74,0,707,708,5,68,0,0,708,713,1,0,0,0,709,710,3,148,74,0,710,711,5,32, - 0,0,711,713,1,0,0,0,712,700,1,0,0,0,712,702,1,0,0,0,712,704,1,0,0,0,712, - 706,1,0,0,0,712,709,1,0,0,0,713,109,1,0,0,0,714,715,3,116,58,0,715,716, - 5,0,0,1,716,111,1,0,0,0,717,774,3,160,80,0,718,719,3,160,80,0,719,720, - 5,130,0,0,720,721,3,160,80,0,721,728,3,112,56,0,722,723,5,116,0,0,723, - 724,3,160,80,0,724,725,3,112,56,0,725,727,1,0,0,0,726,722,1,0,0,0,727, - 730,1,0,0,0,728,726,1,0,0,0,728,729,1,0,0,0,729,732,1,0,0,0,730,728,1, - 0,0,0,731,733,5,116,0,0,732,731,1,0,0,0,732,733,1,0,0,0,733,734,1,0,0, - 0,734,735,5,149,0,0,735,774,1,0,0,0,736,737,3,160,80,0,737,738,5,130, - 0,0,738,743,3,162,81,0,739,740,5,116,0,0,740,742,3,162,81,0,741,739,1, - 0,0,0,742,745,1,0,0,0,743,741,1,0,0,0,743,744,1,0,0,0,744,747,1,0,0,0, - 745,743,1,0,0,0,746,748,5,116,0,0,747,746,1,0,0,0,747,748,1,0,0,0,748, - 749,1,0,0,0,749,750,5,149,0,0,750,774,1,0,0,0,751,752,3,160,80,0,752, - 753,5,130,0,0,753,758,3,112,56,0,754,755,5,116,0,0,755,757,3,112,56,0, - 756,754,1,0,0,0,757,760,1,0,0,0,758,756,1,0,0,0,758,759,1,0,0,0,759,762, - 1,0,0,0,760,758,1,0,0,0,761,763,5,116,0,0,762,761,1,0,0,0,762,763,1,0, - 0,0,763,764,1,0,0,0,764,765,5,149,0,0,765,774,1,0,0,0,766,767,3,160,80, - 0,767,769,5,130,0,0,768,770,3,114,57,0,769,768,1,0,0,0,769,770,1,0,0, - 0,770,771,1,0,0,0,771,772,5,149,0,0,772,774,1,0,0,0,773,717,1,0,0,0,773, - 718,1,0,0,0,773,736,1,0,0,0,773,751,1,0,0,0,773,766,1,0,0,0,774,113,1, - 0,0,0,775,780,3,116,58,0,776,777,5,116,0,0,777,779,3,116,58,0,778,776, - 1,0,0,0,779,782,1,0,0,0,780,778,1,0,0,0,780,781,1,0,0,0,781,784,1,0,0, - 0,782,780,1,0,0,0,783,785,5,116,0,0,784,783,1,0,0,0,784,785,1,0,0,0,785, - 115,1,0,0,0,786,787,6,58,-1,0,787,789,5,12,0,0,788,790,3,116,58,0,789, - 788,1,0,0,0,789,790,1,0,0,0,790,796,1,0,0,0,791,792,5,98,0,0,792,793, - 3,116,58,0,793,794,5,83,0,0,794,795,3,116,58,0,795,797,1,0,0,0,796,791, - 1,0,0,0,797,798,1,0,0,0,798,796,1,0,0,0,798,799,1,0,0,0,799,802,1,0,0, - 0,800,801,5,25,0,0,801,803,3,116,58,0,802,800,1,0,0,0,802,803,1,0,0,0, - 803,804,1,0,0,0,804,805,5,26,0,0,805,936,1,0,0,0,806,807,5,13,0,0,807, - 808,5,130,0,0,808,809,3,116,58,0,809,810,5,6,0,0,810,811,3,112,56,0,811, - 812,5,149,0,0,812,936,1,0,0,0,813,814,5,20,0,0,814,936,5,110,0,0,815, - 816,5,45,0,0,816,817,3,116,58,0,817,818,3,152,76,0,818,936,1,0,0,0,819, - 820,5,82,0,0,820,821,5,130,0,0,821,822,3,116,58,0,822,823,5,34,0,0,823, - 826,3,116,58,0,824,825,5,33,0,0,825,827,3,116,58,0,826,824,1,0,0,0,826, - 827,1,0,0,0,827,828,1,0,0,0,828,829,5,149,0,0,829,936,1,0,0,0,830,831, - 5,86,0,0,831,936,5,110,0,0,832,833,5,91,0,0,833,834,5,130,0,0,834,835, - 7,9,0,0,835,836,3,166,83,0,836,837,5,34,0,0,837,838,3,116,58,0,838,839, - 5,149,0,0,839,936,1,0,0,0,840,841,3,160,80,0,841,843,5,130,0,0,842,844, - 3,114,57,0,843,842,1,0,0,0,843,844,1,0,0,0,844,845,1,0,0,0,845,846,5, - 149,0,0,846,855,1,0,0,0,847,849,5,130,0,0,848,850,5,24,0,0,849,848,1, - 0,0,0,849,850,1,0,0,0,850,852,1,0,0,0,851,853,3,118,59,0,852,851,1,0, - 0,0,852,853,1,0,0,0,853,854,1,0,0,0,854,856,5,149,0,0,855,847,1,0,0,0, - 855,856,1,0,0,0,856,857,1,0,0,0,857,858,5,66,0,0,858,859,5,130,0,0,859, - 860,3,98,49,0,860,861,5,149,0,0,861,936,1,0,0,0,862,863,3,160,80,0,863, - 865,5,130,0,0,864,866,3,114,57,0,865,864,1,0,0,0,865,866,1,0,0,0,866, - 867,1,0,0,0,867,868,5,149,0,0,868,877,1,0,0,0,869,871,5,130,0,0,870,872, - 5,24,0,0,871,870,1,0,0,0,871,872,1,0,0,0,872,874,1,0,0,0,873,875,3,118, - 59,0,874,873,1,0,0,0,874,875,1,0,0,0,875,876,1,0,0,0,876,878,5,149,0, - 0,877,869,1,0,0,0,877,878,1,0,0,0,878,879,1,0,0,0,879,880,5,66,0,0,880, - 881,3,160,80,0,881,936,1,0,0,0,882,888,3,160,80,0,883,885,5,130,0,0,884, - 886,3,114,57,0,885,884,1,0,0,0,885,886,1,0,0,0,886,887,1,0,0,0,887,889, - 5,149,0,0,888,883,1,0,0,0,888,889,1,0,0,0,889,890,1,0,0,0,890,892,5,130, - 0,0,891,893,5,24,0,0,892,891,1,0,0,0,892,893,1,0,0,0,893,895,1,0,0,0, - 894,896,3,118,59,0,895,894,1,0,0,0,895,896,1,0,0,0,896,897,1,0,0,0,897, - 898,5,149,0,0,898,936,1,0,0,0,899,936,3,124,62,0,900,936,3,168,84,0,901, - 936,3,150,75,0,902,903,5,118,0,0,903,936,3,116,58,19,904,905,5,58,0,0, - 905,936,3,116,58,13,906,907,3,140,70,0,907,908,5,120,0,0,908,910,1,0, - 0,0,909,906,1,0,0,0,909,910,1,0,0,0,910,911,1,0,0,0,911,936,5,112,0,0, - 912,913,5,130,0,0,913,914,3,44,22,0,914,915,5,149,0,0,915,936,1,0,0,0, - 916,917,5,130,0,0,917,918,3,116,58,0,918,919,5,149,0,0,919,936,1,0,0, - 0,920,921,5,130,0,0,921,922,3,114,57,0,922,923,5,149,0,0,923,936,1,0, - 0,0,924,926,5,129,0,0,925,927,3,114,57,0,926,925,1,0,0,0,926,927,1,0, - 0,0,927,928,1,0,0,0,928,936,5,148,0,0,929,931,5,128,0,0,930,932,3,40, - 20,0,931,930,1,0,0,0,931,932,1,0,0,0,932,933,1,0,0,0,933,936,5,147,0, - 0,934,936,3,132,66,0,935,786,1,0,0,0,935,806,1,0,0,0,935,813,1,0,0,0, - 935,815,1,0,0,0,935,819,1,0,0,0,935,830,1,0,0,0,935,832,1,0,0,0,935,840, - 1,0,0,0,935,862,1,0,0,0,935,882,1,0,0,0,935,899,1,0,0,0,935,900,1,0,0, - 0,935,901,1,0,0,0,935,902,1,0,0,0,935,904,1,0,0,0,935,909,1,0,0,0,935, - 912,1,0,0,0,935,916,1,0,0,0,935,920,1,0,0,0,935,924,1,0,0,0,935,929,1, - 0,0,0,935,934,1,0,0,0,936,1041,1,0,0,0,937,941,10,18,0,0,938,942,5,112, - 0,0,939,942,5,151,0,0,940,942,5,138,0,0,941,938,1,0,0,0,941,939,1,0,0, - 0,941,940,1,0,0,0,942,943,1,0,0,0,943,1040,3,116,58,19,944,948,10,17, - 0,0,945,949,5,139,0,0,946,949,5,118,0,0,947,949,5,117,0,0,948,945,1,0, - 0,0,948,946,1,0,0,0,948,947,1,0,0,0,949,950,1,0,0,0,950,1040,3,116,58, - 18,951,976,10,16,0,0,952,977,5,121,0,0,953,977,5,122,0,0,954,977,5,133, - 0,0,955,977,5,131,0,0,956,977,5,132,0,0,957,977,5,123,0,0,958,977,5,124, - 0,0,959,961,5,58,0,0,960,959,1,0,0,0,960,961,1,0,0,0,961,962,1,0,0,0, - 962,964,5,42,0,0,963,965,5,15,0,0,964,963,1,0,0,0,964,965,1,0,0,0,965, - 977,1,0,0,0,966,968,5,58,0,0,967,966,1,0,0,0,967,968,1,0,0,0,968,969, - 1,0,0,0,969,977,7,10,0,0,970,977,5,145,0,0,971,977,5,146,0,0,972,977, - 5,135,0,0,973,977,5,126,0,0,974,977,5,127,0,0,975,977,5,134,0,0,976,952, - 1,0,0,0,976,953,1,0,0,0,976,954,1,0,0,0,976,955,1,0,0,0,976,956,1,0,0, - 0,976,957,1,0,0,0,976,958,1,0,0,0,976,960,1,0,0,0,976,967,1,0,0,0,976, - 970,1,0,0,0,976,971,1,0,0,0,976,972,1,0,0,0,976,973,1,0,0,0,976,974,1, - 0,0,0,976,975,1,0,0,0,977,978,1,0,0,0,978,1040,3,116,58,17,979,980,10, - 14,0,0,980,981,5,137,0,0,981,1040,3,116,58,15,982,983,10,12,0,0,983,984, - 5,2,0,0,984,1040,3,116,58,13,985,986,10,11,0,0,986,987,5,63,0,0,987,1040, - 3,116,58,12,988,990,10,10,0,0,989,991,5,58,0,0,990,989,1,0,0,0,990,991, - 1,0,0,0,991,992,1,0,0,0,992,993,5,9,0,0,993,994,3,116,58,0,994,995,5, - 2,0,0,995,996,3,116,58,11,996,1040,1,0,0,0,997,998,10,9,0,0,998,999,5, - 140,0,0,999,1000,3,116,58,0,1000,1001,5,115,0,0,1001,1002,3,116,58,9, - 1002,1040,1,0,0,0,1003,1004,10,25,0,0,1004,1005,5,129,0,0,1005,1006,3, - 116,58,0,1006,1007,5,148,0,0,1007,1040,1,0,0,0,1008,1009,10,24,0,0,1009, - 1010,5,120,0,0,1010,1040,5,108,0,0,1011,1012,10,23,0,0,1012,1013,5,120, - 0,0,1013,1040,3,160,80,0,1014,1015,10,22,0,0,1015,1016,5,136,0,0,1016, - 1017,5,129,0,0,1017,1018,3,116,58,0,1018,1019,5,148,0,0,1019,1040,1,0, - 0,0,1020,1021,10,21,0,0,1021,1022,5,136,0,0,1022,1040,5,108,0,0,1023, - 1024,10,20,0,0,1024,1025,5,136,0,0,1025,1040,3,160,80,0,1026,1027,10, - 15,0,0,1027,1029,5,46,0,0,1028,1030,5,58,0,0,1029,1028,1,0,0,0,1029,1030, - 1,0,0,0,1030,1031,1,0,0,0,1031,1040,5,59,0,0,1032,1037,10,8,0,0,1033, - 1034,5,6,0,0,1034,1038,3,160,80,0,1035,1036,5,6,0,0,1036,1038,5,110,0, - 0,1037,1033,1,0,0,0,1037,1035,1,0,0,0,1038,1040,1,0,0,0,1039,937,1,0, - 0,0,1039,944,1,0,0,0,1039,951,1,0,0,0,1039,979,1,0,0,0,1039,982,1,0,0, - 0,1039,985,1,0,0,0,1039,988,1,0,0,0,1039,997,1,0,0,0,1039,1003,1,0,0, - 0,1039,1008,1,0,0,0,1039,1011,1,0,0,0,1039,1014,1,0,0,0,1039,1020,1,0, - 0,0,1039,1023,1,0,0,0,1039,1026,1,0,0,0,1039,1032,1,0,0,0,1040,1043,1, - 0,0,0,1041,1039,1,0,0,0,1041,1042,1,0,0,0,1042,117,1,0,0,0,1043,1041, - 1,0,0,0,1044,1049,3,120,60,0,1045,1046,5,116,0,0,1046,1048,3,120,60,0, - 1047,1045,1,0,0,0,1048,1051,1,0,0,0,1049,1047,1,0,0,0,1049,1050,1,0,0, - 0,1050,1053,1,0,0,0,1051,1049,1,0,0,0,1052,1054,5,116,0,0,1053,1052,1, - 0,0,0,1053,1054,1,0,0,0,1054,119,1,0,0,0,1055,1058,3,122,61,0,1056,1058, - 3,116,58,0,1057,1055,1,0,0,0,1057,1056,1,0,0,0,1058,121,1,0,0,0,1059, - 1060,5,130,0,0,1060,1065,3,160,80,0,1061,1062,5,116,0,0,1062,1064,3,160, - 80,0,1063,1061,1,0,0,0,1064,1067,1,0,0,0,1065,1063,1,0,0,0,1065,1066, - 1,0,0,0,1066,1069,1,0,0,0,1067,1065,1,0,0,0,1068,1070,5,116,0,0,1069, - 1068,1,0,0,0,1069,1070,1,0,0,0,1070,1071,1,0,0,0,1071,1072,5,149,0,0, - 1072,1085,1,0,0,0,1073,1078,3,160,80,0,1074,1075,5,116,0,0,1075,1077, - 3,160,80,0,1076,1074,1,0,0,0,1077,1080,1,0,0,0,1078,1076,1,0,0,0,1078, - 1079,1,0,0,0,1079,1082,1,0,0,0,1080,1078,1,0,0,0,1081,1083,5,116,0,0, - 1082,1081,1,0,0,0,1082,1083,1,0,0,0,1083,1085,1,0,0,0,1084,1059,1,0,0, - 0,1084,1073,1,0,0,0,1085,1086,1,0,0,0,1086,1087,5,111,0,0,1087,1088,3, - 116,58,0,1088,123,1,0,0,0,1089,1090,5,132,0,0,1090,1094,3,160,80,0,1091, - 1093,3,126,63,0,1092,1091,1,0,0,0,1093,1096,1,0,0,0,1094,1092,1,0,0,0, - 1094,1095,1,0,0,0,1095,1097,1,0,0,0,1096,1094,1,0,0,0,1097,1098,5,151, - 0,0,1098,1099,5,124,0,0,1099,1122,1,0,0,0,1100,1101,5,132,0,0,1101,1105, - 3,160,80,0,1102,1104,3,126,63,0,1103,1102,1,0,0,0,1104,1107,1,0,0,0,1105, - 1103,1,0,0,0,1105,1106,1,0,0,0,1106,1108,1,0,0,0,1107,1105,1,0,0,0,1108, - 1114,5,124,0,0,1109,1115,3,124,62,0,1110,1111,5,128,0,0,1111,1112,3,116, - 58,0,1112,1113,5,147,0,0,1113,1115,1,0,0,0,1114,1109,1,0,0,0,1114,1110, - 1,0,0,0,1114,1115,1,0,0,0,1115,1116,1,0,0,0,1116,1117,5,132,0,0,1117, - 1118,5,151,0,0,1118,1119,3,160,80,0,1119,1120,5,124,0,0,1120,1122,1,0, - 0,0,1121,1089,1,0,0,0,1121,1100,1,0,0,0,1122,125,1,0,0,0,1123,1124,3, - 160,80,0,1124,1125,5,122,0,0,1125,1126,3,166,83,0,1126,1135,1,0,0,0,1127, - 1128,3,160,80,0,1128,1129,5,122,0,0,1129,1130,5,128,0,0,1130,1131,3,116, - 58,0,1131,1132,5,147,0,0,1132,1135,1,0,0,0,1133,1135,3,160,80,0,1134, - 1123,1,0,0,0,1134,1127,1,0,0,0,1134,1133,1,0,0,0,1135,127,1,0,0,0,1136, - 1141,3,130,65,0,1137,1138,5,116,0,0,1138,1140,3,130,65,0,1139,1137,1, - 0,0,0,1140,1143,1,0,0,0,1141,1139,1,0,0,0,1141,1142,1,0,0,0,1142,1145, - 1,0,0,0,1143,1141,1,0,0,0,1144,1146,5,116,0,0,1145,1144,1,0,0,0,1145, - 1146,1,0,0,0,1146,129,1,0,0,0,1147,1148,3,160,80,0,1148,1149,5,6,0,0, - 1149,1150,5,130,0,0,1150,1151,3,44,22,0,1151,1152,5,149,0,0,1152,1158, - 1,0,0,0,1153,1154,3,116,58,0,1154,1155,5,6,0,0,1155,1156,3,160,80,0,1156, - 1158,1,0,0,0,1157,1147,1,0,0,0,1157,1153,1,0,0,0,1158,131,1,0,0,0,1159, - 1167,3,164,82,0,1160,1161,3,140,70,0,1161,1162,5,120,0,0,1162,1164,1, - 0,0,0,1163,1160,1,0,0,0,1163,1164,1,0,0,0,1164,1165,1,0,0,0,1165,1167, - 3,134,67,0,1166,1159,1,0,0,0,1166,1163,1,0,0,0,1167,133,1,0,0,0,1168, - 1173,3,160,80,0,1169,1170,5,120,0,0,1170,1172,3,160,80,0,1171,1169,1, - 0,0,0,1172,1175,1,0,0,0,1173,1171,1,0,0,0,1173,1174,1,0,0,0,1174,135, - 1,0,0,0,1175,1173,1,0,0,0,1176,1177,6,68,-1,0,1177,1186,3,140,70,0,1178, - 1186,3,138,69,0,1179,1180,5,130,0,0,1180,1181,3,44,22,0,1181,1182,5,149, - 0,0,1182,1186,1,0,0,0,1183,1186,3,124,62,0,1184,1186,3,164,82,0,1185, - 1176,1,0,0,0,1185,1178,1,0,0,0,1185,1179,1,0,0,0,1185,1183,1,0,0,0,1185, - 1184,1,0,0,0,1186,1195,1,0,0,0,1187,1191,10,3,0,0,1188,1192,3,158,79, - 0,1189,1190,5,6,0,0,1190,1192,3,160,80,0,1191,1188,1,0,0,0,1191,1189, - 1,0,0,0,1192,1194,1,0,0,0,1193,1187,1,0,0,0,1194,1197,1,0,0,0,1195,1193, - 1,0,0,0,1195,1196,1,0,0,0,1196,137,1,0,0,0,1197,1195,1,0,0,0,1198,1199, - 3,160,80,0,1199,1201,5,130,0,0,1200,1202,3,142,71,0,1201,1200,1,0,0,0, - 1201,1202,1,0,0,0,1202,1203,1,0,0,0,1203,1204,5,149,0,0,1204,139,1,0, - 0,0,1205,1206,3,144,72,0,1206,1207,5,120,0,0,1207,1209,1,0,0,0,1208,1205, - 1,0,0,0,1208,1209,1,0,0,0,1209,1210,1,0,0,0,1210,1211,3,160,80,0,1211, - 141,1,0,0,0,1212,1217,3,116,58,0,1213,1214,5,116,0,0,1214,1216,3,116, - 58,0,1215,1213,1,0,0,0,1216,1219,1,0,0,0,1217,1215,1,0,0,0,1217,1218, - 1,0,0,0,1218,1221,1,0,0,0,1219,1217,1,0,0,0,1220,1222,5,116,0,0,1221, - 1220,1,0,0,0,1221,1222,1,0,0,0,1222,143,1,0,0,0,1223,1224,3,160,80,0, - 1224,145,1,0,0,0,1225,1234,5,106,0,0,1226,1227,5,120,0,0,1227,1234,7, - 11,0,0,1228,1229,5,108,0,0,1229,1231,5,120,0,0,1230,1232,7,11,0,0,1231, - 1230,1,0,0,0,1231,1232,1,0,0,0,1232,1234,1,0,0,0,1233,1225,1,0,0,0,1233, - 1226,1,0,0,0,1233,1228,1,0,0,0,1234,147,1,0,0,0,1235,1237,7,12,0,0,1236, - 1235,1,0,0,0,1236,1237,1,0,0,0,1237,1244,1,0,0,0,1238,1245,3,146,73,0, - 1239,1245,5,107,0,0,1240,1245,5,108,0,0,1241,1245,5,109,0,0,1242,1245, - 5,43,0,0,1243,1245,5,57,0,0,1244,1238,1,0,0,0,1244,1239,1,0,0,0,1244, - 1240,1,0,0,0,1244,1241,1,0,0,0,1244,1242,1,0,0,0,1244,1243,1,0,0,0,1245, - 149,1,0,0,0,1246,1250,3,148,74,0,1247,1250,5,110,0,0,1248,1250,5,59,0, - 0,1249,1246,1,0,0,0,1249,1247,1,0,0,0,1249,1248,1,0,0,0,1250,151,1,0, - 0,0,1251,1252,7,13,0,0,1252,153,1,0,0,0,1253,1254,7,14,0,0,1254,155,1, - 0,0,0,1255,1256,7,15,0,0,1256,157,1,0,0,0,1257,1260,5,105,0,0,1258,1260, - 3,156,78,0,1259,1257,1,0,0,0,1259,1258,1,0,0,0,1260,159,1,0,0,0,1261, - 1265,5,105,0,0,1262,1265,3,152,76,0,1263,1265,3,154,77,0,1264,1261,1, - 0,0,0,1264,1262,1,0,0,0,1264,1263,1,0,0,0,1265,161,1,0,0,0,1266,1267, - 3,166,83,0,1267,1268,5,122,0,0,1268,1269,3,148,74,0,1269,163,1,0,0,0, - 1270,1271,5,128,0,0,1271,1272,3,134,67,0,1272,1273,5,147,0,0,1273,165, - 1,0,0,0,1274,1277,5,110,0,0,1275,1277,3,168,84,0,1276,1274,1,0,0,0,1276, - 1275,1,0,0,0,1277,167,1,0,0,0,1278,1282,5,142,0,0,1279,1281,3,170,85, - 0,1280,1279,1,0,0,0,1281,1284,1,0,0,0,1282,1280,1,0,0,0,1282,1283,1,0, - 0,0,1283,1285,1,0,0,0,1284,1282,1,0,0,0,1285,1286,5,144,0,0,1286,169, - 1,0,0,0,1287,1288,5,157,0,0,1288,1289,3,116,58,0,1289,1290,5,147,0,0, - 1290,1293,1,0,0,0,1291,1293,5,156,0,0,1292,1287,1,0,0,0,1292,1291,1,0, - 0,0,1293,171,1,0,0,0,1294,1298,5,143,0,0,1295,1297,3,174,87,0,1296,1295, - 1,0,0,0,1297,1300,1,0,0,0,1298,1296,1,0,0,0,1298,1299,1,0,0,0,1299,1301, - 1,0,0,0,1300,1298,1,0,0,0,1301,1302,5,0,0,1,1302,173,1,0,0,0,1303,1304, - 5,159,0,0,1304,1305,3,116,58,0,1305,1306,5,147,0,0,1306,1309,1,0,0,0, - 1307,1309,5,158,0,0,1308,1303,1,0,0,0,1308,1307,1,0,0,0,1309,175,1,0, - 0,0,168,179,186,195,202,206,220,224,227,231,234,241,245,254,259,268,276, - 283,287,293,298,306,313,319,331,339,353,357,362,372,381,384,388,391,395, - 398,401,404,407,411,415,418,421,424,428,431,440,446,467,484,501,507,513, - 524,526,537,540,546,554,560,562,566,571,574,577,581,585,588,590,593,597, - 601,604,606,608,613,624,630,637,642,646,650,656,658,665,673,676,679,698, - 712,728,732,743,747,758,762,769,773,780,784,789,798,802,826,843,849,852, - 855,865,871,874,877,885,888,892,895,909,926,931,935,941,948,960,964,967, - 976,990,1029,1037,1039,1041,1049,1053,1057,1065,1069,1078,1082,1084,1094, - 1105,1114,1121,1134,1141,1145,1157,1163,1166,1173,1185,1191,1195,1201, - 1208,1217,1221,1231,1233,1236,1244,1249,1259,1264,1276,1282,1292,1298, - 1308 + 1,58,3,58,1033,8,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,1041,8,58,5,58, + 1043,8,58,10,58,12,58,1046,9,58,1,59,1,59,1,59,1,59,5,59,1052,8,59,10, + 59,12,59,1055,9,59,1,59,3,59,1058,8,59,1,59,1,59,1,59,1,59,1,59,5,59, + 1065,8,59,10,59,12,59,1068,9,59,1,59,3,59,1071,8,59,1,59,1,59,3,59,1075, + 8,59,1,59,1,59,1,59,3,59,1080,8,59,1,60,1,60,1,60,5,60,1085,8,60,10,60, + 12,60,1088,9,60,1,60,1,60,1,60,1,60,1,60,1,60,5,60,1096,8,60,10,60,12, + 60,1099,9,60,1,60,1,60,1,60,1,60,1,60,1,60,3,60,1107,8,60,1,60,1,60,1, + 60,1,60,1,60,3,60,1114,8,60,1,61,1,61,1,61,1,61,1,61,1,61,1,61,1,61,1, + 61,1,61,1,61,3,61,1127,8,61,1,62,1,62,1,62,5,62,1132,8,62,10,62,12,62, + 1135,9,62,1,62,3,62,1138,8,62,1,63,1,63,1,63,1,63,1,63,1,63,1,63,1,63, + 1,63,1,63,3,63,1150,8,63,1,64,1,64,1,64,1,64,3,64,1156,8,64,1,64,3,64, + 1159,8,64,1,65,1,65,1,65,5,65,1164,8,65,10,65,12,65,1167,9,65,1,66,1, + 66,1,66,1,66,1,66,1,66,1,66,1,66,1,66,3,66,1178,8,66,1,66,1,66,1,66,1, + 66,3,66,1184,8,66,5,66,1186,8,66,10,66,12,66,1189,9,66,1,67,1,67,1,67, + 3,67,1194,8,67,1,67,1,67,1,68,1,68,1,68,3,68,1201,8,68,1,68,1,68,1,69, + 1,69,1,69,5,69,1208,8,69,10,69,12,69,1211,9,69,1,69,3,69,1214,8,69,1, + 70,1,70,1,71,1,71,1,71,1,71,1,71,1,71,3,71,1224,8,71,3,71,1226,8,71,1, + 72,3,72,1229,8,72,1,72,1,72,1,72,1,72,1,72,1,72,3,72,1237,8,72,1,73,1, + 73,1,73,3,73,1242,8,73,1,74,1,74,1,75,1,75,1,76,1,76,1,77,1,77,3,77,1252, + 8,77,1,78,1,78,1,78,3,78,1257,8,78,1,79,1,79,1,79,1,79,1,80,1,80,1,80, + 1,80,1,81,1,81,3,81,1269,8,81,1,82,1,82,5,82,1273,8,82,10,82,12,82,1276, + 9,82,1,82,1,82,1,83,1,83,1,83,1,83,1,83,3,83,1285,8,83,1,84,1,84,5,84, + 1289,8,84,10,84,12,84,1292,9,84,1,84,1,84,1,85,1,85,1,85,1,85,1,85,3, + 85,1301,8,85,1,85,0,3,78,116,132,86,0,2,4,6,8,10,12,14,16,18,20,22,24, + 26,28,30,32,34,36,38,40,42,44,46,48,50,52,54,56,58,60,62,64,66,68,70, + 72,74,76,78,80,82,84,86,88,90,92,94,96,98,100,102,104,106,108,110,112, + 114,116,118,120,122,124,126,128,130,132,134,136,138,140,142,144,146,148, + 150,152,154,156,158,160,162,164,166,168,170,0,16,2,0,18,18,74,74,2,0, + 44,44,51,51,3,0,1,1,4,4,8,8,4,0,1,1,3,4,8,8,80,80,2,0,51,51,73,73,2,0, + 1,1,4,4,2,0,7,7,22,23,2,0,30,30,49,49,2,0,71,71,76,76,3,0,10,10,50,50, + 90,90,2,0,41,41,53,53,1,0,107,108,2,0,118,118,139,139,7,0,21,21,38,38, + 55,56,70,70,78,78,97,97,103,103,16,0,1,13,15,20,22,28,30,30,32,37,39, + 42,44,51,53,54,58,58,60,69,71,77,79,83,85,92,94,96,98,99,101,102,4,0, + 20,20,30,30,39,39,48,48,1475,0,175,1,0,0,0,2,182,1,0,0,0,4,184,1,0,0, + 0,6,186,1,0,0,0,8,193,1,0,0,0,10,216,1,0,0,0,12,218,1,0,0,0,14,225,1, + 0,0,0,16,232,1,0,0,0,18,245,1,0,0,0,20,257,1,0,0,0,22,266,1,0,0,0,24, + 274,1,0,0,0,26,296,1,0,0,0,28,311,1,0,0,0,30,320,1,0,0,0,32,325,1,0,0, + 0,34,329,1,0,0,0,36,331,1,0,0,0,38,340,1,0,0,0,40,344,1,0,0,0,42,358, + 1,0,0,0,44,362,1,0,0,0,46,377,1,0,0,0,48,380,1,0,0,0,50,429,1,0,0,0,52, + 432,1,0,0,0,54,438,1,0,0,0,56,442,1,0,0,0,58,448,1,0,0,0,60,466,1,0,0, + 0,62,469,1,0,0,0,64,472,1,0,0,0,66,482,1,0,0,0,68,485,1,0,0,0,70,489, + 1,0,0,0,72,522,1,0,0,0,74,524,1,0,0,0,76,527,1,0,0,0,78,542,1,0,0,0,80, + 604,1,0,0,0,82,609,1,0,0,0,84,620,1,0,0,0,86,622,1,0,0,0,88,628,1,0,0, + 0,90,636,1,0,0,0,92,654,1,0,0,0,94,656,1,0,0,0,96,664,1,0,0,0,98,669, + 1,0,0,0,100,677,1,0,0,0,102,681,1,0,0,0,104,685,1,0,0,0,106,694,1,0,0, + 0,108,708,1,0,0,0,110,710,1,0,0,0,112,769,1,0,0,0,114,771,1,0,0,0,116, + 932,1,0,0,0,118,1074,1,0,0,0,120,1113,1,0,0,0,122,1126,1,0,0,0,124,1128, + 1,0,0,0,126,1149,1,0,0,0,128,1158,1,0,0,0,130,1160,1,0,0,0,132,1177,1, + 0,0,0,134,1190,1,0,0,0,136,1200,1,0,0,0,138,1204,1,0,0,0,140,1215,1,0, + 0,0,142,1225,1,0,0,0,144,1228,1,0,0,0,146,1241,1,0,0,0,148,1243,1,0,0, + 0,150,1245,1,0,0,0,152,1247,1,0,0,0,154,1251,1,0,0,0,156,1256,1,0,0,0, + 158,1258,1,0,0,0,160,1262,1,0,0,0,162,1268,1,0,0,0,164,1270,1,0,0,0,166, + 1284,1,0,0,0,168,1286,1,0,0,0,170,1300,1,0,0,0,172,174,3,2,1,0,173,172, + 1,0,0,0,174,177,1,0,0,0,175,173,1,0,0,0,175,176,1,0,0,0,176,178,1,0,0, + 0,177,175,1,0,0,0,178,179,5,0,0,1,179,1,1,0,0,0,180,183,3,6,3,0,181,183, + 3,10,5,0,182,180,1,0,0,0,182,181,1,0,0,0,183,3,1,0,0,0,184,185,3,116, + 58,0,185,5,1,0,0,0,186,187,5,52,0,0,187,191,3,156,78,0,188,189,5,115, + 0,0,189,190,5,122,0,0,190,192,3,4,2,0,191,188,1,0,0,0,191,192,1,0,0,0, + 192,7,1,0,0,0,193,198,3,156,78,0,194,195,5,116,0,0,195,197,3,156,78,0, + 196,194,1,0,0,0,197,200,1,0,0,0,198,196,1,0,0,0,198,199,1,0,0,0,199,202, + 1,0,0,0,200,198,1,0,0,0,201,203,5,116,0,0,202,201,1,0,0,0,202,203,1,0, + 0,0,203,9,1,0,0,0,204,217,3,12,6,0,205,217,3,14,7,0,206,217,3,18,9,0, + 207,217,3,20,10,0,208,217,3,22,11,0,209,217,3,26,13,0,210,217,3,24,12, + 0,211,217,3,28,14,0,212,217,3,30,15,0,213,217,3,36,18,0,214,217,3,32, + 16,0,215,217,3,34,17,0,216,204,1,0,0,0,216,205,1,0,0,0,216,206,1,0,0, + 0,216,207,1,0,0,0,216,208,1,0,0,0,216,209,1,0,0,0,216,210,1,0,0,0,216, + 211,1,0,0,0,216,212,1,0,0,0,216,213,1,0,0,0,216,214,1,0,0,0,216,215,1, + 0,0,0,217,11,1,0,0,0,218,220,5,72,0,0,219,221,3,4,2,0,220,219,1,0,0,0, + 220,221,1,0,0,0,221,223,1,0,0,0,222,224,5,150,0,0,223,222,1,0,0,0,223, + 224,1,0,0,0,224,13,1,0,0,0,225,227,5,84,0,0,226,228,3,4,2,0,227,226,1, + 0,0,0,227,228,1,0,0,0,228,230,1,0,0,0,229,231,5,150,0,0,230,229,1,0,0, + 0,230,231,1,0,0,0,231,15,1,0,0,0,232,241,5,14,0,0,233,234,5,130,0,0,234, + 237,3,156,78,0,235,236,5,115,0,0,236,238,3,156,78,0,237,235,1,0,0,0,237, + 238,1,0,0,0,238,239,1,0,0,0,239,240,5,149,0,0,240,242,1,0,0,0,241,233, + 1,0,0,0,241,242,1,0,0,0,242,243,1,0,0,0,243,244,3,36,18,0,244,17,1,0, + 0,0,245,246,5,93,0,0,246,250,3,36,18,0,247,249,3,16,8,0,248,247,1,0,0, + 0,249,252,1,0,0,0,250,248,1,0,0,0,250,251,1,0,0,0,251,255,1,0,0,0,252, + 250,1,0,0,0,253,254,5,29,0,0,254,256,3,36,18,0,255,253,1,0,0,0,255,256, + 1,0,0,0,256,19,1,0,0,0,257,258,5,40,0,0,258,259,5,130,0,0,259,260,3,4, + 2,0,260,261,5,149,0,0,261,264,3,10,5,0,262,263,5,25,0,0,263,265,3,10, + 5,0,264,262,1,0,0,0,264,265,1,0,0,0,265,21,1,0,0,0,266,267,5,100,0,0, + 267,268,5,130,0,0,268,269,3,4,2,0,269,270,5,149,0,0,270,272,3,10,5,0, + 271,273,5,150,0,0,272,271,1,0,0,0,272,273,1,0,0,0,273,23,1,0,0,0,274, + 275,5,33,0,0,275,279,5,130,0,0,276,280,3,6,3,0,277,280,3,30,15,0,278, + 280,3,4,2,0,279,276,1,0,0,0,279,277,1,0,0,0,279,278,1,0,0,0,279,280,1, + 0,0,0,280,281,1,0,0,0,281,283,5,150,0,0,282,284,3,4,2,0,283,282,1,0,0, + 0,283,284,1,0,0,0,284,285,1,0,0,0,285,289,5,150,0,0,286,290,3,6,3,0,287, + 290,3,30,15,0,288,290,3,4,2,0,289,286,1,0,0,0,289,287,1,0,0,0,289,288, + 1,0,0,0,289,290,1,0,0,0,290,291,1,0,0,0,291,292,5,149,0,0,292,294,3,10, + 5,0,293,295,5,150,0,0,294,293,1,0,0,0,294,295,1,0,0,0,295,25,1,0,0,0, + 296,297,5,33,0,0,297,298,5,130,0,0,298,299,5,52,0,0,299,302,3,156,78, + 0,300,301,5,116,0,0,301,303,3,156,78,0,302,300,1,0,0,0,302,303,1,0,0, + 0,303,304,1,0,0,0,304,305,5,42,0,0,305,306,3,4,2,0,306,307,5,149,0,0, + 307,309,3,10,5,0,308,310,5,150,0,0,309,308,1,0,0,0,309,310,1,0,0,0,310, + 27,1,0,0,0,311,312,5,31,0,0,312,313,3,156,78,0,313,315,5,130,0,0,314, + 316,3,8,4,0,315,314,1,0,0,0,315,316,1,0,0,0,316,317,1,0,0,0,317,318,5, + 149,0,0,318,319,3,36,18,0,319,29,1,0,0,0,320,321,3,4,2,0,321,322,5,115, + 0,0,322,323,5,122,0,0,323,324,3,4,2,0,324,31,1,0,0,0,325,327,3,4,2,0, + 326,328,5,150,0,0,327,326,1,0,0,0,327,328,1,0,0,0,328,33,1,0,0,0,329, + 330,5,150,0,0,330,35,1,0,0,0,331,335,5,128,0,0,332,334,3,2,1,0,333,332, + 1,0,0,0,334,337,1,0,0,0,335,333,1,0,0,0,335,336,1,0,0,0,336,338,1,0,0, + 0,337,335,1,0,0,0,338,339,5,147,0,0,339,37,1,0,0,0,340,341,3,4,2,0,341, + 342,5,115,0,0,342,343,3,4,2,0,343,39,1,0,0,0,344,349,3,38,19,0,345,346, + 5,116,0,0,346,348,3,38,19,0,347,345,1,0,0,0,348,351,1,0,0,0,349,347,1, + 0,0,0,349,350,1,0,0,0,350,353,1,0,0,0,351,349,1,0,0,0,352,354,5,116,0, + 0,353,352,1,0,0,0,353,354,1,0,0,0,354,41,1,0,0,0,355,359,3,44,22,0,356, + 359,3,48,24,0,357,359,3,120,60,0,358,355,1,0,0,0,358,356,1,0,0,0,358, + 357,1,0,0,0,359,360,1,0,0,0,360,361,5,0,0,1,361,43,1,0,0,0,362,368,3, + 46,23,0,363,364,5,95,0,0,364,365,5,1,0,0,365,367,3,46,23,0,366,363,1, + 0,0,0,367,370,1,0,0,0,368,366,1,0,0,0,368,369,1,0,0,0,369,45,1,0,0,0, + 370,368,1,0,0,0,371,378,3,48,24,0,372,373,5,130,0,0,373,374,3,44,22,0, + 374,375,5,149,0,0,375,378,1,0,0,0,376,378,3,160,80,0,377,371,1,0,0,0, + 377,372,1,0,0,0,377,376,1,0,0,0,378,47,1,0,0,0,379,381,3,50,25,0,380, + 379,1,0,0,0,380,381,1,0,0,0,381,382,1,0,0,0,382,384,5,79,0,0,383,385, + 5,24,0,0,384,383,1,0,0,0,384,385,1,0,0,0,385,387,1,0,0,0,386,388,3,52, + 26,0,387,386,1,0,0,0,387,388,1,0,0,0,388,389,1,0,0,0,389,391,3,114,57, + 0,390,392,3,54,27,0,391,390,1,0,0,0,391,392,1,0,0,0,392,394,1,0,0,0,393, + 395,3,56,28,0,394,393,1,0,0,0,394,395,1,0,0,0,395,397,1,0,0,0,396,398, + 3,60,30,0,397,396,1,0,0,0,397,398,1,0,0,0,398,400,1,0,0,0,399,401,3,62, + 31,0,400,399,1,0,0,0,400,401,1,0,0,0,401,403,1,0,0,0,402,404,3,64,32, + 0,403,402,1,0,0,0,403,404,1,0,0,0,404,407,1,0,0,0,405,406,5,102,0,0,406, + 408,7,0,0,0,407,405,1,0,0,0,407,408,1,0,0,0,408,411,1,0,0,0,409,410,5, + 102,0,0,410,412,5,89,0,0,411,409,1,0,0,0,411,412,1,0,0,0,412,414,1,0, + 0,0,413,415,3,66,33,0,414,413,1,0,0,0,414,415,1,0,0,0,415,417,1,0,0,0, + 416,418,3,58,29,0,417,416,1,0,0,0,417,418,1,0,0,0,418,420,1,0,0,0,419, + 421,3,68,34,0,420,419,1,0,0,0,420,421,1,0,0,0,421,424,1,0,0,0,422,425, + 3,72,36,0,423,425,3,74,37,0,424,422,1,0,0,0,424,423,1,0,0,0,424,425,1, + 0,0,0,425,427,1,0,0,0,426,428,3,76,38,0,427,426,1,0,0,0,427,428,1,0,0, + 0,428,49,1,0,0,0,429,430,5,102,0,0,430,431,3,124,62,0,431,51,1,0,0,0, + 432,433,5,88,0,0,433,436,5,108,0,0,434,435,5,102,0,0,435,437,5,85,0,0, + 436,434,1,0,0,0,436,437,1,0,0,0,437,53,1,0,0,0,438,439,5,34,0,0,439,440, + 3,78,39,0,440,55,1,0,0,0,441,443,7,1,0,0,442,441,1,0,0,0,442,443,1,0, + 0,0,443,444,1,0,0,0,444,445,5,5,0,0,445,446,5,47,0,0,446,447,3,114,57, + 0,447,57,1,0,0,0,448,449,5,101,0,0,449,450,3,156,78,0,450,451,5,6,0,0, + 451,452,5,130,0,0,452,453,3,98,49,0,453,463,5,149,0,0,454,455,5,116,0, + 0,455,456,3,156,78,0,456,457,5,6,0,0,457,458,5,130,0,0,458,459,3,98,49, + 0,459,460,5,149,0,0,460,462,1,0,0,0,461,454,1,0,0,0,462,465,1,0,0,0,463, + 461,1,0,0,0,463,464,1,0,0,0,464,59,1,0,0,0,465,463,1,0,0,0,466,467,5, + 69,0,0,467,468,3,116,58,0,468,61,1,0,0,0,469,470,5,99,0,0,470,471,3,116, + 58,0,471,63,1,0,0,0,472,473,5,36,0,0,473,480,5,11,0,0,474,475,7,0,0,0, + 475,476,5,130,0,0,476,477,3,114,57,0,477,478,5,149,0,0,478,481,1,0,0, + 0,479,481,3,114,57,0,480,474,1,0,0,0,480,479,1,0,0,0,481,65,1,0,0,0,482, + 483,5,37,0,0,483,484,3,116,58,0,484,67,1,0,0,0,485,486,5,64,0,0,486,487, + 5,11,0,0,487,488,3,88,44,0,488,69,1,0,0,0,489,490,5,64,0,0,490,491,5, + 11,0,0,491,492,3,114,57,0,492,71,1,0,0,0,493,494,5,54,0,0,494,497,3,116, + 58,0,495,496,5,116,0,0,496,498,3,116,58,0,497,495,1,0,0,0,497,498,1,0, + 0,0,498,503,1,0,0,0,499,500,5,102,0,0,500,504,5,85,0,0,501,502,5,11,0, + 0,502,504,3,114,57,0,503,499,1,0,0,0,503,501,1,0,0,0,503,504,1,0,0,0, + 504,523,1,0,0,0,505,506,5,54,0,0,506,509,3,116,58,0,507,508,5,102,0,0, + 508,510,5,85,0,0,509,507,1,0,0,0,509,510,1,0,0,0,510,511,1,0,0,0,511, + 512,5,61,0,0,512,513,3,116,58,0,513,523,1,0,0,0,514,515,5,54,0,0,515, + 516,3,116,58,0,516,517,5,61,0,0,517,520,3,116,58,0,518,519,5,11,0,0,519, + 521,3,114,57,0,520,518,1,0,0,0,520,521,1,0,0,0,521,523,1,0,0,0,522,493, + 1,0,0,0,522,505,1,0,0,0,522,514,1,0,0,0,523,73,1,0,0,0,524,525,5,61,0, + 0,525,526,3,116,58,0,526,75,1,0,0,0,527,528,5,81,0,0,528,529,3,94,47, + 0,529,77,1,0,0,0,530,531,6,39,-1,0,531,533,3,132,66,0,532,534,5,28,0, + 0,533,532,1,0,0,0,533,534,1,0,0,0,534,536,1,0,0,0,535,537,3,86,43,0,536, + 535,1,0,0,0,536,537,1,0,0,0,537,543,1,0,0,0,538,539,5,130,0,0,539,540, + 3,78,39,0,540,541,5,149,0,0,541,543,1,0,0,0,542,530,1,0,0,0,542,538,1, + 0,0,0,543,558,1,0,0,0,544,545,10,3,0,0,545,546,3,82,41,0,546,547,3,78, + 39,4,547,557,1,0,0,0,548,550,10,4,0,0,549,551,3,80,40,0,550,549,1,0,0, + 0,550,551,1,0,0,0,551,552,1,0,0,0,552,553,5,47,0,0,553,554,3,78,39,0, + 554,555,3,84,42,0,555,557,1,0,0,0,556,544,1,0,0,0,556,548,1,0,0,0,557, + 560,1,0,0,0,558,556,1,0,0,0,558,559,1,0,0,0,559,79,1,0,0,0,560,558,1, + 0,0,0,561,563,7,2,0,0,562,561,1,0,0,0,562,563,1,0,0,0,563,564,1,0,0,0, + 564,571,5,44,0,0,565,567,5,44,0,0,566,568,7,2,0,0,567,566,1,0,0,0,567, + 568,1,0,0,0,568,571,1,0,0,0,569,571,7,2,0,0,570,562,1,0,0,0,570,565,1, + 0,0,0,570,569,1,0,0,0,571,605,1,0,0,0,572,574,7,3,0,0,573,572,1,0,0,0, + 573,574,1,0,0,0,574,575,1,0,0,0,575,577,7,4,0,0,576,578,5,65,0,0,577, + 576,1,0,0,0,577,578,1,0,0,0,578,587,1,0,0,0,579,581,7,4,0,0,580,582,5, + 65,0,0,581,580,1,0,0,0,581,582,1,0,0,0,582,584,1,0,0,0,583,585,7,3,0, + 0,584,583,1,0,0,0,584,585,1,0,0,0,585,587,1,0,0,0,586,573,1,0,0,0,586, + 579,1,0,0,0,587,605,1,0,0,0,588,590,7,5,0,0,589,588,1,0,0,0,589,590,1, + 0,0,0,590,591,1,0,0,0,591,593,5,35,0,0,592,594,5,65,0,0,593,592,1,0,0, + 0,593,594,1,0,0,0,594,603,1,0,0,0,595,597,5,35,0,0,596,598,5,65,0,0,597, + 596,1,0,0,0,597,598,1,0,0,0,598,600,1,0,0,0,599,601,7,5,0,0,600,599,1, + 0,0,0,600,601,1,0,0,0,601,603,1,0,0,0,602,589,1,0,0,0,602,595,1,0,0,0, + 603,605,1,0,0,0,604,570,1,0,0,0,604,586,1,0,0,0,604,602,1,0,0,0,605,81, + 1,0,0,0,606,607,5,17,0,0,607,610,5,47,0,0,608,610,5,116,0,0,609,606,1, + 0,0,0,609,608,1,0,0,0,610,83,1,0,0,0,611,612,5,62,0,0,612,621,3,114,57, + 0,613,614,5,96,0,0,614,615,5,130,0,0,615,616,3,114,57,0,616,617,5,149, + 0,0,617,621,1,0,0,0,618,619,5,96,0,0,619,621,3,114,57,0,620,611,1,0,0, + 0,620,613,1,0,0,0,620,618,1,0,0,0,621,85,1,0,0,0,622,623,5,77,0,0,623, + 626,3,92,46,0,624,625,5,61,0,0,625,627,3,92,46,0,626,624,1,0,0,0,626, + 627,1,0,0,0,627,87,1,0,0,0,628,633,3,90,45,0,629,630,5,116,0,0,630,632, + 3,90,45,0,631,629,1,0,0,0,632,635,1,0,0,0,633,631,1,0,0,0,633,634,1,0, + 0,0,634,89,1,0,0,0,635,633,1,0,0,0,636,638,3,116,58,0,637,639,7,6,0,0, + 638,637,1,0,0,0,638,639,1,0,0,0,639,642,1,0,0,0,640,641,5,60,0,0,641, + 643,7,7,0,0,642,640,1,0,0,0,642,643,1,0,0,0,643,646,1,0,0,0,644,645,5, + 16,0,0,645,647,5,110,0,0,646,644,1,0,0,0,646,647,1,0,0,0,647,91,1,0,0, + 0,648,655,3,160,80,0,649,652,3,144,72,0,650,651,5,151,0,0,651,653,3,144, + 72,0,652,650,1,0,0,0,652,653,1,0,0,0,653,655,1,0,0,0,654,648,1,0,0,0, + 654,649,1,0,0,0,655,93,1,0,0,0,656,661,3,96,48,0,657,658,5,116,0,0,658, + 660,3,96,48,0,659,657,1,0,0,0,660,663,1,0,0,0,661,659,1,0,0,0,661,662, + 1,0,0,0,662,95,1,0,0,0,663,661,1,0,0,0,664,665,3,156,78,0,665,666,5,122, + 0,0,666,667,3,146,73,0,667,97,1,0,0,0,668,670,3,100,50,0,669,668,1,0, + 0,0,669,670,1,0,0,0,670,672,1,0,0,0,671,673,3,102,51,0,672,671,1,0,0, + 0,672,673,1,0,0,0,673,675,1,0,0,0,674,676,3,104,52,0,675,674,1,0,0,0, + 675,676,1,0,0,0,676,99,1,0,0,0,677,678,5,67,0,0,678,679,5,11,0,0,679, + 680,3,114,57,0,680,101,1,0,0,0,681,682,5,64,0,0,682,683,5,11,0,0,683, + 684,3,88,44,0,684,103,1,0,0,0,685,686,7,8,0,0,686,687,3,106,53,0,687, + 105,1,0,0,0,688,695,3,108,54,0,689,690,5,9,0,0,690,691,3,108,54,0,691, + 692,5,2,0,0,692,693,3,108,54,0,693,695,1,0,0,0,694,688,1,0,0,0,694,689, + 1,0,0,0,695,107,1,0,0,0,696,697,5,19,0,0,697,709,5,75,0,0,698,699,5,94, + 0,0,699,709,5,68,0,0,700,701,5,94,0,0,701,709,5,32,0,0,702,703,3,144, + 72,0,703,704,5,68,0,0,704,709,1,0,0,0,705,706,3,144,72,0,706,707,5,32, + 0,0,707,709,1,0,0,0,708,696,1,0,0,0,708,698,1,0,0,0,708,700,1,0,0,0,708, + 702,1,0,0,0,708,705,1,0,0,0,709,109,1,0,0,0,710,711,3,116,58,0,711,712, + 5,0,0,1,712,111,1,0,0,0,713,770,3,156,78,0,714,715,3,156,78,0,715,716, + 5,130,0,0,716,717,3,156,78,0,717,724,3,112,56,0,718,719,5,116,0,0,719, + 720,3,156,78,0,720,721,3,112,56,0,721,723,1,0,0,0,722,718,1,0,0,0,723, + 726,1,0,0,0,724,722,1,0,0,0,724,725,1,0,0,0,725,728,1,0,0,0,726,724,1, + 0,0,0,727,729,5,116,0,0,728,727,1,0,0,0,728,729,1,0,0,0,729,730,1,0,0, + 0,730,731,5,149,0,0,731,770,1,0,0,0,732,733,3,156,78,0,733,734,5,130, + 0,0,734,739,3,158,79,0,735,736,5,116,0,0,736,738,3,158,79,0,737,735,1, + 0,0,0,738,741,1,0,0,0,739,737,1,0,0,0,739,740,1,0,0,0,740,743,1,0,0,0, + 741,739,1,0,0,0,742,744,5,116,0,0,743,742,1,0,0,0,743,744,1,0,0,0,744, + 745,1,0,0,0,745,746,5,149,0,0,746,770,1,0,0,0,747,748,3,156,78,0,748, + 749,5,130,0,0,749,754,3,112,56,0,750,751,5,116,0,0,751,753,3,112,56,0, + 752,750,1,0,0,0,753,756,1,0,0,0,754,752,1,0,0,0,754,755,1,0,0,0,755,758, + 1,0,0,0,756,754,1,0,0,0,757,759,5,116,0,0,758,757,1,0,0,0,758,759,1,0, + 0,0,759,760,1,0,0,0,760,761,5,149,0,0,761,770,1,0,0,0,762,763,3,156,78, + 0,763,765,5,130,0,0,764,766,3,114,57,0,765,764,1,0,0,0,765,766,1,0,0, + 0,766,767,1,0,0,0,767,768,5,149,0,0,768,770,1,0,0,0,769,713,1,0,0,0,769, + 714,1,0,0,0,769,732,1,0,0,0,769,747,1,0,0,0,769,762,1,0,0,0,770,113,1, + 0,0,0,771,776,3,116,58,0,772,773,5,116,0,0,773,775,3,116,58,0,774,772, + 1,0,0,0,775,778,1,0,0,0,776,774,1,0,0,0,776,777,1,0,0,0,777,780,1,0,0, + 0,778,776,1,0,0,0,779,781,5,116,0,0,780,779,1,0,0,0,780,781,1,0,0,0,781, + 115,1,0,0,0,782,783,6,58,-1,0,783,785,5,12,0,0,784,786,3,116,58,0,785, + 784,1,0,0,0,785,786,1,0,0,0,786,792,1,0,0,0,787,788,5,98,0,0,788,789, + 3,116,58,0,789,790,5,83,0,0,790,791,3,116,58,0,791,793,1,0,0,0,792,787, + 1,0,0,0,793,794,1,0,0,0,794,792,1,0,0,0,794,795,1,0,0,0,795,798,1,0,0, + 0,796,797,5,25,0,0,797,799,3,116,58,0,798,796,1,0,0,0,798,799,1,0,0,0, + 799,800,1,0,0,0,800,801,5,26,0,0,801,933,1,0,0,0,802,803,5,13,0,0,803, + 804,5,130,0,0,804,805,3,116,58,0,805,806,5,6,0,0,806,807,3,112,56,0,807, + 808,5,149,0,0,808,933,1,0,0,0,809,810,5,20,0,0,810,933,5,110,0,0,811, + 812,5,45,0,0,812,813,3,116,58,0,813,814,3,148,74,0,814,933,1,0,0,0,815, + 816,5,82,0,0,816,817,5,130,0,0,817,818,3,116,58,0,818,819,5,34,0,0,819, + 822,3,116,58,0,820,821,5,33,0,0,821,823,3,116,58,0,822,820,1,0,0,0,822, + 823,1,0,0,0,823,824,1,0,0,0,824,825,5,149,0,0,825,933,1,0,0,0,826,827, + 5,86,0,0,827,933,5,110,0,0,828,829,5,91,0,0,829,830,5,130,0,0,830,831, + 7,9,0,0,831,832,3,162,81,0,832,833,5,34,0,0,833,834,3,116,58,0,834,835, + 5,149,0,0,835,933,1,0,0,0,836,837,3,156,78,0,837,839,5,130,0,0,838,840, + 3,114,57,0,839,838,1,0,0,0,839,840,1,0,0,0,840,841,1,0,0,0,841,842,5, + 149,0,0,842,851,1,0,0,0,843,845,5,130,0,0,844,846,5,24,0,0,845,844,1, + 0,0,0,845,846,1,0,0,0,846,848,1,0,0,0,847,849,3,114,57,0,848,847,1,0, + 0,0,848,849,1,0,0,0,849,850,1,0,0,0,850,852,5,149,0,0,851,843,1,0,0,0, + 851,852,1,0,0,0,852,853,1,0,0,0,853,854,5,66,0,0,854,855,5,130,0,0,855, + 856,3,98,49,0,856,857,5,149,0,0,857,933,1,0,0,0,858,859,3,156,78,0,859, + 861,5,130,0,0,860,862,3,114,57,0,861,860,1,0,0,0,861,862,1,0,0,0,862, + 863,1,0,0,0,863,864,5,149,0,0,864,873,1,0,0,0,865,867,5,130,0,0,866,868, + 5,24,0,0,867,866,1,0,0,0,867,868,1,0,0,0,868,870,1,0,0,0,869,871,3,114, + 57,0,870,869,1,0,0,0,870,871,1,0,0,0,871,872,1,0,0,0,872,874,5,149,0, + 0,873,865,1,0,0,0,873,874,1,0,0,0,874,875,1,0,0,0,875,876,5,66,0,0,876, + 877,3,156,78,0,877,933,1,0,0,0,878,884,3,156,78,0,879,881,5,130,0,0,880, + 882,3,114,57,0,881,880,1,0,0,0,881,882,1,0,0,0,882,883,1,0,0,0,883,885, + 5,149,0,0,884,879,1,0,0,0,884,885,1,0,0,0,885,886,1,0,0,0,886,888,5,130, + 0,0,887,889,5,24,0,0,888,887,1,0,0,0,888,889,1,0,0,0,889,891,1,0,0,0, + 890,892,3,114,57,0,891,890,1,0,0,0,891,892,1,0,0,0,892,893,1,0,0,0,893, + 894,5,149,0,0,894,933,1,0,0,0,895,933,3,120,60,0,896,933,3,164,82,0,897, + 933,3,146,73,0,898,899,5,118,0,0,899,933,3,116,58,20,900,901,5,58,0,0, + 901,933,3,116,58,14,902,903,3,136,68,0,903,904,5,120,0,0,904,906,1,0, + 0,0,905,902,1,0,0,0,905,906,1,0,0,0,906,907,1,0,0,0,907,933,5,112,0,0, + 908,909,5,130,0,0,909,910,3,44,22,0,910,911,5,149,0,0,911,933,1,0,0,0, + 912,913,5,130,0,0,913,914,3,116,58,0,914,915,5,149,0,0,915,933,1,0,0, + 0,916,917,5,130,0,0,917,918,3,114,57,0,918,919,5,149,0,0,919,933,1,0, + 0,0,920,922,5,129,0,0,921,923,3,114,57,0,922,921,1,0,0,0,922,923,1,0, + 0,0,923,924,1,0,0,0,924,933,5,148,0,0,925,927,5,128,0,0,926,928,3,40, + 20,0,927,926,1,0,0,0,927,928,1,0,0,0,928,929,1,0,0,0,929,933,5,147,0, + 0,930,933,3,118,59,0,931,933,3,128,64,0,932,782,1,0,0,0,932,802,1,0,0, + 0,932,809,1,0,0,0,932,811,1,0,0,0,932,815,1,0,0,0,932,826,1,0,0,0,932, + 828,1,0,0,0,932,836,1,0,0,0,932,858,1,0,0,0,932,878,1,0,0,0,932,895,1, + 0,0,0,932,896,1,0,0,0,932,897,1,0,0,0,932,898,1,0,0,0,932,900,1,0,0,0, + 932,905,1,0,0,0,932,908,1,0,0,0,932,912,1,0,0,0,932,916,1,0,0,0,932,920, + 1,0,0,0,932,925,1,0,0,0,932,930,1,0,0,0,932,931,1,0,0,0,933,1044,1,0, + 0,0,934,938,10,19,0,0,935,939,5,112,0,0,936,939,5,151,0,0,937,939,5,138, + 0,0,938,935,1,0,0,0,938,936,1,0,0,0,938,937,1,0,0,0,939,940,1,0,0,0,940, + 1043,3,116,58,20,941,945,10,18,0,0,942,946,5,139,0,0,943,946,5,118,0, + 0,944,946,5,117,0,0,945,942,1,0,0,0,945,943,1,0,0,0,945,944,1,0,0,0,946, + 947,1,0,0,0,947,1043,3,116,58,19,948,973,10,17,0,0,949,974,5,121,0,0, + 950,974,5,122,0,0,951,974,5,133,0,0,952,974,5,131,0,0,953,974,5,132,0, + 0,954,974,5,123,0,0,955,974,5,124,0,0,956,958,5,58,0,0,957,956,1,0,0, + 0,957,958,1,0,0,0,958,959,1,0,0,0,959,961,5,42,0,0,960,962,5,15,0,0,961, + 960,1,0,0,0,961,962,1,0,0,0,962,974,1,0,0,0,963,965,5,58,0,0,964,963, + 1,0,0,0,964,965,1,0,0,0,965,966,1,0,0,0,966,974,7,10,0,0,967,974,5,145, + 0,0,968,974,5,146,0,0,969,974,5,135,0,0,970,974,5,126,0,0,971,974,5,127, + 0,0,972,974,5,134,0,0,973,949,1,0,0,0,973,950,1,0,0,0,973,951,1,0,0,0, + 973,952,1,0,0,0,973,953,1,0,0,0,973,954,1,0,0,0,973,955,1,0,0,0,973,957, + 1,0,0,0,973,964,1,0,0,0,973,967,1,0,0,0,973,968,1,0,0,0,973,969,1,0,0, + 0,973,970,1,0,0,0,973,971,1,0,0,0,973,972,1,0,0,0,974,975,1,0,0,0,975, + 1043,3,116,58,18,976,977,10,15,0,0,977,978,5,137,0,0,978,1043,3,116,58, + 16,979,980,10,13,0,0,980,981,5,2,0,0,981,1043,3,116,58,14,982,983,10, + 12,0,0,983,984,5,63,0,0,984,1043,3,116,58,13,985,987,10,11,0,0,986,988, + 5,58,0,0,987,986,1,0,0,0,987,988,1,0,0,0,988,989,1,0,0,0,989,990,5,9, + 0,0,990,991,3,116,58,0,991,992,5,2,0,0,992,993,3,116,58,12,993,1043,1, + 0,0,0,994,995,10,10,0,0,995,996,5,140,0,0,996,997,3,116,58,0,997,998, + 5,115,0,0,998,999,3,116,58,10,999,1043,1,0,0,0,1000,1001,10,30,0,0,1001, + 1003,5,130,0,0,1002,1004,3,114,57,0,1003,1002,1,0,0,0,1003,1004,1,0,0, + 0,1004,1005,1,0,0,0,1005,1043,5,149,0,0,1006,1007,10,26,0,0,1007,1008, + 5,129,0,0,1008,1009,3,116,58,0,1009,1010,5,148,0,0,1010,1043,1,0,0,0, + 1011,1012,10,25,0,0,1012,1013,5,120,0,0,1013,1043,5,108,0,0,1014,1015, + 10,24,0,0,1015,1016,5,120,0,0,1016,1043,3,156,78,0,1017,1018,10,23,0, + 0,1018,1019,5,136,0,0,1019,1020,5,129,0,0,1020,1021,3,116,58,0,1021,1022, + 5,148,0,0,1022,1043,1,0,0,0,1023,1024,10,22,0,0,1024,1025,5,136,0,0,1025, + 1043,5,108,0,0,1026,1027,10,21,0,0,1027,1028,5,136,0,0,1028,1043,3,156, + 78,0,1029,1030,10,16,0,0,1030,1032,5,46,0,0,1031,1033,5,58,0,0,1032,1031, + 1,0,0,0,1032,1033,1,0,0,0,1033,1034,1,0,0,0,1034,1043,5,59,0,0,1035,1040, + 10,9,0,0,1036,1037,5,6,0,0,1037,1041,3,156,78,0,1038,1039,5,6,0,0,1039, + 1041,5,110,0,0,1040,1036,1,0,0,0,1040,1038,1,0,0,0,1041,1043,1,0,0,0, + 1042,934,1,0,0,0,1042,941,1,0,0,0,1042,948,1,0,0,0,1042,976,1,0,0,0,1042, + 979,1,0,0,0,1042,982,1,0,0,0,1042,985,1,0,0,0,1042,994,1,0,0,0,1042,1000, + 1,0,0,0,1042,1006,1,0,0,0,1042,1011,1,0,0,0,1042,1014,1,0,0,0,1042,1017, + 1,0,0,0,1042,1023,1,0,0,0,1042,1026,1,0,0,0,1042,1029,1,0,0,0,1042,1035, + 1,0,0,0,1043,1046,1,0,0,0,1044,1042,1,0,0,0,1044,1045,1,0,0,0,1045,117, + 1,0,0,0,1046,1044,1,0,0,0,1047,1048,5,130,0,0,1048,1053,3,156,78,0,1049, + 1050,5,116,0,0,1050,1052,3,156,78,0,1051,1049,1,0,0,0,1052,1055,1,0,0, + 0,1053,1051,1,0,0,0,1053,1054,1,0,0,0,1054,1057,1,0,0,0,1055,1053,1,0, + 0,0,1056,1058,5,116,0,0,1057,1056,1,0,0,0,1057,1058,1,0,0,0,1058,1059, + 1,0,0,0,1059,1060,5,149,0,0,1060,1075,1,0,0,0,1061,1066,3,156,78,0,1062, + 1063,5,116,0,0,1063,1065,3,156,78,0,1064,1062,1,0,0,0,1065,1068,1,0,0, + 0,1066,1064,1,0,0,0,1066,1067,1,0,0,0,1067,1070,1,0,0,0,1068,1066,1,0, + 0,0,1069,1071,5,116,0,0,1070,1069,1,0,0,0,1070,1071,1,0,0,0,1071,1075, + 1,0,0,0,1072,1073,5,130,0,0,1073,1075,5,149,0,0,1074,1047,1,0,0,0,1074, + 1061,1,0,0,0,1074,1072,1,0,0,0,1075,1076,1,0,0,0,1076,1079,5,111,0,0, + 1077,1080,3,116,58,0,1078,1080,3,36,18,0,1079,1077,1,0,0,0,1079,1078, + 1,0,0,0,1080,119,1,0,0,0,1081,1082,5,132,0,0,1082,1086,3,156,78,0,1083, + 1085,3,122,61,0,1084,1083,1,0,0,0,1085,1088,1,0,0,0,1086,1084,1,0,0,0, + 1086,1087,1,0,0,0,1087,1089,1,0,0,0,1088,1086,1,0,0,0,1089,1090,5,151, + 0,0,1090,1091,5,124,0,0,1091,1114,1,0,0,0,1092,1093,5,132,0,0,1093,1097, + 3,156,78,0,1094,1096,3,122,61,0,1095,1094,1,0,0,0,1096,1099,1,0,0,0,1097, + 1095,1,0,0,0,1097,1098,1,0,0,0,1098,1100,1,0,0,0,1099,1097,1,0,0,0,1100, + 1106,5,124,0,0,1101,1107,3,120,60,0,1102,1103,5,128,0,0,1103,1104,3,116, + 58,0,1104,1105,5,147,0,0,1105,1107,1,0,0,0,1106,1101,1,0,0,0,1106,1102, + 1,0,0,0,1106,1107,1,0,0,0,1107,1108,1,0,0,0,1108,1109,5,132,0,0,1109, + 1110,5,151,0,0,1110,1111,3,156,78,0,1111,1112,5,124,0,0,1112,1114,1,0, + 0,0,1113,1081,1,0,0,0,1113,1092,1,0,0,0,1114,121,1,0,0,0,1115,1116,3, + 156,78,0,1116,1117,5,122,0,0,1117,1118,3,162,81,0,1118,1127,1,0,0,0,1119, + 1120,3,156,78,0,1120,1121,5,122,0,0,1121,1122,5,128,0,0,1122,1123,3,116, + 58,0,1123,1124,5,147,0,0,1124,1127,1,0,0,0,1125,1127,3,156,78,0,1126, + 1115,1,0,0,0,1126,1119,1,0,0,0,1126,1125,1,0,0,0,1127,123,1,0,0,0,1128, + 1133,3,126,63,0,1129,1130,5,116,0,0,1130,1132,3,126,63,0,1131,1129,1, + 0,0,0,1132,1135,1,0,0,0,1133,1131,1,0,0,0,1133,1134,1,0,0,0,1134,1137, + 1,0,0,0,1135,1133,1,0,0,0,1136,1138,5,116,0,0,1137,1136,1,0,0,0,1137, + 1138,1,0,0,0,1138,125,1,0,0,0,1139,1140,3,156,78,0,1140,1141,5,6,0,0, + 1141,1142,5,130,0,0,1142,1143,3,44,22,0,1143,1144,5,149,0,0,1144,1150, + 1,0,0,0,1145,1146,3,116,58,0,1146,1147,5,6,0,0,1147,1148,3,156,78,0,1148, + 1150,1,0,0,0,1149,1139,1,0,0,0,1149,1145,1,0,0,0,1150,127,1,0,0,0,1151, + 1159,3,160,80,0,1152,1153,3,136,68,0,1153,1154,5,120,0,0,1154,1156,1, + 0,0,0,1155,1152,1,0,0,0,1155,1156,1,0,0,0,1156,1157,1,0,0,0,1157,1159, + 3,130,65,0,1158,1151,1,0,0,0,1158,1155,1,0,0,0,1159,129,1,0,0,0,1160, + 1165,3,156,78,0,1161,1162,5,120,0,0,1162,1164,3,156,78,0,1163,1161,1, + 0,0,0,1164,1167,1,0,0,0,1165,1163,1,0,0,0,1165,1166,1,0,0,0,1166,131, + 1,0,0,0,1167,1165,1,0,0,0,1168,1169,6,66,-1,0,1169,1178,3,136,68,0,1170, + 1178,3,134,67,0,1171,1172,5,130,0,0,1172,1173,3,44,22,0,1173,1174,5,149, + 0,0,1174,1178,1,0,0,0,1175,1178,3,120,60,0,1176,1178,3,160,80,0,1177, + 1168,1,0,0,0,1177,1170,1,0,0,0,1177,1171,1,0,0,0,1177,1175,1,0,0,0,1177, + 1176,1,0,0,0,1178,1187,1,0,0,0,1179,1183,10,3,0,0,1180,1184,3,154,77, + 0,1181,1182,5,6,0,0,1182,1184,3,156,78,0,1183,1180,1,0,0,0,1183,1181, + 1,0,0,0,1184,1186,1,0,0,0,1185,1179,1,0,0,0,1186,1189,1,0,0,0,1187,1185, + 1,0,0,0,1187,1188,1,0,0,0,1188,133,1,0,0,0,1189,1187,1,0,0,0,1190,1191, + 3,156,78,0,1191,1193,5,130,0,0,1192,1194,3,138,69,0,1193,1192,1,0,0,0, + 1193,1194,1,0,0,0,1194,1195,1,0,0,0,1195,1196,5,149,0,0,1196,135,1,0, + 0,0,1197,1198,3,140,70,0,1198,1199,5,120,0,0,1199,1201,1,0,0,0,1200,1197, + 1,0,0,0,1200,1201,1,0,0,0,1201,1202,1,0,0,0,1202,1203,3,156,78,0,1203, + 137,1,0,0,0,1204,1209,3,116,58,0,1205,1206,5,116,0,0,1206,1208,3,116, + 58,0,1207,1205,1,0,0,0,1208,1211,1,0,0,0,1209,1207,1,0,0,0,1209,1210, + 1,0,0,0,1210,1213,1,0,0,0,1211,1209,1,0,0,0,1212,1214,5,116,0,0,1213, + 1212,1,0,0,0,1213,1214,1,0,0,0,1214,139,1,0,0,0,1215,1216,3,156,78,0, + 1216,141,1,0,0,0,1217,1226,5,106,0,0,1218,1219,5,120,0,0,1219,1226,7, + 11,0,0,1220,1221,5,108,0,0,1221,1223,5,120,0,0,1222,1224,7,11,0,0,1223, + 1222,1,0,0,0,1223,1224,1,0,0,0,1224,1226,1,0,0,0,1225,1217,1,0,0,0,1225, + 1218,1,0,0,0,1225,1220,1,0,0,0,1226,143,1,0,0,0,1227,1229,7,12,0,0,1228, + 1227,1,0,0,0,1228,1229,1,0,0,0,1229,1236,1,0,0,0,1230,1237,3,142,71,0, + 1231,1237,5,107,0,0,1232,1237,5,108,0,0,1233,1237,5,109,0,0,1234,1237, + 5,43,0,0,1235,1237,5,57,0,0,1236,1230,1,0,0,0,1236,1231,1,0,0,0,1236, + 1232,1,0,0,0,1236,1233,1,0,0,0,1236,1234,1,0,0,0,1236,1235,1,0,0,0,1237, + 145,1,0,0,0,1238,1242,3,144,72,0,1239,1242,5,110,0,0,1240,1242,5,59,0, + 0,1241,1238,1,0,0,0,1241,1239,1,0,0,0,1241,1240,1,0,0,0,1242,147,1,0, + 0,0,1243,1244,7,13,0,0,1244,149,1,0,0,0,1245,1246,7,14,0,0,1246,151,1, + 0,0,0,1247,1248,7,15,0,0,1248,153,1,0,0,0,1249,1252,5,105,0,0,1250,1252, + 3,152,76,0,1251,1249,1,0,0,0,1251,1250,1,0,0,0,1252,155,1,0,0,0,1253, + 1257,5,105,0,0,1254,1257,3,148,74,0,1255,1257,3,150,75,0,1256,1253,1, + 0,0,0,1256,1254,1,0,0,0,1256,1255,1,0,0,0,1257,157,1,0,0,0,1258,1259, + 3,162,81,0,1259,1260,5,122,0,0,1260,1261,3,144,72,0,1261,159,1,0,0,0, + 1262,1263,5,128,0,0,1263,1264,3,130,65,0,1264,1265,5,147,0,0,1265,161, + 1,0,0,0,1266,1269,5,110,0,0,1267,1269,3,164,82,0,1268,1266,1,0,0,0,1268, + 1267,1,0,0,0,1269,163,1,0,0,0,1270,1274,5,142,0,0,1271,1273,3,166,83, + 0,1272,1271,1,0,0,0,1273,1276,1,0,0,0,1274,1272,1,0,0,0,1274,1275,1,0, + 0,0,1275,1277,1,0,0,0,1276,1274,1,0,0,0,1277,1278,5,144,0,0,1278,165, + 1,0,0,0,1279,1280,5,157,0,0,1280,1281,3,116,58,0,1281,1282,5,147,0,0, + 1282,1285,1,0,0,0,1283,1285,5,156,0,0,1284,1279,1,0,0,0,1284,1283,1,0, + 0,0,1285,167,1,0,0,0,1286,1290,5,143,0,0,1287,1289,3,170,85,0,1288,1287, + 1,0,0,0,1289,1292,1,0,0,0,1290,1288,1,0,0,0,1290,1291,1,0,0,0,1291,1293, + 1,0,0,0,1292,1290,1,0,0,0,1293,1294,5,0,0,1,1294,169,1,0,0,0,1295,1296, + 5,159,0,0,1296,1297,3,116,58,0,1297,1298,5,147,0,0,1298,1301,1,0,0,0, + 1299,1301,5,158,0,0,1300,1295,1,0,0,0,1300,1299,1,0,0,0,1301,171,1,0, + 0,0,167,175,182,191,198,202,216,220,223,227,230,237,241,250,255,264,272, + 279,283,289,294,302,309,315,327,335,349,353,358,368,377,380,384,387,391, + 394,397,400,403,407,411,414,417,420,424,427,436,442,463,480,497,503,509, + 520,522,533,536,542,550,556,558,562,567,570,573,577,581,584,586,589,593, + 597,600,602,604,609,620,626,633,638,642,646,652,654,661,669,672,675,694, + 708,724,728,739,743,754,758,765,769,776,780,785,794,798,822,839,845,848, + 851,861,867,870,873,881,884,888,891,905,922,927,932,938,945,957,961,964, + 973,987,1003,1032,1040,1042,1044,1053,1057,1066,1070,1074,1079,1086,1097, + 1106,1113,1126,1133,1137,1149,1155,1158,1165,1177,1183,1187,1193,1200, + 1209,1213,1223,1225,1228,1236,1241,1251,1256,1268,1274,1284,1290,1300 }; staticData->serializedATN = antlr4::atn::SerializedATNView(serializedATNSegment, sizeof(serializedATNSegment) / sizeof(serializedATNSegment[0])); @@ -615,7 +611,7 @@ void hogqlparserParserInitialize() { for (size_t i = 0; i < count; i++) { staticData->decisionToDFA.emplace_back(staticData->atn->getDecisionState(i), i); } - hogqlparserParserStaticData = staticData.release(); + hogqlparserParserStaticData = std::move(staticData); } } @@ -697,20 +693,20 @@ HogQLParser::ProgramContext* HogQLParser::program() { }); try { enterOuterAlt(_localctx, 1); - setState(179); + setState(175); _errHandler->sync(this); _la = _input->LA(1); while ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & -536887298) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493105500848127) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 4212759) != 0)) { - setState(176); + setState(172); declaration(); - setState(181); + setState(177); _errHandler->sync(this); _la = _input->LA(1); } - setState(182); + setState(178); match(HogQLParser::EOF); } @@ -762,12 +758,12 @@ HogQLParser::DeclarationContext* HogQLParser::declaration() { exitRule(); }); try { - setState(186); + setState(182); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::LET: { enterOuterAlt(_localctx, 1); - setState(184); + setState(180); varDecl(); break; } @@ -889,7 +885,7 @@ HogQLParser::DeclarationContext* HogQLParser::declaration() { case HogQLParser::QUOTE_SINGLE_TEMPLATE: case HogQLParser::SEMICOLON: { enterOuterAlt(_localctx, 2); - setState(185); + setState(181); statement(); break; } @@ -944,7 +940,7 @@ HogQLParser::ExpressionContext* HogQLParser::expression() { }); try { enterOuterAlt(_localctx, 1); - setState(188); + setState(184); columnExpr(0); } @@ -1010,20 +1006,20 @@ HogQLParser::VarDeclContext* HogQLParser::varDecl() { }); try { enterOuterAlt(_localctx, 1); - setState(190); + setState(186); match(HogQLParser::LET); - setState(191); + setState(187); identifier(); - setState(195); + setState(191); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COLON) { - setState(192); + setState(188); match(HogQLParser::COLON); - setState(193); + setState(189); match(HogQLParser::EQ_SINGLE); - setState(194); + setState(190); expression(); } @@ -1087,28 +1083,28 @@ HogQLParser::IdentifierListContext* HogQLParser::identifierList() { try { size_t alt; enterOuterAlt(_localctx, 1); - setState(197); + setState(193); identifier(); - setState(202); + setState(198); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 3, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { - setState(198); + setState(194); match(HogQLParser::COMMA); - setState(199); + setState(195); identifier(); } - setState(204); + setState(200); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 3, _ctx); } - setState(206); + setState(202); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COMMA) { - setState(205); + setState(201); match(HogQLParser::COMMA); } @@ -1201,89 +1197,89 @@ HogQLParser::StatementContext* HogQLParser::statement() { exitRule(); }); try { - setState(220); + setState(216); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 5, _ctx)) { case 1: { enterOuterAlt(_localctx, 1); - setState(208); + setState(204); returnStmt(); break; } case 2: { enterOuterAlt(_localctx, 2); - setState(209); + setState(205); throwStmt(); break; } case 3: { enterOuterAlt(_localctx, 3); - setState(210); + setState(206); tryCatchStmt(); break; } case 4: { enterOuterAlt(_localctx, 4); - setState(211); + setState(207); ifStmt(); break; } case 5: { enterOuterAlt(_localctx, 5); - setState(212); + setState(208); whileStmt(); break; } case 6: { enterOuterAlt(_localctx, 6); - setState(213); + setState(209); forInStmt(); break; } case 7: { enterOuterAlt(_localctx, 7); - setState(214); + setState(210); forStmt(); break; } case 8: { enterOuterAlt(_localctx, 8); - setState(215); + setState(211); funcStmt(); break; } case 9: { enterOuterAlt(_localctx, 9); - setState(216); + setState(212); varAssignment(); break; } case 10: { enterOuterAlt(_localctx, 10); - setState(217); + setState(213); block(); break; } case 11: { enterOuterAlt(_localctx, 11); - setState(218); + setState(214); exprStmt(); break; } case 12: { enterOuterAlt(_localctx, 12); - setState(219); + setState(215); emptyStmt(); break; } @@ -1346,14 +1342,14 @@ HogQLParser::ReturnStmtContext* HogQLParser::returnStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(222); + setState(218); match(HogQLParser::RETURN); - setState(224); + setState(220); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 6, _ctx)) { case 1: { - setState(223); + setState(219); expression(); break; } @@ -1361,12 +1357,12 @@ HogQLParser::ReturnStmtContext* HogQLParser::returnStmt() { default: break; } - setState(227); + setState(223); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 7, _ctx)) { case 1: { - setState(226); + setState(222); match(HogQLParser::SEMICOLON); break; } @@ -1429,14 +1425,14 @@ HogQLParser::ThrowStmtContext* HogQLParser::throwStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(229); + setState(225); match(HogQLParser::THROW); - setState(231); + setState(227); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 8, _ctx)) { case 1: { - setState(230); + setState(226); expression(); break; } @@ -1444,12 +1440,12 @@ HogQLParser::ThrowStmtContext* HogQLParser::throwStmt() { default: break; } - setState(234); + setState(230); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 9, _ctx)) { case 1: { - setState(233); + setState(229); match(HogQLParser::SEMICOLON); break; } @@ -1529,31 +1525,31 @@ HogQLParser::CatchBlockContext* HogQLParser::catchBlock() { }); try { enterOuterAlt(_localctx, 1); - setState(236); + setState(232); match(HogQLParser::CATCH); - setState(245); + setState(241); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::LPAREN) { - setState(237); + setState(233); match(HogQLParser::LPAREN); - setState(238); + setState(234); antlrcpp::downCast(_localctx)->catchVar = identifier(); - setState(241); + setState(237); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COLON) { - setState(239); + setState(235); match(HogQLParser::COLON); - setState(240); + setState(236); antlrcpp::downCast(_localctx)->catchType = identifier(); } - setState(243); + setState(239); match(HogQLParser::RPAREN); } - setState(247); + setState(243); antlrcpp::downCast(_localctx)->catchStmt = block(); } @@ -1623,28 +1619,28 @@ HogQLParser::TryCatchStmtContext* HogQLParser::tryCatchStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(249); + setState(245); match(HogQLParser::TRY); - setState(250); + setState(246); antlrcpp::downCast(_localctx)->tryStmt = block(); - setState(254); + setState(250); _errHandler->sync(this); _la = _input->LA(1); while (_la == HogQLParser::CATCH) { - setState(251); + setState(247); catchBlock(); - setState(256); + setState(252); _errHandler->sync(this); _la = _input->LA(1); } - setState(259); + setState(255); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::FINALLY) { - setState(257); + setState(253); match(HogQLParser::FINALLY); - setState(258); + setState(254); antlrcpp::downCast(_localctx)->finallyStmt = block(); } @@ -1718,24 +1714,24 @@ HogQLParser::IfStmtContext* HogQLParser::ifStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(261); + setState(257); match(HogQLParser::IF); - setState(262); + setState(258); match(HogQLParser::LPAREN); - setState(263); + setState(259); expression(); - setState(264); + setState(260); match(HogQLParser::RPAREN); - setState(265); + setState(261); statement(); - setState(268); + setState(264); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 14, _ctx)) { case 1: { - setState(266); + setState(262); match(HogQLParser::ELSE); - setState(267); + setState(263); statement(); break; } @@ -1810,22 +1806,22 @@ HogQLParser::WhileStmtContext* HogQLParser::whileStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(270); + setState(266); match(HogQLParser::WHILE); - setState(271); + setState(267); match(HogQLParser::LPAREN); - setState(272); + setState(268); expression(); - setState(273); + setState(269); match(HogQLParser::RPAREN); - setState(274); + setState(270); statement(); - setState(276); + setState(272); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 15, _ctx)) { case 1: { - setState(275); + setState(271); match(HogQLParser::SEMICOLON); break; } @@ -1925,28 +1921,28 @@ HogQLParser::ForStmtContext* HogQLParser::forStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(278); + setState(274); match(HogQLParser::FOR); - setState(279); + setState(275); match(HogQLParser::LPAREN); - setState(283); + setState(279); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 16, _ctx)) { case 1: { - setState(280); + setState(276); antlrcpp::downCast(_localctx)->initializerVarDeclr = varDecl(); break; } case 2: { - setState(281); + setState(277); antlrcpp::downCast(_localctx)->initializerVarAssignment = varAssignment(); break; } case 3: { - setState(282); + setState(278); antlrcpp::downCast(_localctx)->initializerExpression = expression(); break; } @@ -1954,9 +1950,9 @@ HogQLParser::ForStmtContext* HogQLParser::forStmt() { default: break; } - setState(285); + setState(281); match(HogQLParser::SEMICOLON); - setState(287); + setState(283); _errHandler->sync(this); _la = _input->LA(1); @@ -1964,29 +1960,29 @@ HogQLParser::ForStmtContext* HogQLParser::forStmt() { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(286); + setState(282); antlrcpp::downCast(_localctx)->condition = expression(); } - setState(289); + setState(285); match(HogQLParser::SEMICOLON); - setState(293); + setState(289); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 18, _ctx)) { case 1: { - setState(290); + setState(286); antlrcpp::downCast(_localctx)->incrementVarDeclr = varDecl(); break; } case 2: { - setState(291); + setState(287); antlrcpp::downCast(_localctx)->incrementVarAssignment = varAssignment(); break; } case 3: { - setState(292); + setState(288); antlrcpp::downCast(_localctx)->incrementExpression = expression(); break; } @@ -1994,16 +1990,16 @@ HogQLParser::ForStmtContext* HogQLParser::forStmt() { default: break; } - setState(295); + setState(291); match(HogQLParser::RPAREN); - setState(296); + setState(292); statement(); - setState(298); + setState(294); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 19, _ctx)) { case 1: { - setState(297); + setState(293); match(HogQLParser::SEMICOLON); break; } @@ -2099,38 +2095,38 @@ HogQLParser::ForInStmtContext* HogQLParser::forInStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(300); + setState(296); match(HogQLParser::FOR); - setState(301); + setState(297); match(HogQLParser::LPAREN); - setState(302); + setState(298); match(HogQLParser::LET); - setState(303); + setState(299); identifier(); - setState(306); + setState(302); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COMMA) { - setState(304); + setState(300); match(HogQLParser::COMMA); - setState(305); + setState(301); identifier(); } - setState(308); + setState(304); match(HogQLParser::IN); - setState(309); + setState(305); expression(); - setState(310); + setState(306); match(HogQLParser::RPAREN); - setState(311); + setState(307); statement(); - setState(313); + setState(309); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 21, _ctx)) { case 1: { - setState(312); + setState(308); match(HogQLParser::SEMICOLON); break; } @@ -2206,25 +2202,25 @@ HogQLParser::FuncStmtContext* HogQLParser::funcStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(315); + setState(311); match(HogQLParser::FN); - setState(316); + setState(312); identifier(); - setState(317); + setState(313); match(HogQLParser::LPAREN); - setState(319); + setState(315); _errHandler->sync(this); _la = _input->LA(1); if ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & -725088338784043010) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 3229277487103) != 0)) { - setState(318); + setState(314); identifierList(); } - setState(321); + setState(317); match(HogQLParser::RPAREN); - setState(322); + setState(318); block(); } @@ -2285,13 +2281,13 @@ HogQLParser::VarAssignmentContext* HogQLParser::varAssignment() { }); try { enterOuterAlt(_localctx, 1); - setState(324); + setState(320); expression(); - setState(325); + setState(321); match(HogQLParser::COLON); - setState(326); + setState(322); match(HogQLParser::EQ_SINGLE); - setState(327); + setState(323); expression(); } @@ -2344,14 +2340,14 @@ HogQLParser::ExprStmtContext* HogQLParser::exprStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(329); + setState(325); expression(); - setState(331); + setState(327); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 23, _ctx)) { case 1: { - setState(330); + setState(326); match(HogQLParser::SEMICOLON); break; } @@ -2406,7 +2402,7 @@ HogQLParser::EmptyStmtContext* HogQLParser::emptyStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(333); + setState(329); match(HogQLParser::SEMICOLON); } @@ -2468,22 +2464,22 @@ HogQLParser::BlockContext* HogQLParser::block() { }); try { enterOuterAlt(_localctx, 1); - setState(335); + setState(331); match(HogQLParser::LBRACE); - setState(339); + setState(335); _errHandler->sync(this); _la = _input->LA(1); while ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & -536887298) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493105500848127) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 4212759) != 0)) { - setState(336); + setState(332); declaration(); - setState(341); + setState(337); _errHandler->sync(this); _la = _input->LA(1); } - setState(342); + setState(338); match(HogQLParser::RBRACE); } @@ -2540,11 +2536,11 @@ HogQLParser::KvPairContext* HogQLParser::kvPair() { }); try { enterOuterAlt(_localctx, 1); - setState(344); + setState(340); expression(); - setState(345); + setState(341); match(HogQLParser::COLON); - setState(346); + setState(342); expression(); } @@ -2607,28 +2603,28 @@ HogQLParser::KvPairListContext* HogQLParser::kvPairList() { try { size_t alt; enterOuterAlt(_localctx, 1); - setState(348); + setState(344); kvPair(); - setState(353); + setState(349); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 25, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { - setState(349); + setState(345); match(HogQLParser::COMMA); - setState(350); + setState(346); kvPair(); } - setState(355); + setState(351); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 25, _ctx); } - setState(357); + setState(353); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COMMA) { - setState(356); + setState(352); match(HogQLParser::COMMA); } @@ -2690,23 +2686,23 @@ HogQLParser::SelectContext* HogQLParser::select() { }); try { enterOuterAlt(_localctx, 1); - setState(362); + setState(358); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 27, _ctx)) { case 1: { - setState(359); + setState(355); selectUnionStmt(); break; } case 2: { - setState(360); + setState(356); selectStmt(); break; } case 3: { - setState(361); + setState(357); hogqlxTagElement(); break; } @@ -2714,7 +2710,7 @@ HogQLParser::SelectContext* HogQLParser::select() { default: break; } - setState(364); + setState(360); match(HogQLParser::EOF); } @@ -2784,19 +2780,19 @@ HogQLParser::SelectUnionStmtContext* HogQLParser::selectUnionStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(366); + setState(362); selectStmtWithParens(); - setState(372); + setState(368); _errHandler->sync(this); _la = _input->LA(1); while (_la == HogQLParser::UNION) { - setState(367); + setState(363); match(HogQLParser::UNION); - setState(368); + setState(364); match(HogQLParser::ALL); - setState(369); + setState(365); selectStmtWithParens(); - setState(374); + setState(370); _errHandler->sync(this); _la = _input->LA(1); } @@ -2862,31 +2858,31 @@ HogQLParser::SelectStmtWithParensContext* HogQLParser::selectStmtWithParens() { exitRule(); }); try { - setState(381); + setState(377); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::SELECT: case HogQLParser::WITH: { enterOuterAlt(_localctx, 1); - setState(375); + setState(371); selectStmt(); break; } case HogQLParser::LPAREN: { enterOuterAlt(_localctx, 2); - setState(376); + setState(372); match(HogQLParser::LPAREN); - setState(377); + setState(373); selectUnionStmt(); - setState(378); + setState(374); match(HogQLParser::RPAREN); break; } case HogQLParser::LBRACE: { enterOuterAlt(_localctx, 3); - setState(380); + setState(376); placeholder(); break; } @@ -3022,22 +3018,22 @@ HogQLParser::SelectStmtContext* HogQLParser::selectStmt() { }); try { enterOuterAlt(_localctx, 1); - setState(384); + setState(380); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::WITH) { - setState(383); + setState(379); antlrcpp::downCast(_localctx)->with = withClause(); } - setState(386); + setState(382); match(HogQLParser::SELECT); - setState(388); + setState(384); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 31, _ctx)) { case 1: { - setState(387); + setState(383); match(HogQLParser::DISTINCT); break; } @@ -3045,12 +3041,12 @@ HogQLParser::SelectStmtContext* HogQLParser::selectStmt() { default: break; } - setState(391); + setState(387); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 32, _ctx)) { case 1: { - setState(390); + setState(386); topClause(); break; } @@ -3058,57 +3054,57 @@ HogQLParser::SelectStmtContext* HogQLParser::selectStmt() { default: break; } - setState(393); + setState(389); antlrcpp::downCast(_localctx)->columns = columnExprList(); - setState(395); + setState(391); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::FROM) { - setState(394); + setState(390); antlrcpp::downCast(_localctx)->from = fromClause(); } - setState(398); + setState(394); _errHandler->sync(this); _la = _input->LA(1); if ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 2269391999729696) != 0)) { - setState(397); + setState(393); arrayJoinClause(); } - setState(401); + setState(397); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::PREWHERE) { - setState(400); + setState(396); prewhereClause(); } - setState(404); + setState(400); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::WHERE) { - setState(403); + setState(399); antlrcpp::downCast(_localctx)->where = whereClause(); } - setState(407); + setState(403); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::GROUP) { - setState(406); + setState(402); groupByClause(); } - setState(411); + setState(407); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 38, _ctx)) { case 1: { - setState(409); + setState(405); match(HogQLParser::WITH); - setState(410); + setState(406); _la = _input->LA(1); if (!(_la == HogQLParser::CUBE @@ -3125,51 +3121,51 @@ HogQLParser::SelectStmtContext* HogQLParser::selectStmt() { default: break; } - setState(415); + setState(411); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::WITH) { - setState(413); + setState(409); match(HogQLParser::WITH); - setState(414); + setState(410); match(HogQLParser::TOTALS); } - setState(418); + setState(414); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::HAVING) { - setState(417); + setState(413); havingClause(); } - setState(421); + setState(417); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::WINDOW) { - setState(420); + setState(416); windowClause(); } - setState(424); + setState(420); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::ORDER) { - setState(423); + setState(419); orderByClause(); } - setState(428); + setState(424); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::LIMIT: { - setState(426); + setState(422); limitAndOffsetClause(); break; } case HogQLParser::OFFSET: { - setState(427); + setState(423); offsetOnlyClause(); break; } @@ -3184,12 +3180,12 @@ HogQLParser::SelectStmtContext* HogQLParser::selectStmt() { default: break; } - setState(431); + setState(427); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::SETTINGS) { - setState(430); + setState(426); settingsClause(); } @@ -3243,9 +3239,9 @@ HogQLParser::WithClauseContext* HogQLParser::withClause() { }); try { enterOuterAlt(_localctx, 1); - setState(433); + setState(429); match(HogQLParser::WITH); - setState(434); + setState(430); withExprList(); } @@ -3306,18 +3302,18 @@ HogQLParser::TopClauseContext* HogQLParser::topClause() { }); try { enterOuterAlt(_localctx, 1); - setState(436); + setState(432); match(HogQLParser::TOP); - setState(437); + setState(433); match(HogQLParser::DECIMAL_LITERAL); - setState(440); + setState(436); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 45, _ctx)) { case 1: { - setState(438); + setState(434); match(HogQLParser::WITH); - setState(439); + setState(435); match(HogQLParser::TIES); break; } @@ -3376,9 +3372,9 @@ HogQLParser::FromClauseContext* HogQLParser::fromClause() { }); try { enterOuterAlt(_localctx, 1); - setState(442); + setState(438); match(HogQLParser::FROM); - setState(443); + setState(439); joinExpr(0); } @@ -3444,14 +3440,14 @@ HogQLParser::ArrayJoinClauseContext* HogQLParser::arrayJoinClause() { }); try { enterOuterAlt(_localctx, 1); - setState(446); + setState(442); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::INNER || _la == HogQLParser::LEFT) { - setState(445); + setState(441); _la = _input->LA(1); if (!(_la == HogQLParser::INNER @@ -3463,11 +3459,11 @@ HogQLParser::ArrayJoinClauseContext* HogQLParser::arrayJoinClause() { consume(); } } - setState(448); + setState(444); match(HogQLParser::ARRAY); - setState(449); + setState(445); match(HogQLParser::JOIN); - setState(450); + setState(446); columnExprList(); } @@ -3565,35 +3561,35 @@ HogQLParser::WindowClauseContext* HogQLParser::windowClause() { }); try { enterOuterAlt(_localctx, 1); - setState(452); + setState(448); match(HogQLParser::WINDOW); - setState(453); + setState(449); identifier(); - setState(454); + setState(450); match(HogQLParser::AS); - setState(455); + setState(451); match(HogQLParser::LPAREN); - setState(456); + setState(452); windowExpr(); - setState(457); + setState(453); match(HogQLParser::RPAREN); - setState(467); + setState(463); _errHandler->sync(this); _la = _input->LA(1); while (_la == HogQLParser::COMMA) { - setState(458); + setState(454); match(HogQLParser::COMMA); - setState(459); + setState(455); identifier(); - setState(460); + setState(456); match(HogQLParser::AS); - setState(461); + setState(457); match(HogQLParser::LPAREN); - setState(462); + setState(458); windowExpr(); - setState(463); + setState(459); match(HogQLParser::RPAREN); - setState(469); + setState(465); _errHandler->sync(this); _la = _input->LA(1); } @@ -3648,9 +3644,9 @@ HogQLParser::PrewhereClauseContext* HogQLParser::prewhereClause() { }); try { enterOuterAlt(_localctx, 1); - setState(470); + setState(466); match(HogQLParser::PREWHERE); - setState(471); + setState(467); columnExpr(0); } @@ -3703,9 +3699,9 @@ HogQLParser::WhereClauseContext* HogQLParser::whereClause() { }); try { enterOuterAlt(_localctx, 1); - setState(473); + setState(469); match(HogQLParser::WHERE); - setState(474); + setState(470); columnExpr(0); } @@ -3779,15 +3775,15 @@ HogQLParser::GroupByClauseContext* HogQLParser::groupByClause() { }); try { enterOuterAlt(_localctx, 1); - setState(476); + setState(472); match(HogQLParser::GROUP); - setState(477); + setState(473); match(HogQLParser::BY); - setState(484); + setState(480); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 48, _ctx)) { case 1: { - setState(478); + setState(474); _la = _input->LA(1); if (!(_la == HogQLParser::CUBE @@ -3798,17 +3794,17 @@ HogQLParser::GroupByClauseContext* HogQLParser::groupByClause() { _errHandler->reportMatch(this); consume(); } - setState(479); + setState(475); match(HogQLParser::LPAREN); - setState(480); + setState(476); columnExprList(); - setState(481); + setState(477); match(HogQLParser::RPAREN); break; } case 2: { - setState(483); + setState(479); columnExprList(); break; } @@ -3867,9 +3863,9 @@ HogQLParser::HavingClauseContext* HogQLParser::havingClause() { }); try { enterOuterAlt(_localctx, 1); - setState(486); + setState(482); match(HogQLParser::HAVING); - setState(487); + setState(483); columnExpr(0); } @@ -3926,11 +3922,11 @@ HogQLParser::OrderByClauseContext* HogQLParser::orderByClause() { }); try { enterOuterAlt(_localctx, 1); - setState(489); + setState(485); match(HogQLParser::ORDER); - setState(490); + setState(486); match(HogQLParser::BY); - setState(491); + setState(487); orderExprList(); } @@ -3987,11 +3983,11 @@ HogQLParser::ProjectionOrderByClauseContext* HogQLParser::projectionOrderByClaus }); try { enterOuterAlt(_localctx, 1); - setState(493); + setState(489); match(HogQLParser::ORDER); - setState(494); + setState(490); match(HogQLParser::BY); - setState(495); + setState(491); columnExprList(); } @@ -4072,40 +4068,40 @@ HogQLParser::LimitAndOffsetClauseContext* HogQLParser::limitAndOffsetClause() { exitRule(); }); try { - setState(526); + setState(522); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 53, _ctx)) { case 1: { enterOuterAlt(_localctx, 1); - setState(497); + setState(493); match(HogQLParser::LIMIT); - setState(498); + setState(494); columnExpr(0); - setState(501); + setState(497); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COMMA) { - setState(499); + setState(495); match(HogQLParser::COMMA); - setState(500); + setState(496); columnExpr(0); } - setState(507); + setState(503); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::WITH: { - setState(503); + setState(499); match(HogQLParser::WITH); - setState(504); + setState(500); match(HogQLParser::TIES); break; } case HogQLParser::BY: { - setState(505); + setState(501); match(HogQLParser::BY); - setState(506); + setState(502); columnExprList(); break; } @@ -4125,45 +4121,45 @@ HogQLParser::LimitAndOffsetClauseContext* HogQLParser::limitAndOffsetClause() { case 2: { enterOuterAlt(_localctx, 2); - setState(509); + setState(505); match(HogQLParser::LIMIT); - setState(510); + setState(506); columnExpr(0); - setState(513); + setState(509); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::WITH) { - setState(511); + setState(507); match(HogQLParser::WITH); - setState(512); + setState(508); match(HogQLParser::TIES); } - setState(515); + setState(511); match(HogQLParser::OFFSET); - setState(516); + setState(512); columnExpr(0); break; } case 3: { enterOuterAlt(_localctx, 3); - setState(518); + setState(514); match(HogQLParser::LIMIT); - setState(519); + setState(515); columnExpr(0); - setState(520); + setState(516); match(HogQLParser::OFFSET); - setState(521); + setState(517); columnExpr(0); - setState(524); + setState(520); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::BY) { - setState(522); + setState(518); match(HogQLParser::BY); - setState(523); + setState(519); columnExprList(); } break; @@ -4223,9 +4219,9 @@ HogQLParser::OffsetOnlyClauseContext* HogQLParser::offsetOnlyClause() { }); try { enterOuterAlt(_localctx, 1); - setState(528); + setState(524); match(HogQLParser::OFFSET); - setState(529); + setState(525); columnExpr(0); } @@ -4278,9 +4274,9 @@ HogQLParser::SettingsClauseContext* HogQLParser::settingsClause() { }); try { enterOuterAlt(_localctx, 1); - setState(531); + setState(527); match(HogQLParser::SETTINGS); - setState(532); + setState(528); settingExprList(); } @@ -4434,7 +4430,7 @@ HogQLParser::JoinExprContext* HogQLParser::joinExpr(int precedence) { try { size_t alt; enterOuterAlt(_localctx, 1); - setState(546); + setState(542); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 56, _ctx)) { case 1: { @@ -4442,14 +4438,14 @@ HogQLParser::JoinExprContext* HogQLParser::joinExpr(int precedence) { _ctx = _localctx; previousContext = _localctx; - setState(535); + setState(531); tableExpr(0); - setState(537); + setState(533); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 54, _ctx)) { case 1: { - setState(536); + setState(532); match(HogQLParser::FINAL); break; } @@ -4457,12 +4453,12 @@ HogQLParser::JoinExprContext* HogQLParser::joinExpr(int precedence) { default: break; } - setState(540); + setState(536); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 55, _ctx)) { case 1: { - setState(539); + setState(535); sampleClause(); break; } @@ -4477,11 +4473,11 @@ HogQLParser::JoinExprContext* HogQLParser::joinExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(542); + setState(538); match(HogQLParser::LPAREN); - setState(543); + setState(539); joinExpr(0); - setState(544); + setState(540); match(HogQLParser::RPAREN); break; } @@ -4490,7 +4486,7 @@ HogQLParser::JoinExprContext* HogQLParser::joinExpr(int precedence) { break; } _ctx->stop = _input->LT(-1); - setState(562); + setState(558); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 59, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { @@ -4498,19 +4494,19 @@ HogQLParser::JoinExprContext* HogQLParser::joinExpr(int precedence) { if (!_parseListeners.empty()) triggerExitRuleEvent(); previousContext = _localctx; - setState(560); + setState(556); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 58, _ctx)) { case 1: { auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleJoinExpr); - setState(548); + setState(544); if (!(precpred(_ctx, 3))) throw FailedPredicateException(this, "precpred(_ctx, 3)"); - setState(549); + setState(545); joinOpCross(); - setState(550); + setState(546); joinExpr(4); break; } @@ -4519,10 +4515,10 @@ HogQLParser::JoinExprContext* HogQLParser::joinExpr(int precedence) { auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleJoinExpr); - setState(552); + setState(548); if (!(precpred(_ctx, 4))) throw FailedPredicateException(this, "precpred(_ctx, 4)"); - setState(554); + setState(550); _errHandler->sync(this); _la = _input->LA(1); @@ -4530,14 +4526,14 @@ HogQLParser::JoinExprContext* HogQLParser::joinExpr(int precedence) { ((1ULL << _la) & 2269426359468314) != 0) || _la == HogQLParser::RIGHT || _la == HogQLParser::SEMI) { - setState(553); + setState(549); joinOp(); } - setState(556); + setState(552); match(HogQLParser::JOIN); - setState(557); + setState(553); joinExpr(0); - setState(558); + setState(554); joinConstraintClause(); break; } @@ -4546,7 +4542,7 @@ HogQLParser::JoinExprContext* HogQLParser::joinExpr(int precedence) { break; } } - setState(564); + setState(560); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 59, _ctx); } @@ -4684,23 +4680,23 @@ HogQLParser::JoinOpContext* HogQLParser::joinOp() { exitRule(); }); try { - setState(608); + setState(604); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 73, _ctx)) { case 1: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 1); - setState(574); + setState(570); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 62, _ctx)) { case 1: { - setState(566); + setState(562); _errHandler->sync(this); _la = _input->LA(1); if ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 274) != 0)) { - setState(565); + setState(561); _la = _input->LA(1); if (!((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 274) != 0))) { @@ -4711,21 +4707,21 @@ HogQLParser::JoinOpContext* HogQLParser::joinOp() { consume(); } } - setState(568); + setState(564); match(HogQLParser::INNER); break; } case 2: { - setState(569); + setState(565); match(HogQLParser::INNER); - setState(571); + setState(567); _errHandler->sync(this); _la = _input->LA(1); if ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 274) != 0)) { - setState(570); + setState(566); _la = _input->LA(1); if (!((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 274) != 0))) { @@ -4740,7 +4736,7 @@ HogQLParser::JoinOpContext* HogQLParser::joinOp() { } case 3: { - setState(573); + setState(569); _la = _input->LA(1); if (!((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 274) != 0))) { @@ -4762,17 +4758,17 @@ HogQLParser::JoinOpContext* HogQLParser::joinOp() { case 2: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 2); - setState(590); + setState(586); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 67, _ctx)) { case 1: { - setState(577); + setState(573); _errHandler->sync(this); _la = _input->LA(1); if ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 282) != 0) || _la == HogQLParser::SEMI) { - setState(576); + setState(572); _la = _input->LA(1); if (!((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 282) != 0) || _la == HogQLParser::SEMI)) { @@ -4783,7 +4779,7 @@ HogQLParser::JoinOpContext* HogQLParser::joinOp() { consume(); } } - setState(579); + setState(575); _la = _input->LA(1); if (!(_la == HogQLParser::LEFT @@ -4794,19 +4790,19 @@ HogQLParser::JoinOpContext* HogQLParser::joinOp() { _errHandler->reportMatch(this); consume(); } - setState(581); + setState(577); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::OUTER) { - setState(580); + setState(576); match(HogQLParser::OUTER); } break; } case 2: { - setState(583); + setState(579); _la = _input->LA(1); if (!(_la == HogQLParser::LEFT @@ -4817,21 +4813,21 @@ HogQLParser::JoinOpContext* HogQLParser::joinOp() { _errHandler->reportMatch(this); consume(); } - setState(585); + setState(581); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::OUTER) { - setState(584); + setState(580); match(HogQLParser::OUTER); } - setState(588); + setState(584); _errHandler->sync(this); _la = _input->LA(1); if ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 282) != 0) || _la == HogQLParser::SEMI) { - setState(587); + setState(583); _la = _input->LA(1); if (!((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 282) != 0) || _la == HogQLParser::SEMI)) { @@ -4854,18 +4850,18 @@ HogQLParser::JoinOpContext* HogQLParser::joinOp() { case 3: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 3); - setState(606); + setState(602); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 72, _ctx)) { case 1: { - setState(593); + setState(589); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::ALL || _la == HogQLParser::ANY) { - setState(592); + setState(588); _la = _input->LA(1); if (!(_la == HogQLParser::ALL @@ -4877,38 +4873,38 @@ HogQLParser::JoinOpContext* HogQLParser::joinOp() { consume(); } } - setState(595); + setState(591); match(HogQLParser::FULL); - setState(597); + setState(593); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::OUTER) { - setState(596); + setState(592); match(HogQLParser::OUTER); } break; } case 2: { - setState(599); + setState(595); match(HogQLParser::FULL); - setState(601); + setState(597); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::OUTER) { - setState(600); + setState(596); match(HogQLParser::OUTER); } - setState(604); + setState(600); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::ALL || _la == HogQLParser::ANY) { - setState(603); + setState(599); _la = _input->LA(1); if (!(_la == HogQLParser::ALL @@ -4986,21 +4982,21 @@ HogQLParser::JoinOpCrossContext* HogQLParser::joinOpCross() { exitRule(); }); try { - setState(613); + setState(609); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::CROSS: { enterOuterAlt(_localctx, 1); - setState(610); + setState(606); match(HogQLParser::CROSS); - setState(611); + setState(607); match(HogQLParser::JOIN); break; } case HogQLParser::COMMA: { enterOuterAlt(_localctx, 2); - setState(612); + setState(608); match(HogQLParser::COMMA); break; } @@ -5070,36 +5066,36 @@ HogQLParser::JoinConstraintClauseContext* HogQLParser::joinConstraintClause() { exitRule(); }); try { - setState(624); + setState(620); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 75, _ctx)) { case 1: { enterOuterAlt(_localctx, 1); - setState(615); + setState(611); match(HogQLParser::ON); - setState(616); + setState(612); columnExprList(); break; } case 2: { enterOuterAlt(_localctx, 2); - setState(617); + setState(613); match(HogQLParser::USING); - setState(618); + setState(614); match(HogQLParser::LPAREN); - setState(619); + setState(615); columnExprList(); - setState(620); + setState(616); match(HogQLParser::RPAREN); break; } case 3: { enterOuterAlt(_localctx, 3); - setState(622); + setState(618); match(HogQLParser::USING); - setState(623); + setState(619); columnExprList(); break; } @@ -5166,18 +5162,18 @@ HogQLParser::SampleClauseContext* HogQLParser::sampleClause() { }); try { enterOuterAlt(_localctx, 1); - setState(626); + setState(622); match(HogQLParser::SAMPLE); - setState(627); + setState(623); ratioExpr(); - setState(630); + setState(626); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 76, _ctx)) { case 1: { - setState(628); + setState(624); match(HogQLParser::OFFSET); - setState(629); + setState(625); ratioExpr(); break; } @@ -5245,17 +5241,17 @@ HogQLParser::OrderExprListContext* HogQLParser::orderExprList() { }); try { enterOuterAlt(_localctx, 1); - setState(632); + setState(628); orderExpr(); - setState(637); + setState(633); _errHandler->sync(this); _la = _input->LA(1); while (_la == HogQLParser::COMMA) { - setState(633); + setState(629); match(HogQLParser::COMMA); - setState(634); + setState(630); orderExpr(); - setState(639); + setState(635); _errHandler->sync(this); _la = _input->LA(1); } @@ -5339,15 +5335,15 @@ HogQLParser::OrderExprContext* HogQLParser::orderExpr() { }); try { enterOuterAlt(_localctx, 1); - setState(640); + setState(636); columnExpr(0); - setState(642); + setState(638); _errHandler->sync(this); _la = _input->LA(1); if ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 12583040) != 0)) { - setState(641); + setState(637); _la = _input->LA(1); if (!((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 12583040) != 0))) { @@ -5358,14 +5354,14 @@ HogQLParser::OrderExprContext* HogQLParser::orderExpr() { consume(); } } - setState(646); + setState(642); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::NULLS) { - setState(644); + setState(640); match(HogQLParser::NULLS); - setState(645); + setState(641); _la = _input->LA(1); if (!(_la == HogQLParser::FIRST @@ -5377,14 +5373,14 @@ HogQLParser::OrderExprContext* HogQLParser::orderExpr() { consume(); } } - setState(650); + setState(646); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COLLATE) { - setState(648); + setState(644); match(HogQLParser::COLLATE); - setState(649); + setState(645); match(HogQLParser::STRING_LITERAL); } @@ -5445,12 +5441,12 @@ HogQLParser::RatioExprContext* HogQLParser::ratioExpr() { exitRule(); }); try { - setState(658); + setState(654); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::LBRACE: { enterOuterAlt(_localctx, 1); - setState(652); + setState(648); placeholder(); break; } @@ -5465,16 +5461,16 @@ HogQLParser::RatioExprContext* HogQLParser::ratioExpr() { case HogQLParser::DOT: case HogQLParser::PLUS: { enterOuterAlt(_localctx, 2); - setState(653); + setState(649); numberLiteral(); - setState(656); + setState(652); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 81, _ctx)) { case 1: { - setState(654); + setState(650); match(HogQLParser::SLASH); - setState(655); + setState(651); numberLiteral(); break; } @@ -5548,17 +5544,17 @@ HogQLParser::SettingExprListContext* HogQLParser::settingExprList() { }); try { enterOuterAlt(_localctx, 1); - setState(660); + setState(656); settingExpr(); - setState(665); + setState(661); _errHandler->sync(this); _la = _input->LA(1); while (_la == HogQLParser::COMMA) { - setState(661); + setState(657); match(HogQLParser::COMMA); - setState(662); + setState(658); settingExpr(); - setState(667); + setState(663); _errHandler->sync(this); _la = _input->LA(1); } @@ -5617,11 +5613,11 @@ HogQLParser::SettingExprContext* HogQLParser::settingExpr() { }); try { enterOuterAlt(_localctx, 1); - setState(668); + setState(664); identifier(); - setState(669); + setState(665); match(HogQLParser::EQ_SINGLE); - setState(670); + setState(666); literal(); } @@ -5679,30 +5675,30 @@ HogQLParser::WindowExprContext* HogQLParser::windowExpr() { }); try { enterOuterAlt(_localctx, 1); - setState(673); + setState(669); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::PARTITION) { - setState(672); + setState(668); winPartitionByClause(); } - setState(676); + setState(672); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::ORDER) { - setState(675); + setState(671); winOrderByClause(); } - setState(679); + setState(675); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::RANGE || _la == HogQLParser::ROWS) { - setState(678); + setState(674); winFrameClause(); } @@ -5760,11 +5756,11 @@ HogQLParser::WinPartitionByClauseContext* HogQLParser::winPartitionByClause() { }); try { enterOuterAlt(_localctx, 1); - setState(681); + setState(677); match(HogQLParser::PARTITION); - setState(682); + setState(678); match(HogQLParser::BY); - setState(683); + setState(679); columnExprList(); } @@ -5821,11 +5817,11 @@ HogQLParser::WinOrderByClauseContext* HogQLParser::winOrderByClause() { }); try { enterOuterAlt(_localctx, 1); - setState(685); + setState(681); match(HogQLParser::ORDER); - setState(686); + setState(682); match(HogQLParser::BY); - setState(687); + setState(683); orderExprList(); } @@ -5883,7 +5879,7 @@ HogQLParser::WinFrameClauseContext* HogQLParser::winFrameClause() { }); try { enterOuterAlt(_localctx, 1); - setState(689); + setState(685); _la = _input->LA(1); if (!(_la == HogQLParser::RANGE @@ -5894,7 +5890,7 @@ HogQLParser::WinFrameClauseContext* HogQLParser::winFrameClause() { _errHandler->reportMatch(this); consume(); } - setState(690); + setState(686); winFrameExtend(); } @@ -5976,7 +5972,7 @@ HogQLParser::WinFrameExtendContext* HogQLParser::winFrameExtend() { exitRule(); }); try { - setState(698); + setState(694); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::CURRENT: @@ -5992,7 +5988,7 @@ HogQLParser::WinFrameExtendContext* HogQLParser::winFrameExtend() { case HogQLParser::PLUS: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 1); - setState(692); + setState(688); winFrameBound(); break; } @@ -6000,13 +5996,13 @@ HogQLParser::WinFrameExtendContext* HogQLParser::winFrameExtend() { case HogQLParser::BETWEEN: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 2); - setState(693); + setState(689); match(HogQLParser::BETWEEN); - setState(694); + setState(690); winFrameBound(); - setState(695); + setState(691); match(HogQLParser::AND); - setState(696); + setState(692); winFrameBound(); break; } @@ -6081,45 +6077,45 @@ HogQLParser::WinFrameBoundContext* HogQLParser::winFrameBound() { }); try { enterOuterAlt(_localctx, 1); - setState(712); + setState(708); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 88, _ctx)) { case 1: { - setState(700); + setState(696); match(HogQLParser::CURRENT); - setState(701); + setState(697); match(HogQLParser::ROW); break; } case 2: { - setState(702); + setState(698); match(HogQLParser::UNBOUNDED); - setState(703); + setState(699); match(HogQLParser::PRECEDING); break; } case 3: { - setState(704); + setState(700); match(HogQLParser::UNBOUNDED); - setState(705); + setState(701); match(HogQLParser::FOLLOWING); break; } case 4: { - setState(706); + setState(702); numberLiteral(); - setState(707); + setState(703); match(HogQLParser::PRECEDING); break; } case 5: { - setState(709); + setState(705); numberLiteral(); - setState(710); + setState(706); match(HogQLParser::FOLLOWING); break; } @@ -6178,9 +6174,9 @@ HogQLParser::ExprContext* HogQLParser::expr() { }); try { enterOuterAlt(_localctx, 1); - setState(714); + setState(710); columnExpr(0); - setState(715); + setState(711); match(HogQLParser::EOF); } @@ -6385,13 +6381,13 @@ HogQLParser::ColumnTypeExprContext* HogQLParser::columnTypeExpr() { }); try { size_t alt; - setState(773); + setState(769); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 96, _ctx)) { case 1: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 1); - setState(717); + setState(713); identifier(); break; } @@ -6399,39 +6395,39 @@ HogQLParser::ColumnTypeExprContext* HogQLParser::columnTypeExpr() { case 2: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 2); - setState(718); + setState(714); identifier(); - setState(719); + setState(715); match(HogQLParser::LPAREN); - setState(720); + setState(716); identifier(); - setState(721); + setState(717); columnTypeExpr(); - setState(728); + setState(724); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 89, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { - setState(722); + setState(718); match(HogQLParser::COMMA); - setState(723); + setState(719); identifier(); - setState(724); + setState(720); columnTypeExpr(); } - setState(730); + setState(726); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 89, _ctx); } - setState(732); + setState(728); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COMMA) { - setState(731); + setState(727); match(HogQLParser::COMMA); } - setState(734); + setState(730); match(HogQLParser::RPAREN); break; } @@ -6439,35 +6435,35 @@ HogQLParser::ColumnTypeExprContext* HogQLParser::columnTypeExpr() { case 3: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 3); - setState(736); + setState(732); identifier(); - setState(737); + setState(733); match(HogQLParser::LPAREN); - setState(738); + setState(734); enumValue(); - setState(743); + setState(739); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 91, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { - setState(739); + setState(735); match(HogQLParser::COMMA); - setState(740); + setState(736); enumValue(); } - setState(745); + setState(741); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 91, _ctx); } - setState(747); + setState(743); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COMMA) { - setState(746); + setState(742); match(HogQLParser::COMMA); } - setState(749); + setState(745); match(HogQLParser::RPAREN); break; } @@ -6475,35 +6471,35 @@ HogQLParser::ColumnTypeExprContext* HogQLParser::columnTypeExpr() { case 4: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 4); - setState(751); + setState(747); identifier(); - setState(752); + setState(748); match(HogQLParser::LPAREN); - setState(753); + setState(749); columnTypeExpr(); - setState(758); + setState(754); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 93, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { - setState(754); + setState(750); match(HogQLParser::COMMA); - setState(755); + setState(751); columnTypeExpr(); } - setState(760); + setState(756); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 93, _ctx); } - setState(762); + setState(758); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COMMA) { - setState(761); + setState(757); match(HogQLParser::COMMA); } - setState(764); + setState(760); match(HogQLParser::RPAREN); break; } @@ -6511,11 +6507,11 @@ HogQLParser::ColumnTypeExprContext* HogQLParser::columnTypeExpr() { case 5: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 5); - setState(766); + setState(762); identifier(); - setState(767); + setState(763); match(HogQLParser::LPAREN); - setState(769); + setState(765); _errHandler->sync(this); _la = _input->LA(1); @@ -6523,10 +6519,10 @@ HogQLParser::ColumnTypeExprContext* HogQLParser::columnTypeExpr() { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(768); + setState(764); columnExprList(); } - setState(771); + setState(767); match(HogQLParser::RPAREN); break; } @@ -6594,28 +6590,28 @@ HogQLParser::ColumnExprListContext* HogQLParser::columnExprList() { try { size_t alt; enterOuterAlt(_localctx, 1); - setState(775); + setState(771); columnExpr(0); - setState(780); + setState(776); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 97, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { - setState(776); + setState(772); match(HogQLParser::COMMA); - setState(777); + setState(773); columnExpr(0); } - setState(782); + setState(778); _errHandler->sync(this); alt = getInterpreter()->adaptivePredict(_input, 97, _ctx); } - setState(784); + setState(780); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 98, _ctx)) { case 1: { - setState(783); + setState(779); match(HogQLParser::COMMA); break; } @@ -7159,16 +7155,16 @@ tree::TerminalNode* HogQLParser::ColumnExprWinFunctionTargetContext::RPAREN(size return getToken(HogQLParser::RPAREN, i); } -HogQLParser::ColumnExprListContext* HogQLParser::ColumnExprWinFunctionTargetContext::columnExprList() { - return getRuleContext(0); +std::vector HogQLParser::ColumnExprWinFunctionTargetContext::columnExprList() { + return getRuleContexts(); } -tree::TerminalNode* HogQLParser::ColumnExprWinFunctionTargetContext::DISTINCT() { - return getToken(HogQLParser::DISTINCT, 0); +HogQLParser::ColumnExprListContext* HogQLParser::ColumnExprWinFunctionTargetContext::columnExprList(size_t i) { + return getRuleContext(i); } -HogQLParser::ColumnArgListContext* HogQLParser::ColumnExprWinFunctionTargetContext::columnArgList() { - return getRuleContext(0); +tree::TerminalNode* HogQLParser::ColumnExprWinFunctionTargetContext::DISTINCT() { + return getToken(HogQLParser::DISTINCT, 0); } HogQLParser::ColumnExprWinFunctionTargetContext::ColumnExprWinFunctionTargetContext(ColumnExprContext *ctx) { copyFrom(ctx); } @@ -7303,6 +7299,33 @@ std::any HogQLParser::ColumnExprTupleContext::accept(tree::ParseTreeVisitor *vis else return visitor->visitChildren(this); } +//----------------- ColumnExprCallContext ------------------------------------------------------------------ + +HogQLParser::ColumnExprContext* HogQLParser::ColumnExprCallContext::columnExpr() { + return getRuleContext(0); +} + +tree::TerminalNode* HogQLParser::ColumnExprCallContext::LPAREN() { + return getToken(HogQLParser::LPAREN, 0); +} + +tree::TerminalNode* HogQLParser::ColumnExprCallContext::RPAREN() { + return getToken(HogQLParser::RPAREN, 0); +} + +HogQLParser::ColumnExprListContext* HogQLParser::ColumnExprCallContext::columnExprList() { + return getRuleContext(0); +} + +HogQLParser::ColumnExprCallContext::ColumnExprCallContext(ColumnExprContext *ctx) { copyFrom(ctx); } + + +std::any HogQLParser::ColumnExprCallContext::accept(tree::ParseTreeVisitor *visitor) { + if (auto parserVisitor = dynamic_cast(visitor)) + return parserVisitor->visitColumnExprCall(this); + else + return visitor->visitChildren(this); +} //----------------- ColumnExprArrayAccessContext ------------------------------------------------------------------ std::vector HogQLParser::ColumnExprArrayAccessContext::columnExpr() { @@ -7641,16 +7664,16 @@ tree::TerminalNode* HogQLParser::ColumnExprWinFunctionContext::RPAREN(size_t i) return getToken(HogQLParser::RPAREN, i); } -HogQLParser::ColumnExprListContext* HogQLParser::ColumnExprWinFunctionContext::columnExprList() { - return getRuleContext(0); +std::vector HogQLParser::ColumnExprWinFunctionContext::columnExprList() { + return getRuleContexts(); } -tree::TerminalNode* HogQLParser::ColumnExprWinFunctionContext::DISTINCT() { - return getToken(HogQLParser::DISTINCT, 0); +HogQLParser::ColumnExprListContext* HogQLParser::ColumnExprWinFunctionContext::columnExprList(size_t i) { + return getRuleContext(i); } -HogQLParser::ColumnArgListContext* HogQLParser::ColumnExprWinFunctionContext::columnArgList() { - return getRuleContext(0); +tree::TerminalNode* HogQLParser::ColumnExprWinFunctionContext::DISTINCT() { + return getToken(HogQLParser::DISTINCT, 0); } HogQLParser::ColumnExprWinFunctionContext::ColumnExprWinFunctionContext(ColumnExprContext *ctx) { copyFrom(ctx); } @@ -7662,6 +7685,21 @@ std::any HogQLParser::ColumnExprWinFunctionContext::accept(tree::ParseTreeVisito else return visitor->visitChildren(this); } +//----------------- ColumnExprLambdaContext ------------------------------------------------------------------ + +HogQLParser::ColumnLambdaExprContext* HogQLParser::ColumnExprLambdaContext::columnLambdaExpr() { + return getRuleContext(0); +} + +HogQLParser::ColumnExprLambdaContext::ColumnExprLambdaContext(ColumnExprContext *ctx) { copyFrom(ctx); } + + +std::any HogQLParser::ColumnExprLambdaContext::accept(tree::ParseTreeVisitor *visitor) { + if (auto parserVisitor = dynamic_cast(visitor)) + return parserVisitor->visitColumnExprLambda(this); + else + return visitor->visitChildren(this); +} //----------------- ColumnExprIdentifierContext ------------------------------------------------------------------ HogQLParser::ColumnIdentifierContext* HogQLParser::ColumnExprIdentifierContext::columnIdentifier() { @@ -7703,12 +7741,12 @@ tree::TerminalNode* HogQLParser::ColumnExprFunctionContext::DISTINCT() { return getToken(HogQLParser::DISTINCT, 0); } -HogQLParser::ColumnArgListContext* HogQLParser::ColumnExprFunctionContext::columnArgList() { - return getRuleContext(0); +std::vector HogQLParser::ColumnExprFunctionContext::columnExprList() { + return getRuleContexts(); } -HogQLParser::ColumnExprListContext* HogQLParser::ColumnExprFunctionContext::columnExprList() { - return getRuleContext(0); +HogQLParser::ColumnExprListContext* HogQLParser::ColumnExprFunctionContext::columnExprList(size_t i) { + return getRuleContext(i); } HogQLParser::ColumnExprFunctionContext::ColumnExprFunctionContext(ColumnExprContext *ctx) { copyFrom(ctx); } @@ -7769,7 +7807,7 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { try { size_t alt; enterOuterAlt(_localctx, 1); - setState(935); + setState(932); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 118, _ctx)) { case 1: { @@ -7777,14 +7815,14 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _ctx = _localctx; previousContext = _localctx; - setState(787); + setState(783); match(HogQLParser::CASE); - setState(789); + setState(785); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 99, _ctx)) { case 1: { - setState(788); + setState(784); antlrcpp::downCast(_localctx)->caseExpr = columnExpr(0); break; } @@ -7792,33 +7830,33 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { default: break; } - setState(796); + setState(792); _errHandler->sync(this); _la = _input->LA(1); do { - setState(791); + setState(787); match(HogQLParser::WHEN); - setState(792); + setState(788); antlrcpp::downCast(_localctx)->whenExpr = columnExpr(0); - setState(793); + setState(789); match(HogQLParser::THEN); - setState(794); + setState(790); antlrcpp::downCast(_localctx)->thenExpr = columnExpr(0); - setState(798); + setState(794); _errHandler->sync(this); _la = _input->LA(1); } while (_la == HogQLParser::WHEN); - setState(802); + setState(798); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::ELSE) { - setState(800); + setState(796); match(HogQLParser::ELSE); - setState(801); + setState(797); antlrcpp::downCast(_localctx)->elseExpr = columnExpr(0); } - setState(804); + setState(800); match(HogQLParser::END); break; } @@ -7827,17 +7865,17 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(806); + setState(802); match(HogQLParser::CAST); - setState(807); + setState(803); match(HogQLParser::LPAREN); - setState(808); + setState(804); columnExpr(0); - setState(809); + setState(805); match(HogQLParser::AS); - setState(810); + setState(806); columnTypeExpr(); - setState(811); + setState(807); match(HogQLParser::RPAREN); break; } @@ -7846,9 +7884,9 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(813); + setState(809); match(HogQLParser::DATE); - setState(814); + setState(810); match(HogQLParser::STRING_LITERAL); break; } @@ -7857,11 +7895,11 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(815); + setState(811); match(HogQLParser::INTERVAL); - setState(816); + setState(812); columnExpr(0); - setState(817); + setState(813); interval(); break; } @@ -7870,27 +7908,27 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(819); + setState(815); match(HogQLParser::SUBSTRING); - setState(820); + setState(816); match(HogQLParser::LPAREN); - setState(821); + setState(817); columnExpr(0); - setState(822); + setState(818); match(HogQLParser::FROM); - setState(823); + setState(819); columnExpr(0); - setState(826); + setState(822); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::FOR) { - setState(824); + setState(820); match(HogQLParser::FOR); - setState(825); + setState(821); columnExpr(0); } - setState(828); + setState(824); match(HogQLParser::RPAREN); break; } @@ -7899,9 +7937,9 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(830); + setState(826); match(HogQLParser::TIMESTAMP); - setState(831); + setState(827); match(HogQLParser::STRING_LITERAL); break; } @@ -7910,11 +7948,11 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(832); + setState(828); match(HogQLParser::TRIM); - setState(833); + setState(829); match(HogQLParser::LPAREN); - setState(834); + setState(830); _la = _input->LA(1); if (!(_la == HogQLParser::BOTH @@ -7925,13 +7963,13 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _errHandler->reportMatch(this); consume(); } - setState(835); + setState(831); string(); - setState(836); + setState(832); match(HogQLParser::FROM); - setState(837); + setState(833); columnExpr(0); - setState(838); + setState(834); match(HogQLParser::RPAREN); break; } @@ -7940,12 +7978,12 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(840); + setState(836); identifier(); - setState(841); + setState(837); match(HogQLParser::LPAREN); - setState(843); + setState(839); _errHandler->sync(this); _la = _input->LA(1); @@ -7953,24 +7991,24 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(842); - columnExprList(); + setState(838); + antlrcpp::downCast(_localctx)->columnExprs = columnExprList(); } - setState(845); + setState(841); match(HogQLParser::RPAREN); - setState(855); + setState(851); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::LPAREN) { - setState(847); + setState(843); match(HogQLParser::LPAREN); - setState(849); + setState(845); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 104, _ctx)) { case 1: { - setState(848); + setState(844); match(HogQLParser::DISTINCT); break; } @@ -7978,7 +8016,7 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { default: break; } - setState(852); + setState(848); _errHandler->sync(this); _la = _input->LA(1); @@ -7986,19 +8024,19 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(851); - columnArgList(); + setState(847); + antlrcpp::downCast(_localctx)->columnArgList = columnExprList(); } - setState(854); + setState(850); match(HogQLParser::RPAREN); } - setState(857); + setState(853); match(HogQLParser::OVER); - setState(858); + setState(854); match(HogQLParser::LPAREN); - setState(859); + setState(855); windowExpr(); - setState(860); + setState(856); match(HogQLParser::RPAREN); break; } @@ -8007,12 +8045,12 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(862); + setState(858); identifier(); - setState(863); + setState(859); match(HogQLParser::LPAREN); - setState(865); + setState(861); _errHandler->sync(this); _la = _input->LA(1); @@ -8020,24 +8058,24 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(864); - columnExprList(); + setState(860); + antlrcpp::downCast(_localctx)->columnExprs = columnExprList(); } - setState(867); + setState(863); match(HogQLParser::RPAREN); - setState(877); + setState(873); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::LPAREN) { - setState(869); + setState(865); match(HogQLParser::LPAREN); - setState(871); + setState(867); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 108, _ctx)) { case 1: { - setState(870); + setState(866); match(HogQLParser::DISTINCT); break; } @@ -8045,7 +8083,7 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { default: break; } - setState(874); + setState(870); _errHandler->sync(this); _la = _input->LA(1); @@ -8053,15 +8091,15 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(873); - columnArgList(); + setState(869); + antlrcpp::downCast(_localctx)->columnArgList = columnExprList(); } - setState(876); + setState(872); match(HogQLParser::RPAREN); } - setState(879); + setState(875); match(HogQLParser::OVER); - setState(880); + setState(876); identifier(); break; } @@ -8070,16 +8108,16 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(882); + setState(878); identifier(); - setState(888); + setState(884); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 112, _ctx)) { case 1: { - setState(883); + setState(879); match(HogQLParser::LPAREN); - setState(885); + setState(881); _errHandler->sync(this); _la = _input->LA(1); @@ -8087,10 +8125,10 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(884); - columnExprList(); + setState(880); + antlrcpp::downCast(_localctx)->columnExprs = columnExprList(); } - setState(887); + setState(883); match(HogQLParser::RPAREN); break; } @@ -8098,14 +8136,14 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { default: break; } - setState(890); + setState(886); match(HogQLParser::LPAREN); - setState(892); + setState(888); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 113, _ctx)) { case 1: { - setState(891); + setState(887); match(HogQLParser::DISTINCT); break; } @@ -8113,7 +8151,7 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { default: break; } - setState(895); + setState(891); _errHandler->sync(this); _la = _input->LA(1); @@ -8121,10 +8159,10 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(894); - columnArgList(); + setState(890); + antlrcpp::downCast(_localctx)->columnArgList = columnExprList(); } - setState(897); + setState(893); match(HogQLParser::RPAREN); break; } @@ -8133,7 +8171,7 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(899); + setState(895); hogqlxTagElement(); break; } @@ -8142,7 +8180,7 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(900); + setState(896); templateString(); break; } @@ -8151,7 +8189,7 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(901); + setState(897); literal(); break; } @@ -8160,10 +8198,10 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(902); + setState(898); match(HogQLParser::DASH); - setState(903); - columnExpr(19); + setState(899); + columnExpr(20); break; } @@ -8171,10 +8209,10 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(904); + setState(900); match(HogQLParser::NOT); - setState(905); - columnExpr(13); + setState(901); + columnExpr(14); break; } @@ -8182,19 +8220,19 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(909); + setState(905); _errHandler->sync(this); _la = _input->LA(1); if ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & -725088338784043010) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 3229277487103) != 0)) { - setState(906); + setState(902); tableIdentifier(); - setState(907); + setState(903); match(HogQLParser::DOT); } - setState(911); + setState(907); match(HogQLParser::ASTERISK); break; } @@ -8203,11 +8241,11 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(912); + setState(908); match(HogQLParser::LPAREN); - setState(913); + setState(909); selectUnionStmt(); - setState(914); + setState(910); match(HogQLParser::RPAREN); break; } @@ -8216,11 +8254,11 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(916); + setState(912); match(HogQLParser::LPAREN); - setState(917); + setState(913); columnExpr(0); - setState(918); + setState(914); match(HogQLParser::RPAREN); break; } @@ -8229,11 +8267,11 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(920); + setState(916); match(HogQLParser::LPAREN); - setState(921); + setState(917); columnExprList(); - setState(922); + setState(918); match(HogQLParser::RPAREN); break; } @@ -8242,9 +8280,9 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(924); + setState(920); match(HogQLParser::LBRACKET); - setState(926); + setState(922); _errHandler->sync(this); _la = _input->LA(1); @@ -8252,10 +8290,10 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(925); + setState(921); columnExprList(); } - setState(928); + setState(924); match(HogQLParser::RBRACKET); break; } @@ -8264,9 +8302,9 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(929); + setState(925); match(HogQLParser::LBRACE); - setState(931); + setState(927); _errHandler->sync(this); _la = _input->LA(1); @@ -8274,19 +8312,28 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(930); + setState(926); kvPairList(); } - setState(933); + setState(929); match(HogQLParser::RBRACE); break; } case 22: { + _localctx = _tracker.createInstance(_localctx); + _ctx = _localctx; + previousContext = _localctx; + setState(930); + columnLambdaExpr(); + break; + } + + case 23: { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(934); + setState(931); columnIdentifier(); break; } @@ -8295,42 +8342,42 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { break; } _ctx->stop = _input->LT(-1); - setState(1041); + setState(1044); _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 129, _ctx); + alt = getInterpreter()->adaptivePredict(_input, 130, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { if (!_parseListeners.empty()) triggerExitRuleEvent(); previousContext = _localctx; - setState(1039); + setState(1042); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 128, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 129, _ctx)) { case 1: { auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; newContext->left = previousContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(937); + setState(934); - if (!(precpred(_ctx, 18))) throw FailedPredicateException(this, "precpred(_ctx, 18)"); - setState(941); + if (!(precpred(_ctx, 19))) throw FailedPredicateException(this, "precpred(_ctx, 19)"); + setState(938); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::ASTERISK: { - setState(938); + setState(935); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::ASTERISK); break; } case HogQLParser::SLASH: { - setState(939); + setState(936); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::SLASH); break; } case HogQLParser::PERCENT: { - setState(940); + setState(937); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::PERCENT); break; } @@ -8338,8 +8385,8 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { default: throw NoViableAltException(this); } - setState(943); - antlrcpp::downCast(_localctx)->right = columnExpr(19); + setState(940); + antlrcpp::downCast(_localctx)->right = columnExpr(20); break; } @@ -8348,26 +8395,26 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = newContext; newContext->left = previousContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(944); + setState(941); - if (!(precpred(_ctx, 17))) throw FailedPredicateException(this, "precpred(_ctx, 17)"); - setState(948); + if (!(precpred(_ctx, 18))) throw FailedPredicateException(this, "precpred(_ctx, 18)"); + setState(945); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::PLUS: { - setState(945); + setState(942); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::PLUS); break; } case HogQLParser::DASH: { - setState(946); + setState(943); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::DASH); break; } case HogQLParser::CONCAT: { - setState(947); + setState(944); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::CONCAT); break; } @@ -8375,8 +8422,8 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { default: throw NoViableAltException(this); } - setState(950); - antlrcpp::downCast(_localctx)->right = columnExpr(18); + setState(947); + antlrcpp::downCast(_localctx)->right = columnExpr(19); break; } @@ -8385,71 +8432,71 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { _localctx = newContext; newContext->left = previousContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(951); + setState(948); - if (!(precpred(_ctx, 16))) throw FailedPredicateException(this, "precpred(_ctx, 16)"); - setState(976); + if (!(precpred(_ctx, 17))) throw FailedPredicateException(this, "precpred(_ctx, 17)"); + setState(973); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 124, _ctx)) { case 1: { - setState(952); + setState(949); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::EQ_DOUBLE); break; } case 2: { - setState(953); + setState(950); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::EQ_SINGLE); break; } case 3: { - setState(954); + setState(951); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::NOT_EQ); break; } case 4: { - setState(955); + setState(952); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::LT_EQ); break; } case 5: { - setState(956); + setState(953); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::LT); break; } case 6: { - setState(957); + setState(954); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::GT_EQ); break; } case 7: { - setState(958); + setState(955); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::GT); break; } case 8: { - setState(960); + setState(957); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::NOT) { - setState(959); + setState(956); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::NOT); } - setState(962); + setState(959); match(HogQLParser::IN); - setState(964); + setState(961); _errHandler->sync(this); switch (getInterpreter()->adaptivePredict(_input, 122, _ctx)) { case 1: { - setState(963); + setState(960); match(HogQLParser::COHORT); break; } @@ -8461,15 +8508,15 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { } case 9: { - setState(967); + setState(964); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::NOT) { - setState(966); + setState(963); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::NOT); } - setState(969); + setState(966); _la = _input->LA(1); if (!(_la == HogQLParser::ILIKE @@ -8484,37 +8531,37 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { } case 10: { - setState(970); + setState(967); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::REGEX_SINGLE); break; } case 11: { - setState(971); + setState(968); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::REGEX_DOUBLE); break; } case 12: { - setState(972); + setState(969); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::NOT_REGEX); break; } case 13: { - setState(973); + setState(970); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::IREGEX_SINGLE); break; } case 14: { - setState(974); + setState(971); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::IREGEX_DOUBLE); break; } case 15: { - setState(975); + setState(972); antlrcpp::downCast(_localctx)->operator_ = match(HogQLParser::NOT_IREGEX); break; } @@ -8522,8 +8569,8 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { default: break; } - setState(978); - antlrcpp::downCast(_localctx)->right = columnExpr(17); + setState(975); + antlrcpp::downCast(_localctx)->right = columnExpr(18); break; } @@ -8531,13 +8578,13 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(979); + setState(976); - if (!(precpred(_ctx, 14))) throw FailedPredicateException(this, "precpred(_ctx, 14)"); - setState(980); + if (!(precpred(_ctx, 15))) throw FailedPredicateException(this, "precpred(_ctx, 15)"); + setState(977); match(HogQLParser::NULLISH); - setState(981); - columnExpr(15); + setState(978); + columnExpr(16); break; } @@ -8545,13 +8592,13 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(982); + setState(979); - if (!(precpred(_ctx, 12))) throw FailedPredicateException(this, "precpred(_ctx, 12)"); - setState(983); + if (!(precpred(_ctx, 13))) throw FailedPredicateException(this, "precpred(_ctx, 13)"); + setState(980); match(HogQLParser::AND); - setState(984); - columnExpr(13); + setState(981); + columnExpr(14); break; } @@ -8559,13 +8606,13 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(985); + setState(982); - if (!(precpred(_ctx, 11))) throw FailedPredicateException(this, "precpred(_ctx, 11)"); - setState(986); + if (!(precpred(_ctx, 12))) throw FailedPredicateException(this, "precpred(_ctx, 12)"); + setState(983); match(HogQLParser::OR); - setState(987); - columnExpr(12); + setState(984); + columnExpr(13); break; } @@ -8573,25 +8620,25 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(988); + setState(985); - if (!(precpred(_ctx, 10))) throw FailedPredicateException(this, "precpred(_ctx, 10)"); - setState(990); + if (!(precpred(_ctx, 11))) throw FailedPredicateException(this, "precpred(_ctx, 11)"); + setState(987); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::NOT) { - setState(989); + setState(986); match(HogQLParser::NOT); } - setState(992); + setState(989); match(HogQLParser::BETWEEN); - setState(993); + setState(990); columnExpr(0); - setState(994); + setState(991); match(HogQLParser::AND); - setState(995); - columnExpr(11); + setState(992); + columnExpr(12); break; } @@ -8599,333 +8646,203 @@ HogQLParser::ColumnExprContext* HogQLParser::columnExpr(int precedence) { auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(997); + setState(994); - if (!(precpred(_ctx, 9))) throw FailedPredicateException(this, "precpred(_ctx, 9)"); - setState(998); + if (!(precpred(_ctx, 10))) throw FailedPredicateException(this, "precpred(_ctx, 10)"); + setState(995); match(HogQLParser::QUERY); - setState(999); + setState(996); columnExpr(0); - setState(1000); + setState(997); match(HogQLParser::COLON); - setState(1001); - columnExpr(9); + setState(998); + columnExpr(10); break; } case 9: { - auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); + auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); + setState(1000); + + if (!(precpred(_ctx, 30))) throw FailedPredicateException(this, "precpred(_ctx, 30)"); + setState(1001); + match(HogQLParser::LPAREN); setState(1003); + _errHandler->sync(this); - if (!(precpred(_ctx, 25))) throw FailedPredicateException(this, "precpred(_ctx, 25)"); - setState(1004); - match(HogQLParser::LBRACKET); + _la = _input->LA(1); + if ((((_la & ~ 0x3fULL) == 0) && + ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && + ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && + ((1ULL << (_la - 128)) & 18455) != 0)) { + setState(1002); + columnExprList(); + } setState(1005); - columnExpr(0); - setState(1006); - match(HogQLParser::RBRACKET); + match(HogQLParser::RPAREN); break; } case 10: { - auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); + auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(1008); + setState(1006); - if (!(precpred(_ctx, 24))) throw FailedPredicateException(this, "precpred(_ctx, 24)"); + if (!(precpred(_ctx, 26))) throw FailedPredicateException(this, "precpred(_ctx, 26)"); + setState(1007); + match(HogQLParser::LBRACKET); + setState(1008); + columnExpr(0); setState(1009); - match(HogQLParser::DOT); - setState(1010); - match(HogQLParser::DECIMAL_LITERAL); + match(HogQLParser::RBRACKET); break; } case 11: { - auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); + auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); setState(1011); - if (!(precpred(_ctx, 23))) throw FailedPredicateException(this, "precpred(_ctx, 23)"); + if (!(precpred(_ctx, 25))) throw FailedPredicateException(this, "precpred(_ctx, 25)"); setState(1012); match(HogQLParser::DOT); setState(1013); - identifier(); + match(HogQLParser::DECIMAL_LITERAL); break; } case 12: { - auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); + auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); setState(1014); - if (!(precpred(_ctx, 22))) throw FailedPredicateException(this, "precpred(_ctx, 22)"); + if (!(precpred(_ctx, 24))) throw FailedPredicateException(this, "precpred(_ctx, 24)"); setState(1015); - match(HogQLParser::NULL_PROPERTY); + match(HogQLParser::DOT); setState(1016); - match(HogQLParser::LBRACKET); - setState(1017); - columnExpr(0); - setState(1018); - match(HogQLParser::RBRACKET); + identifier(); break; } case 13: { - auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); + auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(1020); + setState(1017); - if (!(precpred(_ctx, 21))) throw FailedPredicateException(this, "precpred(_ctx, 21)"); - setState(1021); + if (!(precpred(_ctx, 23))) throw FailedPredicateException(this, "precpred(_ctx, 23)"); + setState(1018); match(HogQLParser::NULL_PROPERTY); - setState(1022); - match(HogQLParser::DECIMAL_LITERAL); + setState(1019); + match(HogQLParser::LBRACKET); + setState(1020); + columnExpr(0); + setState(1021); + match(HogQLParser::RBRACKET); break; } case 14: { - auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); + auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); setState(1023); - if (!(precpred(_ctx, 20))) throw FailedPredicateException(this, "precpred(_ctx, 20)"); + if (!(precpred(_ctx, 22))) throw FailedPredicateException(this, "precpred(_ctx, 22)"); setState(1024); match(HogQLParser::NULL_PROPERTY); setState(1025); - identifier(); + match(HogQLParser::DECIMAL_LITERAL); break; } case 15: { - auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); + auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); setState(1026); - if (!(precpred(_ctx, 15))) throw FailedPredicateException(this, "precpred(_ctx, 15)"); + if (!(precpred(_ctx, 21))) throw FailedPredicateException(this, "precpred(_ctx, 21)"); setState(1027); - match(HogQLParser::IS); + match(HogQLParser::NULL_PROPERTY); + setState(1028); + identifier(); + break; + } + + case 16: { + auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); + _localctx = newContext; + pushNewRecursionContext(newContext, startState, RuleColumnExpr); setState(1029); + + if (!(precpred(_ctx, 16))) throw FailedPredicateException(this, "precpred(_ctx, 16)"); + setState(1030); + match(HogQLParser::IS); + setState(1032); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::NOT) { - setState(1028); + setState(1031); match(HogQLParser::NOT); } - setState(1031); + setState(1034); match(HogQLParser::NULL_SQL); break; } - case 16: { + case 17: { auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleColumnExpr); - setState(1032); + setState(1035); - if (!(precpred(_ctx, 8))) throw FailedPredicateException(this, "precpred(_ctx, 8)"); - setState(1037); + if (!(precpred(_ctx, 9))) throw FailedPredicateException(this, "precpred(_ctx, 9)"); + setState(1040); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 127, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 128, _ctx)) { case 1: { - setState(1033); - match(HogQLParser::AS); - setState(1034); - identifier(); - break; - } - - case 2: { - setState(1035); - match(HogQLParser::AS); setState(1036); - match(HogQLParser::STRING_LITERAL); - break; - } - - default: - break; - } - break; - } - - default: - break; - } - } - setState(1043); - _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 129, _ctx); - } - } - catch (RecognitionException &e) { - _errHandler->reportError(this, e); - _localctx->exception = std::current_exception(); - _errHandler->recover(this, _localctx->exception); - } - return _localctx; -} - -//----------------- ColumnArgListContext ------------------------------------------------------------------ - -HogQLParser::ColumnArgListContext::ColumnArgListContext(ParserRuleContext *parent, size_t invokingState) - : ParserRuleContext(parent, invokingState) { -} - -std::vector HogQLParser::ColumnArgListContext::columnArgExpr() { - return getRuleContexts(); -} - -HogQLParser::ColumnArgExprContext* HogQLParser::ColumnArgListContext::columnArgExpr(size_t i) { - return getRuleContext(i); -} - -std::vector HogQLParser::ColumnArgListContext::COMMA() { - return getTokens(HogQLParser::COMMA); -} - -tree::TerminalNode* HogQLParser::ColumnArgListContext::COMMA(size_t i) { - return getToken(HogQLParser::COMMA, i); -} - - -size_t HogQLParser::ColumnArgListContext::getRuleIndex() const { - return HogQLParser::RuleColumnArgList; -} - - -std::any HogQLParser::ColumnArgListContext::accept(tree::ParseTreeVisitor *visitor) { - if (auto parserVisitor = dynamic_cast(visitor)) - return parserVisitor->visitColumnArgList(this); - else - return visitor->visitChildren(this); -} - -HogQLParser::ColumnArgListContext* HogQLParser::columnArgList() { - ColumnArgListContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 118, HogQLParser::RuleColumnArgList); - size_t _la = 0; - -#if __cplusplus > 201703L - auto onExit = finally([=, this] { -#else - auto onExit = finally([=] { -#endif - exitRule(); - }); - try { - size_t alt; - enterOuterAlt(_localctx, 1); - setState(1044); - columnArgExpr(); - setState(1049); - _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 130, _ctx); - while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { - if (alt == 1) { - setState(1045); - match(HogQLParser::COMMA); - setState(1046); - columnArgExpr(); - } - setState(1051); - _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 130, _ctx); - } - setState(1053); - _errHandler->sync(this); - - _la = _input->LA(1); - if (_la == HogQLParser::COMMA) { - setState(1052); - match(HogQLParser::COMMA); - } - - } - catch (RecognitionException &e) { - _errHandler->reportError(this, e); - _localctx->exception = std::current_exception(); - _errHandler->recover(this, _localctx->exception); - } - - return _localctx; -} - -//----------------- ColumnArgExprContext ------------------------------------------------------------------ - -HogQLParser::ColumnArgExprContext::ColumnArgExprContext(ParserRuleContext *parent, size_t invokingState) - : ParserRuleContext(parent, invokingState) { -} - -HogQLParser::ColumnLambdaExprContext* HogQLParser::ColumnArgExprContext::columnLambdaExpr() { - return getRuleContext(0); -} - -HogQLParser::ColumnExprContext* HogQLParser::ColumnArgExprContext::columnExpr() { - return getRuleContext(0); -} - - -size_t HogQLParser::ColumnArgExprContext::getRuleIndex() const { - return HogQLParser::RuleColumnArgExpr; -} - - -std::any HogQLParser::ColumnArgExprContext::accept(tree::ParseTreeVisitor *visitor) { - if (auto parserVisitor = dynamic_cast(visitor)) - return parserVisitor->visitColumnArgExpr(this); - else - return visitor->visitChildren(this); -} - -HogQLParser::ColumnArgExprContext* HogQLParser::columnArgExpr() { - ColumnArgExprContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 120, HogQLParser::RuleColumnArgExpr); - -#if __cplusplus > 201703L - auto onExit = finally([=, this] { -#else - auto onExit = finally([=] { -#endif - exitRule(); - }); - try { - setState(1057); - _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 132, _ctx)) { - case 1: { - enterOuterAlt(_localctx, 1); - setState(1055); - columnLambdaExpr(); - break; - } + match(HogQLParser::AS); + setState(1037); + identifier(); + break; + } - case 2: { - enterOuterAlt(_localctx, 2); - setState(1056); - columnExpr(0); - break; - } + case 2: { + setState(1038); + match(HogQLParser::AS); + setState(1039); + match(HogQLParser::STRING_LITERAL); + break; + } - default: - break; + default: + break; + } + break; + } + + default: + break; + } + } + setState(1046); + _errHandler->sync(this); + alt = getInterpreter()->adaptivePredict(_input, 130, _ctx); } - } catch (RecognitionException &e) { _errHandler->reportError(this, e); _localctx->exception = std::current_exception(); _errHandler->recover(this, _localctx->exception); } - return _localctx; } @@ -8939,10 +8856,6 @@ tree::TerminalNode* HogQLParser::ColumnLambdaExprContext::ARROW() { return getToken(HogQLParser::ARROW, 0); } -HogQLParser::ColumnExprContext* HogQLParser::ColumnLambdaExprContext::columnExpr() { - return getRuleContext(0); -} - tree::TerminalNode* HogQLParser::ColumnLambdaExprContext::LPAREN() { return getToken(HogQLParser::LPAREN, 0); } @@ -8959,6 +8872,14 @@ tree::TerminalNode* HogQLParser::ColumnLambdaExprContext::RPAREN() { return getToken(HogQLParser::RPAREN, 0); } +HogQLParser::ColumnExprContext* HogQLParser::ColumnLambdaExprContext::columnExpr() { + return getRuleContext(0); +} + +HogQLParser::BlockContext* HogQLParser::ColumnLambdaExprContext::block() { + return getRuleContext(0); +} + std::vector HogQLParser::ColumnLambdaExprContext::COMMA() { return getTokens(HogQLParser::COMMA); } @@ -8982,7 +8903,7 @@ std::any HogQLParser::ColumnLambdaExprContext::accept(tree::ParseTreeVisitor *vi HogQLParser::ColumnLambdaExprContext* HogQLParser::columnLambdaExpr() { ColumnLambdaExprContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 122, HogQLParser::RuleColumnLambdaExpr); + enterRule(_localctx, 118, HogQLParser::RuleColumnLambdaExpr); size_t _la = 0; #if __cplusplus > 201703L @@ -8995,169 +8916,100 @@ HogQLParser::ColumnLambdaExprContext* HogQLParser::columnLambdaExpr() { try { size_t alt; enterOuterAlt(_localctx, 1); - setState(1084); + setState(1074); _errHandler->sync(this); - switch (_input->LA(1)) { - case HogQLParser::LPAREN: { - setState(1059); - match(HogQLParser::LPAREN); - setState(1060); - identifier(); - setState(1065); - _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 133, _ctx); - while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { - if (alt == 1) { - setState(1061); - match(HogQLParser::COMMA); - setState(1062); - identifier(); - } - setState(1067); - _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 133, _ctx); + switch (getInterpreter()->adaptivePredict(_input, 135, _ctx)) { + case 1: { + setState(1047); + match(HogQLParser::LPAREN); + setState(1048); + identifier(); + setState(1053); + _errHandler->sync(this); + alt = getInterpreter()->adaptivePredict(_input, 131, _ctx); + while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { + if (alt == 1) { + setState(1049); + match(HogQLParser::COMMA); + setState(1050); + identifier(); } - setState(1069); + setState(1055); _errHandler->sync(this); + alt = getInterpreter()->adaptivePredict(_input, 131, _ctx); + } + setState(1057); + _errHandler->sync(this); - _la = _input->LA(1); - if (_la == HogQLParser::COMMA) { - setState(1068); - match(HogQLParser::COMMA); - } - setState(1071); - match(HogQLParser::RPAREN); - break; + _la = _input->LA(1); + if (_la == HogQLParser::COMMA) { + setState(1056); + match(HogQLParser::COMMA); } + setState(1059); + match(HogQLParser::RPAREN); + break; + } - case HogQLParser::ALL: - case HogQLParser::AND: - case HogQLParser::ANTI: - case HogQLParser::ANY: - case HogQLParser::ARRAY: - case HogQLParser::AS: - case HogQLParser::ASCENDING: - case HogQLParser::ASOF: - case HogQLParser::BETWEEN: - case HogQLParser::BOTH: - case HogQLParser::BY: - case HogQLParser::CASE: - case HogQLParser::CAST: - case HogQLParser::COHORT: - case HogQLParser::COLLATE: - case HogQLParser::CROSS: - case HogQLParser::CUBE: - case HogQLParser::CURRENT: - case HogQLParser::DATE: - case HogQLParser::DAY: - case HogQLParser::DESC: - case HogQLParser::DESCENDING: - case HogQLParser::DISTINCT: - case HogQLParser::ELSE: - case HogQLParser::END: - case HogQLParser::EXTRACT: - case HogQLParser::FINAL: - case HogQLParser::FIRST: - case HogQLParser::FOLLOWING: - case HogQLParser::FOR: - case HogQLParser::FROM: - case HogQLParser::FULL: - case HogQLParser::GROUP: - case HogQLParser::HAVING: - case HogQLParser::HOUR: - case HogQLParser::ID: - case HogQLParser::IF: - case HogQLParser::ILIKE: - case HogQLParser::IN: - case HogQLParser::INNER: - case HogQLParser::INTERVAL: - case HogQLParser::IS: - case HogQLParser::JOIN: - case HogQLParser::KEY: - case HogQLParser::LAST: - case HogQLParser::LEADING: - case HogQLParser::LEFT: - case HogQLParser::LIKE: - case HogQLParser::LIMIT: - case HogQLParser::MINUTE: - case HogQLParser::MONTH: - case HogQLParser::NOT: - case HogQLParser::NULLS: - case HogQLParser::OFFSET: - case HogQLParser::ON: - case HogQLParser::OR: - case HogQLParser::ORDER: - case HogQLParser::OUTER: - case HogQLParser::OVER: - case HogQLParser::PARTITION: - case HogQLParser::PRECEDING: - case HogQLParser::PREWHERE: - case HogQLParser::QUARTER: - case HogQLParser::RANGE: - case HogQLParser::RETURN: - case HogQLParser::RIGHT: - case HogQLParser::ROLLUP: - case HogQLParser::ROW: - case HogQLParser::ROWS: - case HogQLParser::SAMPLE: - case HogQLParser::SECOND: - case HogQLParser::SELECT: - case HogQLParser::SEMI: - case HogQLParser::SETTINGS: - case HogQLParser::SUBSTRING: - case HogQLParser::THEN: - case HogQLParser::TIES: - case HogQLParser::TIMESTAMP: - case HogQLParser::TO: - case HogQLParser::TOP: - case HogQLParser::TOTALS: - case HogQLParser::TRAILING: - case HogQLParser::TRIM: - case HogQLParser::TRUNCATE: - case HogQLParser::UNBOUNDED: - case HogQLParser::UNION: - case HogQLParser::USING: - case HogQLParser::WEEK: - case HogQLParser::WHEN: - case HogQLParser::WHERE: - case HogQLParser::WINDOW: - case HogQLParser::WITH: - case HogQLParser::YEAR: - case HogQLParser::IDENTIFIER: { - setState(1073); - identifier(); - setState(1078); - _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 135, _ctx); - while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { - if (alt == 1) { - setState(1074); - match(HogQLParser::COMMA); - setState(1075); - identifier(); - } - setState(1080); - _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 135, _ctx); + case 2: { + setState(1061); + identifier(); + setState(1066); + _errHandler->sync(this); + alt = getInterpreter()->adaptivePredict(_input, 133, _ctx); + while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { + if (alt == 1) { + setState(1062); + match(HogQLParser::COMMA); + setState(1063); + identifier(); } - setState(1082); + setState(1068); _errHandler->sync(this); + alt = getInterpreter()->adaptivePredict(_input, 133, _ctx); + } + setState(1070); + _errHandler->sync(this); - _la = _input->LA(1); - if (_la == HogQLParser::COMMA) { - setState(1081); - match(HogQLParser::COMMA); - } - break; + _la = _input->LA(1); + if (_la == HogQLParser::COMMA) { + setState(1069); + match(HogQLParser::COMMA); } + break; + } + + case 3: { + setState(1072); + match(HogQLParser::LPAREN); + setState(1073); + match(HogQLParser::RPAREN); + break; + } default: - throw NoViableAltException(this); + break; } - setState(1086); + setState(1076); match(HogQLParser::ARROW); - setState(1087); - columnExpr(0); + setState(1079); + _errHandler->sync(this); + switch (getInterpreter()->adaptivePredict(_input, 136, _ctx)) { + case 1: { + setState(1077); + columnExpr(0); + break; + } + + case 2: { + setState(1078); + block(); + break; + } + + default: + break; + } } catch (RecognitionException &e) { @@ -9284,7 +9136,7 @@ std::any HogQLParser::HogqlxTagElementNestedContext::accept(tree::ParseTreeVisit } HogQLParser::HogqlxTagElementContext* HogQLParser::hogqlxTagElement() { HogqlxTagElementContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 124, HogQLParser::RuleHogqlxTagElement); + enterRule(_localctx, 120, HogQLParser::RuleHogqlxTagElement); size_t _la = 0; #if __cplusplus > 201703L @@ -9295,31 +9147,31 @@ HogQLParser::HogqlxTagElementContext* HogQLParser::hogqlxTagElement() { exitRule(); }); try { - setState(1121); + setState(1113); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 141, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 140, _ctx)) { case 1: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 1); - setState(1089); + setState(1081); match(HogQLParser::LT); - setState(1090); + setState(1082); identifier(); - setState(1094); + setState(1086); _errHandler->sync(this); _la = _input->LA(1); while ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & -725088338784043010) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 3229277487103) != 0)) { - setState(1091); + setState(1083); hogqlxTagAttribute(); - setState(1096); + setState(1088); _errHandler->sync(this); _la = _input->LA(1); } - setState(1097); + setState(1089); match(HogQLParser::SLASH); - setState(1098); + setState(1090); match(HogQLParser::GT); break; } @@ -9327,40 +9179,40 @@ HogQLParser::HogqlxTagElementContext* HogQLParser::hogqlxTagElement() { case 2: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 2); - setState(1100); + setState(1092); match(HogQLParser::LT); - setState(1101); + setState(1093); identifier(); - setState(1105); + setState(1097); _errHandler->sync(this); _la = _input->LA(1); while ((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & -725088338784043010) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 3229277487103) != 0)) { - setState(1102); + setState(1094); hogqlxTagAttribute(); - setState(1107); + setState(1099); _errHandler->sync(this); _la = _input->LA(1); } - setState(1108); + setState(1100); match(HogQLParser::GT); - setState(1114); + setState(1106); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 140, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 139, _ctx)) { case 1: { - setState(1109); + setState(1101); hogqlxTagElement(); break; } case 2: { - setState(1110); + setState(1102); match(HogQLParser::LBRACE); - setState(1111); + setState(1103); columnExpr(0); - setState(1112); + setState(1104); match(HogQLParser::RBRACE); break; } @@ -9368,13 +9220,13 @@ HogQLParser::HogqlxTagElementContext* HogQLParser::hogqlxTagElement() { default: break; } - setState(1116); + setState(1108); match(HogQLParser::LT); - setState(1117); + setState(1109); match(HogQLParser::SLASH); - setState(1118); + setState(1110); identifier(); - setState(1119); + setState(1111); match(HogQLParser::GT); break; } @@ -9438,7 +9290,7 @@ std::any HogQLParser::HogqlxTagAttributeContext::accept(tree::ParseTreeVisitor * HogQLParser::HogqlxTagAttributeContext* HogQLParser::hogqlxTagAttribute() { HogqlxTagAttributeContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 126, HogQLParser::RuleHogqlxTagAttribute); + enterRule(_localctx, 122, HogQLParser::RuleHogqlxTagAttribute); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -9448,38 +9300,38 @@ HogQLParser::HogqlxTagAttributeContext* HogQLParser::hogqlxTagAttribute() { exitRule(); }); try { - setState(1134); + setState(1126); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 142, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 141, _ctx)) { case 1: { enterOuterAlt(_localctx, 1); - setState(1123); + setState(1115); identifier(); - setState(1124); + setState(1116); match(HogQLParser::EQ_SINGLE); - setState(1125); + setState(1117); string(); break; } case 2: { enterOuterAlt(_localctx, 2); - setState(1127); + setState(1119); identifier(); - setState(1128); + setState(1120); match(HogQLParser::EQ_SINGLE); - setState(1129); + setState(1121); match(HogQLParser::LBRACE); - setState(1130); + setState(1122); columnExpr(0); - setState(1131); + setState(1123); match(HogQLParser::RBRACE); break; } case 3: { enterOuterAlt(_localctx, 3); - setState(1133); + setState(1125); identifier(); break; } @@ -9535,7 +9387,7 @@ std::any HogQLParser::WithExprListContext::accept(tree::ParseTreeVisitor *visito HogQLParser::WithExprListContext* HogQLParser::withExprList() { WithExprListContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 128, HogQLParser::RuleWithExprList); + enterRule(_localctx, 124, HogQLParser::RuleWithExprList); size_t _la = 0; #if __cplusplus > 201703L @@ -9548,28 +9400,28 @@ HogQLParser::WithExprListContext* HogQLParser::withExprList() { try { size_t alt; enterOuterAlt(_localctx, 1); - setState(1136); + setState(1128); withExpr(); - setState(1141); + setState(1133); _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 143, _ctx); + alt = getInterpreter()->adaptivePredict(_input, 142, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { - setState(1137); + setState(1129); match(HogQLParser::COMMA); - setState(1138); + setState(1130); withExpr(); } - setState(1143); + setState(1135); _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 143, _ctx); + alt = getInterpreter()->adaptivePredict(_input, 142, _ctx); } - setState(1145); + setState(1137); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COMMA) { - setState(1144); + setState(1136); match(HogQLParser::COMMA); } @@ -9654,7 +9506,7 @@ std::any HogQLParser::WithExprSubqueryContext::accept(tree::ParseTreeVisitor *vi } HogQLParser::WithExprContext* HogQLParser::withExpr() { WithExprContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 130, HogQLParser::RuleWithExpr); + enterRule(_localctx, 126, HogQLParser::RuleWithExpr); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -9664,21 +9516,21 @@ HogQLParser::WithExprContext* HogQLParser::withExpr() { exitRule(); }); try { - setState(1157); + setState(1149); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 145, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 144, _ctx)) { case 1: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 1); - setState(1147); + setState(1139); identifier(); - setState(1148); + setState(1140); match(HogQLParser::AS); - setState(1149); + setState(1141); match(HogQLParser::LPAREN); - setState(1150); + setState(1142); selectUnionStmt(); - setState(1151); + setState(1143); match(HogQLParser::RPAREN); break; } @@ -9686,11 +9538,11 @@ HogQLParser::WithExprContext* HogQLParser::withExpr() { case 2: { _localctx = _tracker.createInstance(_localctx); enterOuterAlt(_localctx, 2); - setState(1153); + setState(1145); columnExpr(0); - setState(1154); + setState(1146); match(HogQLParser::AS); - setState(1155); + setState(1147); identifier(); break; } @@ -9746,7 +9598,7 @@ std::any HogQLParser::ColumnIdentifierContext::accept(tree::ParseTreeVisitor *vi HogQLParser::ColumnIdentifierContext* HogQLParser::columnIdentifier() { ColumnIdentifierContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 132, HogQLParser::RuleColumnIdentifier); + enterRule(_localctx, 128, HogQLParser::RuleColumnIdentifier); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -9756,12 +9608,12 @@ HogQLParser::ColumnIdentifierContext* HogQLParser::columnIdentifier() { exitRule(); }); try { - setState(1166); + setState(1158); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::LBRACE: { enterOuterAlt(_localctx, 1); - setState(1159); + setState(1151); placeholder(); break; } @@ -9861,14 +9713,14 @@ HogQLParser::ColumnIdentifierContext* HogQLParser::columnIdentifier() { case HogQLParser::YEAR: case HogQLParser::IDENTIFIER: { enterOuterAlt(_localctx, 2); - setState(1163); + setState(1155); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 146, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 145, _ctx)) { case 1: { - setState(1160); + setState(1152); tableIdentifier(); - setState(1161); + setState(1153); match(HogQLParser::DOT); break; } @@ -9876,7 +9728,7 @@ HogQLParser::ColumnIdentifierContext* HogQLParser::columnIdentifier() { default: break; } - setState(1165); + setState(1157); nestedIdentifier(); break; } @@ -9932,7 +9784,7 @@ std::any HogQLParser::NestedIdentifierContext::accept(tree::ParseTreeVisitor *vi HogQLParser::NestedIdentifierContext* HogQLParser::nestedIdentifier() { NestedIdentifierContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 134, HogQLParser::RuleNestedIdentifier); + enterRule(_localctx, 130, HogQLParser::RuleNestedIdentifier); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -9944,21 +9796,21 @@ HogQLParser::NestedIdentifierContext* HogQLParser::nestedIdentifier() { try { size_t alt; enterOuterAlt(_localctx, 1); - setState(1168); + setState(1160); identifier(); - setState(1173); + setState(1165); _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 148, _ctx); + alt = getInterpreter()->adaptivePredict(_input, 147, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { - setState(1169); + setState(1161); match(HogQLParser::DOT); - setState(1170); + setState(1162); identifier(); } - setState(1175); + setState(1167); _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 148, _ctx); + alt = getInterpreter()->adaptivePredict(_input, 147, _ctx); } } @@ -10107,8 +9959,8 @@ HogQLParser::TableExprContext* HogQLParser::tableExpr(int precedence) { HogQLParser::TableExprContext *_localctx = _tracker.createInstance(_ctx, parentState); HogQLParser::TableExprContext *previousContext = _localctx; (void)previousContext; // Silence compiler, in case the context is not used by generated code. - size_t startState = 136; - enterRecursionRule(_localctx, 136, HogQLParser::RuleTableExpr, precedence); + size_t startState = 132; + enterRecursionRule(_localctx, 132, HogQLParser::RuleTableExpr, precedence); @@ -10122,15 +9974,15 @@ HogQLParser::TableExprContext* HogQLParser::tableExpr(int precedence) { try { size_t alt; enterOuterAlt(_localctx, 1); - setState(1185); + setState(1177); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 149, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 148, _ctx)) { case 1: { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(1177); + setState(1169); tableIdentifier(); break; } @@ -10139,7 +9991,7 @@ HogQLParser::TableExprContext* HogQLParser::tableExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(1178); + setState(1170); tableFunctionExpr(); break; } @@ -10148,11 +10000,11 @@ HogQLParser::TableExprContext* HogQLParser::tableExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(1179); + setState(1171); match(HogQLParser::LPAREN); - setState(1180); + setState(1172); selectUnionStmt(); - setState(1181); + setState(1173); match(HogQLParser::RPAREN); break; } @@ -10161,7 +10013,7 @@ HogQLParser::TableExprContext* HogQLParser::tableExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(1183); + setState(1175); hogqlxTagElement(); break; } @@ -10170,7 +10022,7 @@ HogQLParser::TableExprContext* HogQLParser::tableExpr(int precedence) { _localctx = _tracker.createInstance(_localctx); _ctx = _localctx; previousContext = _localctx; - setState(1184); + setState(1176); placeholder(); break; } @@ -10179,9 +10031,9 @@ HogQLParser::TableExprContext* HogQLParser::tableExpr(int precedence) { break; } _ctx->stop = _input->LT(-1); - setState(1195); + setState(1187); _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 151, _ctx); + alt = getInterpreter()->adaptivePredict(_input, 150, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { if (!_parseListeners.empty()) @@ -10190,10 +10042,10 @@ HogQLParser::TableExprContext* HogQLParser::tableExpr(int precedence) { auto newContext = _tracker.createInstance(_tracker.createInstance(parentContext, parentState)); _localctx = newContext; pushNewRecursionContext(newContext, startState, RuleTableExpr); - setState(1187); + setState(1179); if (!(precpred(_ctx, 3))) throw FailedPredicateException(this, "precpred(_ctx, 3)"); - setState(1191); + setState(1183); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::DATE: @@ -10201,15 +10053,15 @@ HogQLParser::TableExprContext* HogQLParser::tableExpr(int precedence) { case HogQLParser::ID: case HogQLParser::KEY: case HogQLParser::IDENTIFIER: { - setState(1188); + setState(1180); alias(); break; } case HogQLParser::AS: { - setState(1189); + setState(1181); match(HogQLParser::AS); - setState(1190); + setState(1182); identifier(); break; } @@ -10218,9 +10070,9 @@ HogQLParser::TableExprContext* HogQLParser::tableExpr(int precedence) { throw NoViableAltException(this); } } - setState(1197); + setState(1189); _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 151, _ctx); + alt = getInterpreter()->adaptivePredict(_input, 150, _ctx); } } catch (RecognitionException &e) { @@ -10268,7 +10120,7 @@ std::any HogQLParser::TableFunctionExprContext::accept(tree::ParseTreeVisitor *v HogQLParser::TableFunctionExprContext* HogQLParser::tableFunctionExpr() { TableFunctionExprContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 138, HogQLParser::RuleTableFunctionExpr); + enterRule(_localctx, 134, HogQLParser::RuleTableFunctionExpr); size_t _la = 0; #if __cplusplus > 201703L @@ -10280,11 +10132,11 @@ HogQLParser::TableFunctionExprContext* HogQLParser::tableFunctionExpr() { }); try { enterOuterAlt(_localctx, 1); - setState(1198); + setState(1190); identifier(); - setState(1199); + setState(1191); match(HogQLParser::LPAREN); - setState(1201); + setState(1193); _errHandler->sync(this); _la = _input->LA(1); @@ -10292,10 +10144,10 @@ HogQLParser::TableFunctionExprContext* HogQLParser::tableFunctionExpr() { ((1ULL << _la) & -4503602311741442) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 64)) & 90493036243451903) != 0) || ((((_la - 128) & ~ 0x3fULL) == 0) && ((1ULL << (_la - 128)) & 18455) != 0)) { - setState(1200); + setState(1192); tableArgList(); } - setState(1203); + setState(1195); match(HogQLParser::RPAREN); } @@ -10341,7 +10193,7 @@ std::any HogQLParser::TableIdentifierContext::accept(tree::ParseTreeVisitor *vis HogQLParser::TableIdentifierContext* HogQLParser::tableIdentifier() { TableIdentifierContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 140, HogQLParser::RuleTableIdentifier); + enterRule(_localctx, 136, HogQLParser::RuleTableIdentifier); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -10352,14 +10204,14 @@ HogQLParser::TableIdentifierContext* HogQLParser::tableIdentifier() { }); try { enterOuterAlt(_localctx, 1); - setState(1208); + setState(1200); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 153, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 152, _ctx)) { case 1: { - setState(1205); + setState(1197); databaseIdentifier(); - setState(1206); + setState(1198); match(HogQLParser::DOT); break; } @@ -10367,7 +10219,7 @@ HogQLParser::TableIdentifierContext* HogQLParser::tableIdentifier() { default: break; } - setState(1210); + setState(1202); identifier(); } @@ -10417,7 +10269,7 @@ std::any HogQLParser::TableArgListContext::accept(tree::ParseTreeVisitor *visito HogQLParser::TableArgListContext* HogQLParser::tableArgList() { TableArgListContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 142, HogQLParser::RuleTableArgList); + enterRule(_localctx, 138, HogQLParser::RuleTableArgList); size_t _la = 0; #if __cplusplus > 201703L @@ -10430,28 +10282,28 @@ HogQLParser::TableArgListContext* HogQLParser::tableArgList() { try { size_t alt; enterOuterAlt(_localctx, 1); - setState(1212); + setState(1204); columnExpr(0); - setState(1217); + setState(1209); _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 154, _ctx); + alt = getInterpreter()->adaptivePredict(_input, 153, _ctx); while (alt != 2 && alt != atn::ATN::INVALID_ALT_NUMBER) { if (alt == 1) { - setState(1213); + setState(1205); match(HogQLParser::COMMA); - setState(1214); + setState(1206); columnExpr(0); } - setState(1219); + setState(1211); _errHandler->sync(this); - alt = getInterpreter()->adaptivePredict(_input, 154, _ctx); + alt = getInterpreter()->adaptivePredict(_input, 153, _ctx); } - setState(1221); + setState(1213); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::COMMA) { - setState(1220); + setState(1212); match(HogQLParser::COMMA); } @@ -10490,7 +10342,7 @@ std::any HogQLParser::DatabaseIdentifierContext::accept(tree::ParseTreeVisitor * HogQLParser::DatabaseIdentifierContext* HogQLParser::databaseIdentifier() { DatabaseIdentifierContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 144, HogQLParser::RuleDatabaseIdentifier); + enterRule(_localctx, 140, HogQLParser::RuleDatabaseIdentifier); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -10501,7 +10353,7 @@ HogQLParser::DatabaseIdentifierContext* HogQLParser::databaseIdentifier() { }); try { enterOuterAlt(_localctx, 1); - setState(1223); + setState(1215); identifier(); } @@ -10555,7 +10407,7 @@ std::any HogQLParser::FloatingLiteralContext::accept(tree::ParseTreeVisitor *vis HogQLParser::FloatingLiteralContext* HogQLParser::floatingLiteral() { FloatingLiteralContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 146, HogQLParser::RuleFloatingLiteral); + enterRule(_localctx, 142, HogQLParser::RuleFloatingLiteral); size_t _la = 0; #if __cplusplus > 201703L @@ -10566,21 +10418,21 @@ HogQLParser::FloatingLiteralContext* HogQLParser::floatingLiteral() { exitRule(); }); try { - setState(1233); + setState(1225); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::FLOATING_LITERAL: { enterOuterAlt(_localctx, 1); - setState(1225); + setState(1217); match(HogQLParser::FLOATING_LITERAL); break; } case HogQLParser::DOT: { enterOuterAlt(_localctx, 2); - setState(1226); + setState(1218); match(HogQLParser::DOT); - setState(1227); + setState(1219); _la = _input->LA(1); if (!(_la == HogQLParser::OCTAL_LITERAL @@ -10596,16 +10448,16 @@ HogQLParser::FloatingLiteralContext* HogQLParser::floatingLiteral() { case HogQLParser::DECIMAL_LITERAL: { enterOuterAlt(_localctx, 3); - setState(1228); + setState(1220); match(HogQLParser::DECIMAL_LITERAL); - setState(1229); + setState(1221); match(HogQLParser::DOT); - setState(1231); + setState(1223); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 156, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 155, _ctx)) { case 1: { - setState(1230); + setState(1222); _la = _input->LA(1); if (!(_la == HogQLParser::OCTAL_LITERAL @@ -10692,7 +10544,7 @@ std::any HogQLParser::NumberLiteralContext::accept(tree::ParseTreeVisitor *visit HogQLParser::NumberLiteralContext* HogQLParser::numberLiteral() { NumberLiteralContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 148, HogQLParser::RuleNumberLiteral); + enterRule(_localctx, 144, HogQLParser::RuleNumberLiteral); size_t _la = 0; #if __cplusplus > 201703L @@ -10704,14 +10556,14 @@ HogQLParser::NumberLiteralContext* HogQLParser::numberLiteral() { }); try { enterOuterAlt(_localctx, 1); - setState(1236); + setState(1228); _errHandler->sync(this); _la = _input->LA(1); if (_la == HogQLParser::DASH || _la == HogQLParser::PLUS) { - setState(1235); + setState(1227); _la = _input->LA(1); if (!(_la == HogQLParser::DASH @@ -10723,41 +10575,41 @@ HogQLParser::NumberLiteralContext* HogQLParser::numberLiteral() { consume(); } } - setState(1244); + setState(1236); _errHandler->sync(this); - switch (getInterpreter()->adaptivePredict(_input, 159, _ctx)) { + switch (getInterpreter()->adaptivePredict(_input, 158, _ctx)) { case 1: { - setState(1238); + setState(1230); floatingLiteral(); break; } case 2: { - setState(1239); + setState(1231); match(HogQLParser::OCTAL_LITERAL); break; } case 3: { - setState(1240); + setState(1232); match(HogQLParser::DECIMAL_LITERAL); break; } case 4: { - setState(1241); + setState(1233); match(HogQLParser::HEXADECIMAL_LITERAL); break; } case 5: { - setState(1242); + setState(1234); match(HogQLParser::INF); break; } case 6: { - setState(1243); + setState(1235); match(HogQLParser::NAN_SQL); break; } @@ -10809,7 +10661,7 @@ std::any HogQLParser::LiteralContext::accept(tree::ParseTreeVisitor *visitor) { HogQLParser::LiteralContext* HogQLParser::literal() { LiteralContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 150, HogQLParser::RuleLiteral); + enterRule(_localctx, 146, HogQLParser::RuleLiteral); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -10819,7 +10671,7 @@ HogQLParser::LiteralContext* HogQLParser::literal() { exitRule(); }); try { - setState(1249); + setState(1241); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::INF: @@ -10832,21 +10684,21 @@ HogQLParser::LiteralContext* HogQLParser::literal() { case HogQLParser::DOT: case HogQLParser::PLUS: { enterOuterAlt(_localctx, 1); - setState(1246); + setState(1238); numberLiteral(); break; } case HogQLParser::STRING_LITERAL: { enterOuterAlt(_localctx, 2); - setState(1247); + setState(1239); match(HogQLParser::STRING_LITERAL); break; } case HogQLParser::NULL_SQL: { enterOuterAlt(_localctx, 3); - setState(1248); + setState(1240); match(HogQLParser::NULL_SQL); break; } @@ -10918,7 +10770,7 @@ std::any HogQLParser::IntervalContext::accept(tree::ParseTreeVisitor *visitor) { HogQLParser::IntervalContext* HogQLParser::interval() { IntervalContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 152, HogQLParser::RuleInterval); + enterRule(_localctx, 148, HogQLParser::RuleInterval); size_t _la = 0; #if __cplusplus > 201703L @@ -10930,7 +10782,7 @@ HogQLParser::IntervalContext* HogQLParser::interval() { }); try { enterOuterAlt(_localctx, 1); - setState(1251); + setState(1243); _la = _input->LA(1); if (!((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 108086665936896000) != 0) || ((((_la - 70) & ~ 0x3fULL) == 0) && @@ -11313,7 +11165,7 @@ std::any HogQLParser::KeywordContext::accept(tree::ParseTreeVisitor *visitor) { HogQLParser::KeywordContext* HogQLParser::keyword() { KeywordContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 154, HogQLParser::RuleKeyword); + enterRule(_localctx, 150, HogQLParser::RuleKeyword); size_t _la = 0; #if __cplusplus > 201703L @@ -11325,7 +11177,7 @@ HogQLParser::KeywordContext* HogQLParser::keyword() { }); try { enterOuterAlt(_localctx, 1); - setState(1253); + setState(1245); _la = _input->LA(1); if (!((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & -833175004720939010) != 0) || ((((_la - 64) & ~ 0x3fULL) == 0) && @@ -11384,7 +11236,7 @@ std::any HogQLParser::KeywordForAliasContext::accept(tree::ParseTreeVisitor *vis HogQLParser::KeywordForAliasContext* HogQLParser::keywordForAlias() { KeywordForAliasContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 156, HogQLParser::RuleKeywordForAlias); + enterRule(_localctx, 152, HogQLParser::RuleKeywordForAlias); size_t _la = 0; #if __cplusplus > 201703L @@ -11396,7 +11248,7 @@ HogQLParser::KeywordForAliasContext* HogQLParser::keywordForAlias() { }); try { enterOuterAlt(_localctx, 1); - setState(1255); + setState(1247); _la = _input->LA(1); if (!((((_la & ~ 0x3fULL) == 0) && ((1ULL << _la) & 282025807314944) != 0))) { @@ -11446,7 +11298,7 @@ std::any HogQLParser::AliasContext::accept(tree::ParseTreeVisitor *visitor) { HogQLParser::AliasContext* HogQLParser::alias() { AliasContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 158, HogQLParser::RuleAlias); + enterRule(_localctx, 154, HogQLParser::RuleAlias); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -11456,12 +11308,12 @@ HogQLParser::AliasContext* HogQLParser::alias() { exitRule(); }); try { - setState(1259); + setState(1251); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::IDENTIFIER: { enterOuterAlt(_localctx, 1); - setState(1257); + setState(1249); match(HogQLParser::IDENTIFIER); break; } @@ -11471,7 +11323,7 @@ HogQLParser::AliasContext* HogQLParser::alias() { case HogQLParser::ID: case HogQLParser::KEY: { enterOuterAlt(_localctx, 2); - setState(1258); + setState(1250); keywordForAlias(); break; } @@ -11523,7 +11375,7 @@ std::any HogQLParser::IdentifierContext::accept(tree::ParseTreeVisitor *visitor) HogQLParser::IdentifierContext* HogQLParser::identifier() { IdentifierContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 160, HogQLParser::RuleIdentifier); + enterRule(_localctx, 156, HogQLParser::RuleIdentifier); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -11533,12 +11385,12 @@ HogQLParser::IdentifierContext* HogQLParser::identifier() { exitRule(); }); try { - setState(1264); + setState(1256); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::IDENTIFIER: { enterOuterAlt(_localctx, 1); - setState(1261); + setState(1253); match(HogQLParser::IDENTIFIER); break; } @@ -11552,7 +11404,7 @@ HogQLParser::IdentifierContext* HogQLParser::identifier() { case HogQLParser::WEEK: case HogQLParser::YEAR: { enterOuterAlt(_localctx, 2); - setState(1262); + setState(1254); interval(); break; } @@ -11643,7 +11495,7 @@ HogQLParser::IdentifierContext* HogQLParser::identifier() { case HogQLParser::WINDOW: case HogQLParser::WITH: { enterOuterAlt(_localctx, 3); - setState(1263); + setState(1255); keyword(); break; } @@ -11695,7 +11547,7 @@ std::any HogQLParser::EnumValueContext::accept(tree::ParseTreeVisitor *visitor) HogQLParser::EnumValueContext* HogQLParser::enumValue() { EnumValueContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 162, HogQLParser::RuleEnumValue); + enterRule(_localctx, 158, HogQLParser::RuleEnumValue); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -11706,11 +11558,11 @@ HogQLParser::EnumValueContext* HogQLParser::enumValue() { }); try { enterOuterAlt(_localctx, 1); - setState(1266); + setState(1258); string(); - setState(1267); + setState(1259); match(HogQLParser::EQ_SINGLE); - setState(1268); + setState(1260); numberLiteral(); } @@ -11756,7 +11608,7 @@ std::any HogQLParser::PlaceholderContext::accept(tree::ParseTreeVisitor *visitor HogQLParser::PlaceholderContext* HogQLParser::placeholder() { PlaceholderContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 164, HogQLParser::RulePlaceholder); + enterRule(_localctx, 160, HogQLParser::RulePlaceholder); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -11767,11 +11619,11 @@ HogQLParser::PlaceholderContext* HogQLParser::placeholder() { }); try { enterOuterAlt(_localctx, 1); - setState(1270); + setState(1262); match(HogQLParser::LBRACE); - setState(1271); + setState(1263); nestedIdentifier(); - setState(1272); + setState(1264); match(HogQLParser::RBRACE); } @@ -11813,7 +11665,7 @@ std::any HogQLParser::StringContext::accept(tree::ParseTreeVisitor *visitor) { HogQLParser::StringContext* HogQLParser::string() { StringContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 166, HogQLParser::RuleString); + enterRule(_localctx, 162, HogQLParser::RuleString); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -11823,19 +11675,19 @@ HogQLParser::StringContext* HogQLParser::string() { exitRule(); }); try { - setState(1276); + setState(1268); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::STRING_LITERAL: { enterOuterAlt(_localctx, 1); - setState(1274); + setState(1266); match(HogQLParser::STRING_LITERAL); break; } case HogQLParser::QUOTE_SINGLE_TEMPLATE: { enterOuterAlt(_localctx, 2); - setState(1275); + setState(1267); templateString(); break; } @@ -11891,7 +11743,7 @@ std::any HogQLParser::TemplateStringContext::accept(tree::ParseTreeVisitor *visi HogQLParser::TemplateStringContext* HogQLParser::templateString() { TemplateStringContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 168, HogQLParser::RuleTemplateString); + enterRule(_localctx, 164, HogQLParser::RuleTemplateString); size_t _la = 0; #if __cplusplus > 201703L @@ -11903,21 +11755,21 @@ HogQLParser::TemplateStringContext* HogQLParser::templateString() { }); try { enterOuterAlt(_localctx, 1); - setState(1278); + setState(1270); match(HogQLParser::QUOTE_SINGLE_TEMPLATE); - setState(1282); + setState(1274); _errHandler->sync(this); _la = _input->LA(1); while (_la == HogQLParser::STRING_TEXT || _la == HogQLParser::STRING_ESCAPE_TRIGGER) { - setState(1279); + setState(1271); stringContents(); - setState(1284); + setState(1276); _errHandler->sync(this); _la = _input->LA(1); } - setState(1285); + setState(1277); match(HogQLParser::QUOTE_SINGLE); } @@ -11967,7 +11819,7 @@ std::any HogQLParser::StringContentsContext::accept(tree::ParseTreeVisitor *visi HogQLParser::StringContentsContext* HogQLParser::stringContents() { StringContentsContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 170, HogQLParser::RuleStringContents); + enterRule(_localctx, 166, HogQLParser::RuleStringContents); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -11977,23 +11829,23 @@ HogQLParser::StringContentsContext* HogQLParser::stringContents() { exitRule(); }); try { - setState(1292); + setState(1284); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::STRING_ESCAPE_TRIGGER: { enterOuterAlt(_localctx, 1); - setState(1287); + setState(1279); match(HogQLParser::STRING_ESCAPE_TRIGGER); - setState(1288); + setState(1280); columnExpr(0); - setState(1289); + setState(1281); match(HogQLParser::RBRACE); break; } case HogQLParser::STRING_TEXT: { enterOuterAlt(_localctx, 2); - setState(1291); + setState(1283); match(HogQLParser::STRING_TEXT); break; } @@ -12049,7 +11901,7 @@ std::any HogQLParser::FullTemplateStringContext::accept(tree::ParseTreeVisitor * HogQLParser::FullTemplateStringContext* HogQLParser::fullTemplateString() { FullTemplateStringContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 172, HogQLParser::RuleFullTemplateString); + enterRule(_localctx, 168, HogQLParser::RuleFullTemplateString); size_t _la = 0; #if __cplusplus > 201703L @@ -12061,21 +11913,21 @@ HogQLParser::FullTemplateStringContext* HogQLParser::fullTemplateString() { }); try { enterOuterAlt(_localctx, 1); - setState(1294); + setState(1286); match(HogQLParser::QUOTE_SINGLE_TEMPLATE_FULL); - setState(1298); + setState(1290); _errHandler->sync(this); _la = _input->LA(1); while (_la == HogQLParser::FULL_STRING_TEXT || _la == HogQLParser::FULL_STRING_ESCAPE_TRIGGER) { - setState(1295); + setState(1287); stringContentsFull(); - setState(1300); + setState(1292); _errHandler->sync(this); _la = _input->LA(1); } - setState(1301); + setState(1293); match(HogQLParser::EOF); } @@ -12125,7 +11977,7 @@ std::any HogQLParser::StringContentsFullContext::accept(tree::ParseTreeVisitor * HogQLParser::StringContentsFullContext* HogQLParser::stringContentsFull() { StringContentsFullContext *_localctx = _tracker.createInstance(_ctx, getState()); - enterRule(_localctx, 174, HogQLParser::RuleStringContentsFull); + enterRule(_localctx, 170, HogQLParser::RuleStringContentsFull); #if __cplusplus > 201703L auto onExit = finally([=, this] { @@ -12135,23 +11987,23 @@ HogQLParser::StringContentsFullContext* HogQLParser::stringContentsFull() { exitRule(); }); try { - setState(1308); + setState(1300); _errHandler->sync(this); switch (_input->LA(1)) { case HogQLParser::FULL_STRING_ESCAPE_TRIGGER: { enterOuterAlt(_localctx, 1); - setState(1303); + setState(1295); match(HogQLParser::FULL_STRING_ESCAPE_TRIGGER); - setState(1304); + setState(1296); columnExpr(0); - setState(1305); + setState(1297); match(HogQLParser::RBRACE); break; } case HogQLParser::FULL_STRING_TEXT: { enterOuterAlt(_localctx, 2); - setState(1307); + setState(1299); match(HogQLParser::FULL_STRING_TEXT); break; } @@ -12174,7 +12026,7 @@ bool HogQLParser::sempred(RuleContext *context, size_t ruleIndex, size_t predica switch (ruleIndex) { case 39: return joinExprSempred(antlrcpp::downCast(context), predicateIndex); case 58: return columnExprSempred(antlrcpp::downCast(context), predicateIndex); - case 68: return tableExprSempred(antlrcpp::downCast(context), predicateIndex); + case 66: return tableExprSempred(antlrcpp::downCast(context), predicateIndex); default: break; @@ -12195,22 +12047,23 @@ bool HogQLParser::joinExprSempred(JoinExprContext *_localctx, size_t predicateIn bool HogQLParser::columnExprSempred(ColumnExprContext *_localctx, size_t predicateIndex) { switch (predicateIndex) { - case 2: return precpred(_ctx, 18); - case 3: return precpred(_ctx, 17); - case 4: return precpred(_ctx, 16); - case 5: return precpred(_ctx, 14); - case 6: return precpred(_ctx, 12); - case 7: return precpred(_ctx, 11); - case 8: return precpred(_ctx, 10); - case 9: return precpred(_ctx, 9); - case 10: return precpred(_ctx, 25); - case 11: return precpred(_ctx, 24); - case 12: return precpred(_ctx, 23); - case 13: return precpred(_ctx, 22); - case 14: return precpred(_ctx, 21); - case 15: return precpred(_ctx, 20); - case 16: return precpred(_ctx, 15); - case 17: return precpred(_ctx, 8); + case 2: return precpred(_ctx, 19); + case 3: return precpred(_ctx, 18); + case 4: return precpred(_ctx, 17); + case 5: return precpred(_ctx, 15); + case 6: return precpred(_ctx, 13); + case 7: return precpred(_ctx, 12); + case 8: return precpred(_ctx, 11); + case 9: return precpred(_ctx, 10); + case 10: return precpred(_ctx, 30); + case 11: return precpred(_ctx, 26); + case 12: return precpred(_ctx, 25); + case 13: return precpred(_ctx, 24); + case 14: return precpred(_ctx, 23); + case 15: return precpred(_ctx, 22); + case 16: return precpred(_ctx, 21); + case 17: return precpred(_ctx, 16); + case 18: return precpred(_ctx, 9); default: break; @@ -12220,7 +12073,7 @@ bool HogQLParser::columnExprSempred(ColumnExprContext *_localctx, size_t predica bool HogQLParser::tableExprSempred(TableExprContext *_localctx, size_t predicateIndex) { switch (predicateIndex) { - case 18: return precpred(_ctx, 3); + case 19: return precpred(_ctx, 3); default: break; diff --git a/hogql_parser/HogQLParser.h b/hogql_parser/HogQLParser.h index acabb96a0baa5..a96c1ee007954 100644 --- a/hogql_parser/HogQLParser.h +++ b/hogql_parser/HogQLParser.h @@ -1,5 +1,5 @@ -// Generated from HogQLParser.g4 by ANTLR 4.13.1 +// Generated from HogQLParser.g4 by ANTLR 4.13.2 #pragma once @@ -61,16 +61,15 @@ class HogQLParser : public antlr4::Parser { RuleSettingExprList = 47, RuleSettingExpr = 48, RuleWindowExpr = 49, RuleWinPartitionByClause = 50, RuleWinOrderByClause = 51, RuleWinFrameClause = 52, RuleWinFrameExtend = 53, RuleWinFrameBound = 54, RuleExpr = 55, RuleColumnTypeExpr = 56, - RuleColumnExprList = 57, RuleColumnExpr = 58, RuleColumnArgList = 59, - RuleColumnArgExpr = 60, RuleColumnLambdaExpr = 61, RuleHogqlxTagElement = 62, - RuleHogqlxTagAttribute = 63, RuleWithExprList = 64, RuleWithExpr = 65, - RuleColumnIdentifier = 66, RuleNestedIdentifier = 67, RuleTableExpr = 68, - RuleTableFunctionExpr = 69, RuleTableIdentifier = 70, RuleTableArgList = 71, - RuleDatabaseIdentifier = 72, RuleFloatingLiteral = 73, RuleNumberLiteral = 74, - RuleLiteral = 75, RuleInterval = 76, RuleKeyword = 77, RuleKeywordForAlias = 78, - RuleAlias = 79, RuleIdentifier = 80, RuleEnumValue = 81, RulePlaceholder = 82, - RuleString = 83, RuleTemplateString = 84, RuleStringContents = 85, RuleFullTemplateString = 86, - RuleStringContentsFull = 87 + RuleColumnExprList = 57, RuleColumnExpr = 58, RuleColumnLambdaExpr = 59, + RuleHogqlxTagElement = 60, RuleHogqlxTagAttribute = 61, RuleWithExprList = 62, + RuleWithExpr = 63, RuleColumnIdentifier = 64, RuleNestedIdentifier = 65, + RuleTableExpr = 66, RuleTableFunctionExpr = 67, RuleTableIdentifier = 68, + RuleTableArgList = 69, RuleDatabaseIdentifier = 70, RuleFloatingLiteral = 71, + RuleNumberLiteral = 72, RuleLiteral = 73, RuleInterval = 74, RuleKeyword = 75, + RuleKeywordForAlias = 76, RuleAlias = 77, RuleIdentifier = 78, RuleEnumValue = 79, + RulePlaceholder = 80, RuleString = 81, RuleTemplateString = 82, RuleStringContents = 83, + RuleFullTemplateString = 84, RuleStringContentsFull = 85 }; explicit HogQLParser(antlr4::TokenStream *input); @@ -149,8 +148,6 @@ class HogQLParser : public antlr4::Parser { class ColumnTypeExprContext; class ColumnExprListContext; class ColumnExprContext; - class ColumnArgListContext; - class ColumnArgExprContext; class ColumnLambdaExprContext; class HogqlxTagElementContext; class HogqlxTagAttributeContext; @@ -1567,6 +1564,8 @@ class HogQLParser : public antlr4::Parser { public: ColumnExprWinFunctionTargetContext(ColumnExprContext *ctx); + HogQLParser::ColumnExprListContext *columnExprs = nullptr; + HogQLParser::ColumnExprListContext *columnArgList = nullptr; std::vector identifier(); IdentifierContext* identifier(size_t i); antlr4::tree::TerminalNode *OVER(); @@ -1574,9 +1573,9 @@ class HogQLParser : public antlr4::Parser { antlr4::tree::TerminalNode* LPAREN(size_t i); std::vector RPAREN(); antlr4::tree::TerminalNode* RPAREN(size_t i); - ColumnExprListContext *columnExprList(); + std::vector columnExprList(); + ColumnExprListContext* columnExprList(size_t i); antlr4::tree::TerminalNode *DISTINCT(); - ColumnArgListContext *columnArgList(); virtual std::any accept(antlr4::tree::ParseTreeVisitor *visitor) override; }; @@ -1638,6 +1637,18 @@ class HogQLParser : public antlr4::Parser { virtual std::any accept(antlr4::tree::ParseTreeVisitor *visitor) override; }; + class ColumnExprCallContext : public ColumnExprContext { + public: + ColumnExprCallContext(ColumnExprContext *ctx); + + ColumnExprContext *columnExpr(); + antlr4::tree::TerminalNode *LPAREN(); + antlr4::tree::TerminalNode *RPAREN(); + ColumnExprListContext *columnExprList(); + + virtual std::any accept(antlr4::tree::ParseTreeVisitor *visitor) override; + }; + class ColumnExprArrayAccessContext : public ColumnExprContext { public: ColumnExprArrayAccessContext(ColumnExprContext *ctx); @@ -1786,6 +1797,8 @@ class HogQLParser : public antlr4::Parser { public: ColumnExprWinFunctionContext(ColumnExprContext *ctx); + HogQLParser::ColumnExprListContext *columnExprs = nullptr; + HogQLParser::ColumnExprListContext *columnArgList = nullptr; IdentifierContext *identifier(); antlr4::tree::TerminalNode *OVER(); std::vector LPAREN(); @@ -1793,9 +1806,18 @@ class HogQLParser : public antlr4::Parser { WindowExprContext *windowExpr(); std::vector RPAREN(); antlr4::tree::TerminalNode* RPAREN(size_t i); - ColumnExprListContext *columnExprList(); + std::vector columnExprList(); + ColumnExprListContext* columnExprList(size_t i); antlr4::tree::TerminalNode *DISTINCT(); - ColumnArgListContext *columnArgList(); + + virtual std::any accept(antlr4::tree::ParseTreeVisitor *visitor) override; + }; + + class ColumnExprLambdaContext : public ColumnExprContext { + public: + ColumnExprLambdaContext(ColumnExprContext *ctx); + + ColumnLambdaExprContext *columnLambdaExpr(); virtual std::any accept(antlr4::tree::ParseTreeVisitor *visitor) override; }; @@ -1813,14 +1835,16 @@ class HogQLParser : public antlr4::Parser { public: ColumnExprFunctionContext(ColumnExprContext *ctx); + HogQLParser::ColumnExprListContext *columnExprs = nullptr; + HogQLParser::ColumnExprListContext *columnArgList = nullptr; IdentifierContext *identifier(); std::vector LPAREN(); antlr4::tree::TerminalNode* LPAREN(size_t i); std::vector RPAREN(); antlr4::tree::TerminalNode* RPAREN(size_t i); antlr4::tree::TerminalNode *DISTINCT(); - ColumnArgListContext *columnArgList(); - ColumnExprListContext *columnExprList(); + std::vector columnExprList(); + ColumnExprListContext* columnExprList(size_t i); virtual std::any accept(antlr4::tree::ParseTreeVisitor *visitor) override; }; @@ -1838,46 +1862,17 @@ class HogQLParser : public antlr4::Parser { ColumnExprContext* columnExpr(); ColumnExprContext* columnExpr(int precedence); - class ColumnArgListContext : public antlr4::ParserRuleContext { - public: - ColumnArgListContext(antlr4::ParserRuleContext *parent, size_t invokingState); - virtual size_t getRuleIndex() const override; - std::vector columnArgExpr(); - ColumnArgExprContext* columnArgExpr(size_t i); - std::vector COMMA(); - antlr4::tree::TerminalNode* COMMA(size_t i); - - - virtual std::any accept(antlr4::tree::ParseTreeVisitor *visitor) override; - - }; - - ColumnArgListContext* columnArgList(); - - class ColumnArgExprContext : public antlr4::ParserRuleContext { - public: - ColumnArgExprContext(antlr4::ParserRuleContext *parent, size_t invokingState); - virtual size_t getRuleIndex() const override; - ColumnLambdaExprContext *columnLambdaExpr(); - ColumnExprContext *columnExpr(); - - - virtual std::any accept(antlr4::tree::ParseTreeVisitor *visitor) override; - - }; - - ColumnArgExprContext* columnArgExpr(); - class ColumnLambdaExprContext : public antlr4::ParserRuleContext { public: ColumnLambdaExprContext(antlr4::ParserRuleContext *parent, size_t invokingState); virtual size_t getRuleIndex() const override; antlr4::tree::TerminalNode *ARROW(); - ColumnExprContext *columnExpr(); antlr4::tree::TerminalNode *LPAREN(); std::vector identifier(); IdentifierContext* identifier(size_t i); antlr4::tree::TerminalNode *RPAREN(); + ColumnExprContext *columnExpr(); + BlockContext *block(); std::vector COMMA(); antlr4::tree::TerminalNode* COMMA(size_t i); diff --git a/hogql_parser/HogQLParser.interp b/hogql_parser/HogQLParser.interp index 183059ec0fd79..b965cfbb577c7 100644 --- a/hogql_parser/HogQLParser.interp +++ b/hogql_parser/HogQLParser.interp @@ -382,8 +382,6 @@ expr columnTypeExpr columnExprList columnExpr -columnArgList -columnArgExpr columnLambdaExpr hogqlxTagElement hogqlxTagAttribute @@ -414,4 +412,4 @@ stringContentsFull atn: -[4, 1, 159, 1311, 2, 0, 7, 0, 2, 1, 7, 1, 2, 2, 7, 2, 2, 3, 7, 3, 2, 4, 7, 4, 2, 5, 7, 5, 2, 6, 7, 6, 2, 7, 7, 7, 2, 8, 7, 8, 2, 9, 7, 9, 2, 10, 7, 10, 2, 11, 7, 11, 2, 12, 7, 12, 2, 13, 7, 13, 2, 14, 7, 14, 2, 15, 7, 15, 2, 16, 7, 16, 2, 17, 7, 17, 2, 18, 7, 18, 2, 19, 7, 19, 2, 20, 7, 20, 2, 21, 7, 21, 2, 22, 7, 22, 2, 23, 7, 23, 2, 24, 7, 24, 2, 25, 7, 25, 2, 26, 7, 26, 2, 27, 7, 27, 2, 28, 7, 28, 2, 29, 7, 29, 2, 30, 7, 30, 2, 31, 7, 31, 2, 32, 7, 32, 2, 33, 7, 33, 2, 34, 7, 34, 2, 35, 7, 35, 2, 36, 7, 36, 2, 37, 7, 37, 2, 38, 7, 38, 2, 39, 7, 39, 2, 40, 7, 40, 2, 41, 7, 41, 2, 42, 7, 42, 2, 43, 7, 43, 2, 44, 7, 44, 2, 45, 7, 45, 2, 46, 7, 46, 2, 47, 7, 47, 2, 48, 7, 48, 2, 49, 7, 49, 2, 50, 7, 50, 2, 51, 7, 51, 2, 52, 7, 52, 2, 53, 7, 53, 2, 54, 7, 54, 2, 55, 7, 55, 2, 56, 7, 56, 2, 57, 7, 57, 2, 58, 7, 58, 2, 59, 7, 59, 2, 60, 7, 60, 2, 61, 7, 61, 2, 62, 7, 62, 2, 63, 7, 63, 2, 64, 7, 64, 2, 65, 7, 65, 2, 66, 7, 66, 2, 67, 7, 67, 2, 68, 7, 68, 2, 69, 7, 69, 2, 70, 7, 70, 2, 71, 7, 71, 2, 72, 7, 72, 2, 73, 7, 73, 2, 74, 7, 74, 2, 75, 7, 75, 2, 76, 7, 76, 2, 77, 7, 77, 2, 78, 7, 78, 2, 79, 7, 79, 2, 80, 7, 80, 2, 81, 7, 81, 2, 82, 7, 82, 2, 83, 7, 83, 2, 84, 7, 84, 2, 85, 7, 85, 2, 86, 7, 86, 2, 87, 7, 87, 1, 0, 5, 0, 178, 8, 0, 10, 0, 12, 0, 181, 9, 0, 1, 0, 1, 0, 1, 1, 1, 1, 3, 1, 187, 8, 1, 1, 2, 1, 2, 1, 3, 1, 3, 1, 3, 1, 3, 1, 3, 3, 3, 196, 8, 3, 1, 4, 1, 4, 1, 4, 5, 4, 201, 8, 4, 10, 4, 12, 4, 204, 9, 4, 1, 4, 3, 4, 207, 8, 4, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 3, 5, 221, 8, 5, 1, 6, 1, 6, 3, 6, 225, 8, 6, 1, 6, 3, 6, 228, 8, 6, 1, 7, 1, 7, 3, 7, 232, 8, 7, 1, 7, 3, 7, 235, 8, 7, 1, 8, 1, 8, 1, 8, 1, 8, 1, 8, 3, 8, 242, 8, 8, 1, 8, 1, 8, 3, 8, 246, 8, 8, 1, 8, 1, 8, 1, 9, 1, 9, 1, 9, 5, 9, 253, 8, 9, 10, 9, 12, 9, 256, 9, 9, 1, 9, 1, 9, 3, 9, 260, 8, 9, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 3, 10, 269, 8, 10, 1, 11, 1, 11, 1, 11, 1, 11, 1, 11, 1, 11, 3, 11, 277, 8, 11, 1, 12, 1, 12, 1, 12, 1, 12, 1, 12, 3, 12, 284, 8, 12, 1, 12, 1, 12, 3, 12, 288, 8, 12, 1, 12, 1, 12, 1, 12, 1, 12, 3, 12, 294, 8, 12, 1, 12, 1, 12, 1, 12, 3, 12, 299, 8, 12, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 3, 13, 307, 8, 13, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 3, 13, 314, 8, 13, 1, 14, 1, 14, 1, 14, 1, 14, 3, 14, 320, 8, 14, 1, 14, 1, 14, 1, 14, 1, 15, 1, 15, 1, 15, 1, 15, 1, 15, 1, 16, 1, 16, 3, 16, 332, 8, 16, 1, 17, 1, 17, 1, 18, 1, 18, 5, 18, 338, 8, 18, 10, 18, 12, 18, 341, 9, 18, 1, 18, 1, 18, 1, 19, 1, 19, 1, 19, 1, 19, 1, 20, 1, 20, 1, 20, 5, 20, 352, 8, 20, 10, 20, 12, 20, 355, 9, 20, 1, 20, 3, 20, 358, 8, 20, 1, 21, 1, 21, 1, 21, 3, 21, 363, 8, 21, 1, 21, 1, 21, 1, 22, 1, 22, 1, 22, 1, 22, 5, 22, 371, 8, 22, 10, 22, 12, 22, 374, 9, 22, 1, 23, 1, 23, 1, 23, 1, 23, 1, 23, 1, 23, 3, 23, 382, 8, 23, 1, 24, 3, 24, 385, 8, 24, 1, 24, 1, 24, 3, 24, 389, 8, 24, 1, 24, 3, 24, 392, 8, 24, 1, 24, 1, 24, 3, 24, 396, 8, 24, 1, 24, 3, 24, 399, 8, 24, 1, 24, 3, 24, 402, 8, 24, 1, 24, 3, 24, 405, 8, 24, 1, 24, 3, 24, 408, 8, 24, 1, 24, 1, 24, 3, 24, 412, 8, 24, 1, 24, 1, 24, 3, 24, 416, 8, 24, 1, 24, 3, 24, 419, 8, 24, 1, 24, 3, 24, 422, 8, 24, 1, 24, 3, 24, 425, 8, 24, 1, 24, 1, 24, 3, 24, 429, 8, 24, 1, 24, 3, 24, 432, 8, 24, 1, 25, 1, 25, 1, 25, 1, 26, 1, 26, 1, 26, 1, 26, 3, 26, 441, 8, 26, 1, 27, 1, 27, 1, 27, 1, 28, 3, 28, 447, 8, 28, 1, 28, 1, 28, 1, 28, 1, 28, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 5, 29, 466, 8, 29, 10, 29, 12, 29, 469, 9, 29, 1, 30, 1, 30, 1, 30, 1, 31, 1, 31, 1, 31, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 3, 32, 485, 8, 32, 1, 33, 1, 33, 1, 33, 1, 34, 1, 34, 1, 34, 1, 34, 1, 35, 1, 35, 1, 35, 1, 35, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 502, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 508, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 514, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 525, 8, 36, 3, 36, 527, 8, 36, 1, 37, 1, 37, 1, 37, 1, 38, 1, 38, 1, 38, 1, 39, 1, 39, 1, 39, 3, 39, 538, 8, 39, 1, 39, 3, 39, 541, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 3, 39, 547, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 1, 39, 1, 39, 3, 39, 555, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 5, 39, 561, 8, 39, 10, 39, 12, 39, 564, 9, 39, 1, 40, 3, 40, 567, 8, 40, 1, 40, 1, 40, 1, 40, 3, 40, 572, 8, 40, 1, 40, 3, 40, 575, 8, 40, 1, 40, 3, 40, 578, 8, 40, 1, 40, 1, 40, 3, 40, 582, 8, 40, 1, 40, 1, 40, 3, 40, 586, 8, 40, 1, 40, 3, 40, 589, 8, 40, 3, 40, 591, 8, 40, 1, 40, 3, 40, 594, 8, 40, 1, 40, 1, 40, 3, 40, 598, 8, 40, 1, 40, 1, 40, 3, 40, 602, 8, 40, 1, 40, 3, 40, 605, 8, 40, 3, 40, 607, 8, 40, 3, 40, 609, 8, 40, 1, 41, 1, 41, 1, 41, 3, 41, 614, 8, 41, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 3, 42, 625, 8, 42, 1, 43, 1, 43, 1, 43, 1, 43, 3, 43, 631, 8, 43, 1, 44, 1, 44, 1, 44, 5, 44, 636, 8, 44, 10, 44, 12, 44, 639, 9, 44, 1, 45, 1, 45, 3, 45, 643, 8, 45, 1, 45, 1, 45, 3, 45, 647, 8, 45, 1, 45, 1, 45, 3, 45, 651, 8, 45, 1, 46, 1, 46, 1, 46, 1, 46, 3, 46, 657, 8, 46, 3, 46, 659, 8, 46, 1, 47, 1, 47, 1, 47, 5, 47, 664, 8, 47, 10, 47, 12, 47, 667, 9, 47, 1, 48, 1, 48, 1, 48, 1, 48, 1, 49, 3, 49, 674, 8, 49, 1, 49, 3, 49, 677, 8, 49, 1, 49, 3, 49, 680, 8, 49, 1, 50, 1, 50, 1, 50, 1, 50, 1, 51, 1, 51, 1, 51, 1, 51, 1, 52, 1, 52, 1, 52, 1, 53, 1, 53, 1, 53, 1, 53, 1, 53, 1, 53, 3, 53, 699, 8, 53, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 3, 54, 713, 8, 54, 1, 55, 1, 55, 1, 55, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 727, 8, 56, 10, 56, 12, 56, 730, 9, 56, 1, 56, 3, 56, 733, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 742, 8, 56, 10, 56, 12, 56, 745, 9, 56, 1, 56, 3, 56, 748, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 757, 8, 56, 10, 56, 12, 56, 760, 9, 56, 1, 56, 3, 56, 763, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 3, 56, 770, 8, 56, 1, 56, 1, 56, 3, 56, 774, 8, 56, 1, 57, 1, 57, 1, 57, 5, 57, 779, 8, 57, 10, 57, 12, 57, 782, 9, 57, 1, 57, 3, 57, 785, 8, 57, 1, 58, 1, 58, 1, 58, 3, 58, 790, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 4, 58, 797, 8, 58, 11, 58, 12, 58, 798, 1, 58, 1, 58, 3, 58, 803, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 827, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 844, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 850, 8, 58, 1, 58, 3, 58, 853, 8, 58, 1, 58, 3, 58, 856, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 866, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 872, 8, 58, 1, 58, 3, 58, 875, 8, 58, 1, 58, 3, 58, 878, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 886, 8, 58, 1, 58, 3, 58, 889, 8, 58, 1, 58, 1, 58, 3, 58, 893, 8, 58, 1, 58, 3, 58, 896, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 910, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 927, 8, 58, 1, 58, 1, 58, 1, 58, 3, 58, 932, 8, 58, 1, 58, 1, 58, 3, 58, 936, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 942, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 949, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 961, 8, 58, 1, 58, 1, 58, 3, 58, 965, 8, 58, 1, 58, 3, 58, 968, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 977, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 991, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 1030, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 1038, 8, 58, 5, 58, 1040, 8, 58, 10, 58, 12, 58, 1043, 9, 58, 1, 59, 1, 59, 1, 59, 5, 59, 1048, 8, 59, 10, 59, 12, 59, 1051, 9, 59, 1, 59, 3, 59, 1054, 8, 59, 1, 60, 1, 60, 3, 60, 1058, 8, 60, 1, 61, 1, 61, 1, 61, 1, 61, 5, 61, 1064, 8, 61, 10, 61, 12, 61, 1067, 9, 61, 1, 61, 3, 61, 1070, 8, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 5, 61, 1077, 8, 61, 10, 61, 12, 61, 1080, 9, 61, 1, 61, 3, 61, 1083, 8, 61, 3, 61, 1085, 8, 61, 1, 61, 1, 61, 1, 61, 1, 62, 1, 62, 1, 62, 5, 62, 1093, 8, 62, 10, 62, 12, 62, 1096, 9, 62, 1, 62, 1, 62, 1, 62, 1, 62, 1, 62, 1, 62, 5, 62, 1104, 8, 62, 10, 62, 12, 62, 1107, 9, 62, 1, 62, 1, 62, 1, 62, 1, 62, 1, 62, 1, 62, 3, 62, 1115, 8, 62, 1, 62, 1, 62, 1, 62, 1, 62, 1, 62, 3, 62, 1122, 8, 62, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 3, 63, 1135, 8, 63, 1, 64, 1, 64, 1, 64, 5, 64, 1140, 8, 64, 10, 64, 12, 64, 1143, 9, 64, 1, 64, 3, 64, 1146, 8, 64, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 3, 65, 1158, 8, 65, 1, 66, 1, 66, 1, 66, 1, 66, 3, 66, 1164, 8, 66, 1, 66, 3, 66, 1167, 8, 66, 1, 67, 1, 67, 1, 67, 5, 67, 1172, 8, 67, 10, 67, 12, 67, 1175, 9, 67, 1, 68, 1, 68, 1, 68, 1, 68, 1, 68, 1, 68, 1, 68, 1, 68, 1, 68, 3, 68, 1186, 8, 68, 1, 68, 1, 68, 1, 68, 1, 68, 3, 68, 1192, 8, 68, 5, 68, 1194, 8, 68, 10, 68, 12, 68, 1197, 9, 68, 1, 69, 1, 69, 1, 69, 3, 69, 1202, 8, 69, 1, 69, 1, 69, 1, 70, 1, 70, 1, 70, 3, 70, 1209, 8, 70, 1, 70, 1, 70, 1, 71, 1, 71, 1, 71, 5, 71, 1216, 8, 71, 10, 71, 12, 71, 1219, 9, 71, 1, 71, 3, 71, 1222, 8, 71, 1, 72, 1, 72, 1, 73, 1, 73, 1, 73, 1, 73, 1, 73, 1, 73, 3, 73, 1232, 8, 73, 3, 73, 1234, 8, 73, 1, 74, 3, 74, 1237, 8, 74, 1, 74, 1, 74, 1, 74, 1, 74, 1, 74, 1, 74, 3, 74, 1245, 8, 74, 1, 75, 1, 75, 1, 75, 3, 75, 1250, 8, 75, 1, 76, 1, 76, 1, 77, 1, 77, 1, 78, 1, 78, 1, 79, 1, 79, 3, 79, 1260, 8, 79, 1, 80, 1, 80, 1, 80, 3, 80, 1265, 8, 80, 1, 81, 1, 81, 1, 81, 1, 81, 1, 82, 1, 82, 1, 82, 1, 82, 1, 83, 1, 83, 3, 83, 1277, 8, 83, 1, 84, 1, 84, 5, 84, 1281, 8, 84, 10, 84, 12, 84, 1284, 9, 84, 1, 84, 1, 84, 1, 85, 1, 85, 1, 85, 1, 85, 1, 85, 3, 85, 1293, 8, 85, 1, 86, 1, 86, 5, 86, 1297, 8, 86, 10, 86, 12, 86, 1300, 9, 86, 1, 86, 1, 86, 1, 87, 1, 87, 1, 87, 1, 87, 1, 87, 3, 87, 1309, 8, 87, 1, 87, 0, 3, 78, 116, 136, 88, 0, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28, 30, 32, 34, 36, 38, 40, 42, 44, 46, 48, 50, 52, 54, 56, 58, 60, 62, 64, 66, 68, 70, 72, 74, 76, 78, 80, 82, 84, 86, 88, 90, 92, 94, 96, 98, 100, 102, 104, 106, 108, 110, 112, 114, 116, 118, 120, 122, 124, 126, 128, 130, 132, 134, 136, 138, 140, 142, 144, 146, 148, 150, 152, 154, 156, 158, 160, 162, 164, 166, 168, 170, 172, 174, 0, 16, 2, 0, 18, 18, 74, 74, 2, 0, 44, 44, 51, 51, 3, 0, 1, 1, 4, 4, 8, 8, 4, 0, 1, 1, 3, 4, 8, 8, 80, 80, 2, 0, 51, 51, 73, 73, 2, 0, 1, 1, 4, 4, 2, 0, 7, 7, 22, 23, 2, 0, 30, 30, 49, 49, 2, 0, 71, 71, 76, 76, 3, 0, 10, 10, 50, 50, 90, 90, 2, 0, 41, 41, 53, 53, 1, 0, 107, 108, 2, 0, 118, 118, 139, 139, 7, 0, 21, 21, 38, 38, 55, 56, 70, 70, 78, 78, 97, 97, 103, 103, 16, 0, 1, 13, 15, 20, 22, 28, 30, 30, 32, 37, 39, 42, 44, 51, 53, 54, 58, 58, 60, 69, 71, 77, 79, 83, 85, 92, 94, 96, 98, 99, 101, 102, 4, 0, 20, 20, 30, 30, 39, 39, 48, 48, 1479, 0, 179, 1, 0, 0, 0, 2, 186, 1, 0, 0, 0, 4, 188, 1, 0, 0, 0, 6, 190, 1, 0, 0, 0, 8, 197, 1, 0, 0, 0, 10, 220, 1, 0, 0, 0, 12, 222, 1, 0, 0, 0, 14, 229, 1, 0, 0, 0, 16, 236, 1, 0, 0, 0, 18, 249, 1, 0, 0, 0, 20, 261, 1, 0, 0, 0, 22, 270, 1, 0, 0, 0, 24, 278, 1, 0, 0, 0, 26, 300, 1, 0, 0, 0, 28, 315, 1, 0, 0, 0, 30, 324, 1, 0, 0, 0, 32, 329, 1, 0, 0, 0, 34, 333, 1, 0, 0, 0, 36, 335, 1, 0, 0, 0, 38, 344, 1, 0, 0, 0, 40, 348, 1, 0, 0, 0, 42, 362, 1, 0, 0, 0, 44, 366, 1, 0, 0, 0, 46, 381, 1, 0, 0, 0, 48, 384, 1, 0, 0, 0, 50, 433, 1, 0, 0, 0, 52, 436, 1, 0, 0, 0, 54, 442, 1, 0, 0, 0, 56, 446, 1, 0, 0, 0, 58, 452, 1, 0, 0, 0, 60, 470, 1, 0, 0, 0, 62, 473, 1, 0, 0, 0, 64, 476, 1, 0, 0, 0, 66, 486, 1, 0, 0, 0, 68, 489, 1, 0, 0, 0, 70, 493, 1, 0, 0, 0, 72, 526, 1, 0, 0, 0, 74, 528, 1, 0, 0, 0, 76, 531, 1, 0, 0, 0, 78, 546, 1, 0, 0, 0, 80, 608, 1, 0, 0, 0, 82, 613, 1, 0, 0, 0, 84, 624, 1, 0, 0, 0, 86, 626, 1, 0, 0, 0, 88, 632, 1, 0, 0, 0, 90, 640, 1, 0, 0, 0, 92, 658, 1, 0, 0, 0, 94, 660, 1, 0, 0, 0, 96, 668, 1, 0, 0, 0, 98, 673, 1, 0, 0, 0, 100, 681, 1, 0, 0, 0, 102, 685, 1, 0, 0, 0, 104, 689, 1, 0, 0, 0, 106, 698, 1, 0, 0, 0, 108, 712, 1, 0, 0, 0, 110, 714, 1, 0, 0, 0, 112, 773, 1, 0, 0, 0, 114, 775, 1, 0, 0, 0, 116, 935, 1, 0, 0, 0, 118, 1044, 1, 0, 0, 0, 120, 1057, 1, 0, 0, 0, 122, 1084, 1, 0, 0, 0, 124, 1121, 1, 0, 0, 0, 126, 1134, 1, 0, 0, 0, 128, 1136, 1, 0, 0, 0, 130, 1157, 1, 0, 0, 0, 132, 1166, 1, 0, 0, 0, 134, 1168, 1, 0, 0, 0, 136, 1185, 1, 0, 0, 0, 138, 1198, 1, 0, 0, 0, 140, 1208, 1, 0, 0, 0, 142, 1212, 1, 0, 0, 0, 144, 1223, 1, 0, 0, 0, 146, 1233, 1, 0, 0, 0, 148, 1236, 1, 0, 0, 0, 150, 1249, 1, 0, 0, 0, 152, 1251, 1, 0, 0, 0, 154, 1253, 1, 0, 0, 0, 156, 1255, 1, 0, 0, 0, 158, 1259, 1, 0, 0, 0, 160, 1264, 1, 0, 0, 0, 162, 1266, 1, 0, 0, 0, 164, 1270, 1, 0, 0, 0, 166, 1276, 1, 0, 0, 0, 168, 1278, 1, 0, 0, 0, 170, 1292, 1, 0, 0, 0, 172, 1294, 1, 0, 0, 0, 174, 1308, 1, 0, 0, 0, 176, 178, 3, 2, 1, 0, 177, 176, 1, 0, 0, 0, 178, 181, 1, 0, 0, 0, 179, 177, 1, 0, 0, 0, 179, 180, 1, 0, 0, 0, 180, 182, 1, 0, 0, 0, 181, 179, 1, 0, 0, 0, 182, 183, 5, 0, 0, 1, 183, 1, 1, 0, 0, 0, 184, 187, 3, 6, 3, 0, 185, 187, 3, 10, 5, 0, 186, 184, 1, 0, 0, 0, 186, 185, 1, 0, 0, 0, 187, 3, 1, 0, 0, 0, 188, 189, 3, 116, 58, 0, 189, 5, 1, 0, 0, 0, 190, 191, 5, 52, 0, 0, 191, 195, 3, 160, 80, 0, 192, 193, 5, 115, 0, 0, 193, 194, 5, 122, 0, 0, 194, 196, 3, 4, 2, 0, 195, 192, 1, 0, 0, 0, 195, 196, 1, 0, 0, 0, 196, 7, 1, 0, 0, 0, 197, 202, 3, 160, 80, 0, 198, 199, 5, 116, 0, 0, 199, 201, 3, 160, 80, 0, 200, 198, 1, 0, 0, 0, 201, 204, 1, 0, 0, 0, 202, 200, 1, 0, 0, 0, 202, 203, 1, 0, 0, 0, 203, 206, 1, 0, 0, 0, 204, 202, 1, 0, 0, 0, 205, 207, 5, 116, 0, 0, 206, 205, 1, 0, 0, 0, 206, 207, 1, 0, 0, 0, 207, 9, 1, 0, 0, 0, 208, 221, 3, 12, 6, 0, 209, 221, 3, 14, 7, 0, 210, 221, 3, 18, 9, 0, 211, 221, 3, 20, 10, 0, 212, 221, 3, 22, 11, 0, 213, 221, 3, 26, 13, 0, 214, 221, 3, 24, 12, 0, 215, 221, 3, 28, 14, 0, 216, 221, 3, 30, 15, 0, 217, 221, 3, 36, 18, 0, 218, 221, 3, 32, 16, 0, 219, 221, 3, 34, 17, 0, 220, 208, 1, 0, 0, 0, 220, 209, 1, 0, 0, 0, 220, 210, 1, 0, 0, 0, 220, 211, 1, 0, 0, 0, 220, 212, 1, 0, 0, 0, 220, 213, 1, 0, 0, 0, 220, 214, 1, 0, 0, 0, 220, 215, 1, 0, 0, 0, 220, 216, 1, 0, 0, 0, 220, 217, 1, 0, 0, 0, 220, 218, 1, 0, 0, 0, 220, 219, 1, 0, 0, 0, 221, 11, 1, 0, 0, 0, 222, 224, 5, 72, 0, 0, 223, 225, 3, 4, 2, 0, 224, 223, 1, 0, 0, 0, 224, 225, 1, 0, 0, 0, 225, 227, 1, 0, 0, 0, 226, 228, 5, 150, 0, 0, 227, 226, 1, 0, 0, 0, 227, 228, 1, 0, 0, 0, 228, 13, 1, 0, 0, 0, 229, 231, 5, 84, 0, 0, 230, 232, 3, 4, 2, 0, 231, 230, 1, 0, 0, 0, 231, 232, 1, 0, 0, 0, 232, 234, 1, 0, 0, 0, 233, 235, 5, 150, 0, 0, 234, 233, 1, 0, 0, 0, 234, 235, 1, 0, 0, 0, 235, 15, 1, 0, 0, 0, 236, 245, 5, 14, 0, 0, 237, 238, 5, 130, 0, 0, 238, 241, 3, 160, 80, 0, 239, 240, 5, 115, 0, 0, 240, 242, 3, 160, 80, 0, 241, 239, 1, 0, 0, 0, 241, 242, 1, 0, 0, 0, 242, 243, 1, 0, 0, 0, 243, 244, 5, 149, 0, 0, 244, 246, 1, 0, 0, 0, 245, 237, 1, 0, 0, 0, 245, 246, 1, 0, 0, 0, 246, 247, 1, 0, 0, 0, 247, 248, 3, 36, 18, 0, 248, 17, 1, 0, 0, 0, 249, 250, 5, 93, 0, 0, 250, 254, 3, 36, 18, 0, 251, 253, 3, 16, 8, 0, 252, 251, 1, 0, 0, 0, 253, 256, 1, 0, 0, 0, 254, 252, 1, 0, 0, 0, 254, 255, 1, 0, 0, 0, 255, 259, 1, 0, 0, 0, 256, 254, 1, 0, 0, 0, 257, 258, 5, 29, 0, 0, 258, 260, 3, 36, 18, 0, 259, 257, 1, 0, 0, 0, 259, 260, 1, 0, 0, 0, 260, 19, 1, 0, 0, 0, 261, 262, 5, 40, 0, 0, 262, 263, 5, 130, 0, 0, 263, 264, 3, 4, 2, 0, 264, 265, 5, 149, 0, 0, 265, 268, 3, 10, 5, 0, 266, 267, 5, 25, 0, 0, 267, 269, 3, 10, 5, 0, 268, 266, 1, 0, 0, 0, 268, 269, 1, 0, 0, 0, 269, 21, 1, 0, 0, 0, 270, 271, 5, 100, 0, 0, 271, 272, 5, 130, 0, 0, 272, 273, 3, 4, 2, 0, 273, 274, 5, 149, 0, 0, 274, 276, 3, 10, 5, 0, 275, 277, 5, 150, 0, 0, 276, 275, 1, 0, 0, 0, 276, 277, 1, 0, 0, 0, 277, 23, 1, 0, 0, 0, 278, 279, 5, 33, 0, 0, 279, 283, 5, 130, 0, 0, 280, 284, 3, 6, 3, 0, 281, 284, 3, 30, 15, 0, 282, 284, 3, 4, 2, 0, 283, 280, 1, 0, 0, 0, 283, 281, 1, 0, 0, 0, 283, 282, 1, 0, 0, 0, 283, 284, 1, 0, 0, 0, 284, 285, 1, 0, 0, 0, 285, 287, 5, 150, 0, 0, 286, 288, 3, 4, 2, 0, 287, 286, 1, 0, 0, 0, 287, 288, 1, 0, 0, 0, 288, 289, 1, 0, 0, 0, 289, 293, 5, 150, 0, 0, 290, 294, 3, 6, 3, 0, 291, 294, 3, 30, 15, 0, 292, 294, 3, 4, 2, 0, 293, 290, 1, 0, 0, 0, 293, 291, 1, 0, 0, 0, 293, 292, 1, 0, 0, 0, 293, 294, 1, 0, 0, 0, 294, 295, 1, 0, 0, 0, 295, 296, 5, 149, 0, 0, 296, 298, 3, 10, 5, 0, 297, 299, 5, 150, 0, 0, 298, 297, 1, 0, 0, 0, 298, 299, 1, 0, 0, 0, 299, 25, 1, 0, 0, 0, 300, 301, 5, 33, 0, 0, 301, 302, 5, 130, 0, 0, 302, 303, 5, 52, 0, 0, 303, 306, 3, 160, 80, 0, 304, 305, 5, 116, 0, 0, 305, 307, 3, 160, 80, 0, 306, 304, 1, 0, 0, 0, 306, 307, 1, 0, 0, 0, 307, 308, 1, 0, 0, 0, 308, 309, 5, 42, 0, 0, 309, 310, 3, 4, 2, 0, 310, 311, 5, 149, 0, 0, 311, 313, 3, 10, 5, 0, 312, 314, 5, 150, 0, 0, 313, 312, 1, 0, 0, 0, 313, 314, 1, 0, 0, 0, 314, 27, 1, 0, 0, 0, 315, 316, 5, 31, 0, 0, 316, 317, 3, 160, 80, 0, 317, 319, 5, 130, 0, 0, 318, 320, 3, 8, 4, 0, 319, 318, 1, 0, 0, 0, 319, 320, 1, 0, 0, 0, 320, 321, 1, 0, 0, 0, 321, 322, 5, 149, 0, 0, 322, 323, 3, 36, 18, 0, 323, 29, 1, 0, 0, 0, 324, 325, 3, 4, 2, 0, 325, 326, 5, 115, 0, 0, 326, 327, 5, 122, 0, 0, 327, 328, 3, 4, 2, 0, 328, 31, 1, 0, 0, 0, 329, 331, 3, 4, 2, 0, 330, 332, 5, 150, 0, 0, 331, 330, 1, 0, 0, 0, 331, 332, 1, 0, 0, 0, 332, 33, 1, 0, 0, 0, 333, 334, 5, 150, 0, 0, 334, 35, 1, 0, 0, 0, 335, 339, 5, 128, 0, 0, 336, 338, 3, 2, 1, 0, 337, 336, 1, 0, 0, 0, 338, 341, 1, 0, 0, 0, 339, 337, 1, 0, 0, 0, 339, 340, 1, 0, 0, 0, 340, 342, 1, 0, 0, 0, 341, 339, 1, 0, 0, 0, 342, 343, 5, 147, 0, 0, 343, 37, 1, 0, 0, 0, 344, 345, 3, 4, 2, 0, 345, 346, 5, 115, 0, 0, 346, 347, 3, 4, 2, 0, 347, 39, 1, 0, 0, 0, 348, 353, 3, 38, 19, 0, 349, 350, 5, 116, 0, 0, 350, 352, 3, 38, 19, 0, 351, 349, 1, 0, 0, 0, 352, 355, 1, 0, 0, 0, 353, 351, 1, 0, 0, 0, 353, 354, 1, 0, 0, 0, 354, 357, 1, 0, 0, 0, 355, 353, 1, 0, 0, 0, 356, 358, 5, 116, 0, 0, 357, 356, 1, 0, 0, 0, 357, 358, 1, 0, 0, 0, 358, 41, 1, 0, 0, 0, 359, 363, 3, 44, 22, 0, 360, 363, 3, 48, 24, 0, 361, 363, 3, 124, 62, 0, 362, 359, 1, 0, 0, 0, 362, 360, 1, 0, 0, 0, 362, 361, 1, 0, 0, 0, 363, 364, 1, 0, 0, 0, 364, 365, 5, 0, 0, 1, 365, 43, 1, 0, 0, 0, 366, 372, 3, 46, 23, 0, 367, 368, 5, 95, 0, 0, 368, 369, 5, 1, 0, 0, 369, 371, 3, 46, 23, 0, 370, 367, 1, 0, 0, 0, 371, 374, 1, 0, 0, 0, 372, 370, 1, 0, 0, 0, 372, 373, 1, 0, 0, 0, 373, 45, 1, 0, 0, 0, 374, 372, 1, 0, 0, 0, 375, 382, 3, 48, 24, 0, 376, 377, 5, 130, 0, 0, 377, 378, 3, 44, 22, 0, 378, 379, 5, 149, 0, 0, 379, 382, 1, 0, 0, 0, 380, 382, 3, 164, 82, 0, 381, 375, 1, 0, 0, 0, 381, 376, 1, 0, 0, 0, 381, 380, 1, 0, 0, 0, 382, 47, 1, 0, 0, 0, 383, 385, 3, 50, 25, 0, 384, 383, 1, 0, 0, 0, 384, 385, 1, 0, 0, 0, 385, 386, 1, 0, 0, 0, 386, 388, 5, 79, 0, 0, 387, 389, 5, 24, 0, 0, 388, 387, 1, 0, 0, 0, 388, 389, 1, 0, 0, 0, 389, 391, 1, 0, 0, 0, 390, 392, 3, 52, 26, 0, 391, 390, 1, 0, 0, 0, 391, 392, 1, 0, 0, 0, 392, 393, 1, 0, 0, 0, 393, 395, 3, 114, 57, 0, 394, 396, 3, 54, 27, 0, 395, 394, 1, 0, 0, 0, 395, 396, 1, 0, 0, 0, 396, 398, 1, 0, 0, 0, 397, 399, 3, 56, 28, 0, 398, 397, 1, 0, 0, 0, 398, 399, 1, 0, 0, 0, 399, 401, 1, 0, 0, 0, 400, 402, 3, 60, 30, 0, 401, 400, 1, 0, 0, 0, 401, 402, 1, 0, 0, 0, 402, 404, 1, 0, 0, 0, 403, 405, 3, 62, 31, 0, 404, 403, 1, 0, 0, 0, 404, 405, 1, 0, 0, 0, 405, 407, 1, 0, 0, 0, 406, 408, 3, 64, 32, 0, 407, 406, 1, 0, 0, 0, 407, 408, 1, 0, 0, 0, 408, 411, 1, 0, 0, 0, 409, 410, 5, 102, 0, 0, 410, 412, 7, 0, 0, 0, 411, 409, 1, 0, 0, 0, 411, 412, 1, 0, 0, 0, 412, 415, 1, 0, 0, 0, 413, 414, 5, 102, 0, 0, 414, 416, 5, 89, 0, 0, 415, 413, 1, 0, 0, 0, 415, 416, 1, 0, 0, 0, 416, 418, 1, 0, 0, 0, 417, 419, 3, 66, 33, 0, 418, 417, 1, 0, 0, 0, 418, 419, 1, 0, 0, 0, 419, 421, 1, 0, 0, 0, 420, 422, 3, 58, 29, 0, 421, 420, 1, 0, 0, 0, 421, 422, 1, 0, 0, 0, 422, 424, 1, 0, 0, 0, 423, 425, 3, 68, 34, 0, 424, 423, 1, 0, 0, 0, 424, 425, 1, 0, 0, 0, 425, 428, 1, 0, 0, 0, 426, 429, 3, 72, 36, 0, 427, 429, 3, 74, 37, 0, 428, 426, 1, 0, 0, 0, 428, 427, 1, 0, 0, 0, 428, 429, 1, 0, 0, 0, 429, 431, 1, 0, 0, 0, 430, 432, 3, 76, 38, 0, 431, 430, 1, 0, 0, 0, 431, 432, 1, 0, 0, 0, 432, 49, 1, 0, 0, 0, 433, 434, 5, 102, 0, 0, 434, 435, 3, 128, 64, 0, 435, 51, 1, 0, 0, 0, 436, 437, 5, 88, 0, 0, 437, 440, 5, 108, 0, 0, 438, 439, 5, 102, 0, 0, 439, 441, 5, 85, 0, 0, 440, 438, 1, 0, 0, 0, 440, 441, 1, 0, 0, 0, 441, 53, 1, 0, 0, 0, 442, 443, 5, 34, 0, 0, 443, 444, 3, 78, 39, 0, 444, 55, 1, 0, 0, 0, 445, 447, 7, 1, 0, 0, 446, 445, 1, 0, 0, 0, 446, 447, 1, 0, 0, 0, 447, 448, 1, 0, 0, 0, 448, 449, 5, 5, 0, 0, 449, 450, 5, 47, 0, 0, 450, 451, 3, 114, 57, 0, 451, 57, 1, 0, 0, 0, 452, 453, 5, 101, 0, 0, 453, 454, 3, 160, 80, 0, 454, 455, 5, 6, 0, 0, 455, 456, 5, 130, 0, 0, 456, 457, 3, 98, 49, 0, 457, 467, 5, 149, 0, 0, 458, 459, 5, 116, 0, 0, 459, 460, 3, 160, 80, 0, 460, 461, 5, 6, 0, 0, 461, 462, 5, 130, 0, 0, 462, 463, 3, 98, 49, 0, 463, 464, 5, 149, 0, 0, 464, 466, 1, 0, 0, 0, 465, 458, 1, 0, 0, 0, 466, 469, 1, 0, 0, 0, 467, 465, 1, 0, 0, 0, 467, 468, 1, 0, 0, 0, 468, 59, 1, 0, 0, 0, 469, 467, 1, 0, 0, 0, 470, 471, 5, 69, 0, 0, 471, 472, 3, 116, 58, 0, 472, 61, 1, 0, 0, 0, 473, 474, 5, 99, 0, 0, 474, 475, 3, 116, 58, 0, 475, 63, 1, 0, 0, 0, 476, 477, 5, 36, 0, 0, 477, 484, 5, 11, 0, 0, 478, 479, 7, 0, 0, 0, 479, 480, 5, 130, 0, 0, 480, 481, 3, 114, 57, 0, 481, 482, 5, 149, 0, 0, 482, 485, 1, 0, 0, 0, 483, 485, 3, 114, 57, 0, 484, 478, 1, 0, 0, 0, 484, 483, 1, 0, 0, 0, 485, 65, 1, 0, 0, 0, 486, 487, 5, 37, 0, 0, 487, 488, 3, 116, 58, 0, 488, 67, 1, 0, 0, 0, 489, 490, 5, 64, 0, 0, 490, 491, 5, 11, 0, 0, 491, 492, 3, 88, 44, 0, 492, 69, 1, 0, 0, 0, 493, 494, 5, 64, 0, 0, 494, 495, 5, 11, 0, 0, 495, 496, 3, 114, 57, 0, 496, 71, 1, 0, 0, 0, 497, 498, 5, 54, 0, 0, 498, 501, 3, 116, 58, 0, 499, 500, 5, 116, 0, 0, 500, 502, 3, 116, 58, 0, 501, 499, 1, 0, 0, 0, 501, 502, 1, 0, 0, 0, 502, 507, 1, 0, 0, 0, 503, 504, 5, 102, 0, 0, 504, 508, 5, 85, 0, 0, 505, 506, 5, 11, 0, 0, 506, 508, 3, 114, 57, 0, 507, 503, 1, 0, 0, 0, 507, 505, 1, 0, 0, 0, 507, 508, 1, 0, 0, 0, 508, 527, 1, 0, 0, 0, 509, 510, 5, 54, 0, 0, 510, 513, 3, 116, 58, 0, 511, 512, 5, 102, 0, 0, 512, 514, 5, 85, 0, 0, 513, 511, 1, 0, 0, 0, 513, 514, 1, 0, 0, 0, 514, 515, 1, 0, 0, 0, 515, 516, 5, 61, 0, 0, 516, 517, 3, 116, 58, 0, 517, 527, 1, 0, 0, 0, 518, 519, 5, 54, 0, 0, 519, 520, 3, 116, 58, 0, 520, 521, 5, 61, 0, 0, 521, 524, 3, 116, 58, 0, 522, 523, 5, 11, 0, 0, 523, 525, 3, 114, 57, 0, 524, 522, 1, 0, 0, 0, 524, 525, 1, 0, 0, 0, 525, 527, 1, 0, 0, 0, 526, 497, 1, 0, 0, 0, 526, 509, 1, 0, 0, 0, 526, 518, 1, 0, 0, 0, 527, 73, 1, 0, 0, 0, 528, 529, 5, 61, 0, 0, 529, 530, 3, 116, 58, 0, 530, 75, 1, 0, 0, 0, 531, 532, 5, 81, 0, 0, 532, 533, 3, 94, 47, 0, 533, 77, 1, 0, 0, 0, 534, 535, 6, 39, -1, 0, 535, 537, 3, 136, 68, 0, 536, 538, 5, 28, 0, 0, 537, 536, 1, 0, 0, 0, 537, 538, 1, 0, 0, 0, 538, 540, 1, 0, 0, 0, 539, 541, 3, 86, 43, 0, 540, 539, 1, 0, 0, 0, 540, 541, 1, 0, 0, 0, 541, 547, 1, 0, 0, 0, 542, 543, 5, 130, 0, 0, 543, 544, 3, 78, 39, 0, 544, 545, 5, 149, 0, 0, 545, 547, 1, 0, 0, 0, 546, 534, 1, 0, 0, 0, 546, 542, 1, 0, 0, 0, 547, 562, 1, 0, 0, 0, 548, 549, 10, 3, 0, 0, 549, 550, 3, 82, 41, 0, 550, 551, 3, 78, 39, 4, 551, 561, 1, 0, 0, 0, 552, 554, 10, 4, 0, 0, 553, 555, 3, 80, 40, 0, 554, 553, 1, 0, 0, 0, 554, 555, 1, 0, 0, 0, 555, 556, 1, 0, 0, 0, 556, 557, 5, 47, 0, 0, 557, 558, 3, 78, 39, 0, 558, 559, 3, 84, 42, 0, 559, 561, 1, 0, 0, 0, 560, 548, 1, 0, 0, 0, 560, 552, 1, 0, 0, 0, 561, 564, 1, 0, 0, 0, 562, 560, 1, 0, 0, 0, 562, 563, 1, 0, 0, 0, 563, 79, 1, 0, 0, 0, 564, 562, 1, 0, 0, 0, 565, 567, 7, 2, 0, 0, 566, 565, 1, 0, 0, 0, 566, 567, 1, 0, 0, 0, 567, 568, 1, 0, 0, 0, 568, 575, 5, 44, 0, 0, 569, 571, 5, 44, 0, 0, 570, 572, 7, 2, 0, 0, 571, 570, 1, 0, 0, 0, 571, 572, 1, 0, 0, 0, 572, 575, 1, 0, 0, 0, 573, 575, 7, 2, 0, 0, 574, 566, 1, 0, 0, 0, 574, 569, 1, 0, 0, 0, 574, 573, 1, 0, 0, 0, 575, 609, 1, 0, 0, 0, 576, 578, 7, 3, 0, 0, 577, 576, 1, 0, 0, 0, 577, 578, 1, 0, 0, 0, 578, 579, 1, 0, 0, 0, 579, 581, 7, 4, 0, 0, 580, 582, 5, 65, 0, 0, 581, 580, 1, 0, 0, 0, 581, 582, 1, 0, 0, 0, 582, 591, 1, 0, 0, 0, 583, 585, 7, 4, 0, 0, 584, 586, 5, 65, 0, 0, 585, 584, 1, 0, 0, 0, 585, 586, 1, 0, 0, 0, 586, 588, 1, 0, 0, 0, 587, 589, 7, 3, 0, 0, 588, 587, 1, 0, 0, 0, 588, 589, 1, 0, 0, 0, 589, 591, 1, 0, 0, 0, 590, 577, 1, 0, 0, 0, 590, 583, 1, 0, 0, 0, 591, 609, 1, 0, 0, 0, 592, 594, 7, 5, 0, 0, 593, 592, 1, 0, 0, 0, 593, 594, 1, 0, 0, 0, 594, 595, 1, 0, 0, 0, 595, 597, 5, 35, 0, 0, 596, 598, 5, 65, 0, 0, 597, 596, 1, 0, 0, 0, 597, 598, 1, 0, 0, 0, 598, 607, 1, 0, 0, 0, 599, 601, 5, 35, 0, 0, 600, 602, 5, 65, 0, 0, 601, 600, 1, 0, 0, 0, 601, 602, 1, 0, 0, 0, 602, 604, 1, 0, 0, 0, 603, 605, 7, 5, 0, 0, 604, 603, 1, 0, 0, 0, 604, 605, 1, 0, 0, 0, 605, 607, 1, 0, 0, 0, 606, 593, 1, 0, 0, 0, 606, 599, 1, 0, 0, 0, 607, 609, 1, 0, 0, 0, 608, 574, 1, 0, 0, 0, 608, 590, 1, 0, 0, 0, 608, 606, 1, 0, 0, 0, 609, 81, 1, 0, 0, 0, 610, 611, 5, 17, 0, 0, 611, 614, 5, 47, 0, 0, 612, 614, 5, 116, 0, 0, 613, 610, 1, 0, 0, 0, 613, 612, 1, 0, 0, 0, 614, 83, 1, 0, 0, 0, 615, 616, 5, 62, 0, 0, 616, 625, 3, 114, 57, 0, 617, 618, 5, 96, 0, 0, 618, 619, 5, 130, 0, 0, 619, 620, 3, 114, 57, 0, 620, 621, 5, 149, 0, 0, 621, 625, 1, 0, 0, 0, 622, 623, 5, 96, 0, 0, 623, 625, 3, 114, 57, 0, 624, 615, 1, 0, 0, 0, 624, 617, 1, 0, 0, 0, 624, 622, 1, 0, 0, 0, 625, 85, 1, 0, 0, 0, 626, 627, 5, 77, 0, 0, 627, 630, 3, 92, 46, 0, 628, 629, 5, 61, 0, 0, 629, 631, 3, 92, 46, 0, 630, 628, 1, 0, 0, 0, 630, 631, 1, 0, 0, 0, 631, 87, 1, 0, 0, 0, 632, 637, 3, 90, 45, 0, 633, 634, 5, 116, 0, 0, 634, 636, 3, 90, 45, 0, 635, 633, 1, 0, 0, 0, 636, 639, 1, 0, 0, 0, 637, 635, 1, 0, 0, 0, 637, 638, 1, 0, 0, 0, 638, 89, 1, 0, 0, 0, 639, 637, 1, 0, 0, 0, 640, 642, 3, 116, 58, 0, 641, 643, 7, 6, 0, 0, 642, 641, 1, 0, 0, 0, 642, 643, 1, 0, 0, 0, 643, 646, 1, 0, 0, 0, 644, 645, 5, 60, 0, 0, 645, 647, 7, 7, 0, 0, 646, 644, 1, 0, 0, 0, 646, 647, 1, 0, 0, 0, 647, 650, 1, 0, 0, 0, 648, 649, 5, 16, 0, 0, 649, 651, 5, 110, 0, 0, 650, 648, 1, 0, 0, 0, 650, 651, 1, 0, 0, 0, 651, 91, 1, 0, 0, 0, 652, 659, 3, 164, 82, 0, 653, 656, 3, 148, 74, 0, 654, 655, 5, 151, 0, 0, 655, 657, 3, 148, 74, 0, 656, 654, 1, 0, 0, 0, 656, 657, 1, 0, 0, 0, 657, 659, 1, 0, 0, 0, 658, 652, 1, 0, 0, 0, 658, 653, 1, 0, 0, 0, 659, 93, 1, 0, 0, 0, 660, 665, 3, 96, 48, 0, 661, 662, 5, 116, 0, 0, 662, 664, 3, 96, 48, 0, 663, 661, 1, 0, 0, 0, 664, 667, 1, 0, 0, 0, 665, 663, 1, 0, 0, 0, 665, 666, 1, 0, 0, 0, 666, 95, 1, 0, 0, 0, 667, 665, 1, 0, 0, 0, 668, 669, 3, 160, 80, 0, 669, 670, 5, 122, 0, 0, 670, 671, 3, 150, 75, 0, 671, 97, 1, 0, 0, 0, 672, 674, 3, 100, 50, 0, 673, 672, 1, 0, 0, 0, 673, 674, 1, 0, 0, 0, 674, 676, 1, 0, 0, 0, 675, 677, 3, 102, 51, 0, 676, 675, 1, 0, 0, 0, 676, 677, 1, 0, 0, 0, 677, 679, 1, 0, 0, 0, 678, 680, 3, 104, 52, 0, 679, 678, 1, 0, 0, 0, 679, 680, 1, 0, 0, 0, 680, 99, 1, 0, 0, 0, 681, 682, 5, 67, 0, 0, 682, 683, 5, 11, 0, 0, 683, 684, 3, 114, 57, 0, 684, 101, 1, 0, 0, 0, 685, 686, 5, 64, 0, 0, 686, 687, 5, 11, 0, 0, 687, 688, 3, 88, 44, 0, 688, 103, 1, 0, 0, 0, 689, 690, 7, 8, 0, 0, 690, 691, 3, 106, 53, 0, 691, 105, 1, 0, 0, 0, 692, 699, 3, 108, 54, 0, 693, 694, 5, 9, 0, 0, 694, 695, 3, 108, 54, 0, 695, 696, 5, 2, 0, 0, 696, 697, 3, 108, 54, 0, 697, 699, 1, 0, 0, 0, 698, 692, 1, 0, 0, 0, 698, 693, 1, 0, 0, 0, 699, 107, 1, 0, 0, 0, 700, 701, 5, 19, 0, 0, 701, 713, 5, 75, 0, 0, 702, 703, 5, 94, 0, 0, 703, 713, 5, 68, 0, 0, 704, 705, 5, 94, 0, 0, 705, 713, 5, 32, 0, 0, 706, 707, 3, 148, 74, 0, 707, 708, 5, 68, 0, 0, 708, 713, 1, 0, 0, 0, 709, 710, 3, 148, 74, 0, 710, 711, 5, 32, 0, 0, 711, 713, 1, 0, 0, 0, 712, 700, 1, 0, 0, 0, 712, 702, 1, 0, 0, 0, 712, 704, 1, 0, 0, 0, 712, 706, 1, 0, 0, 0, 712, 709, 1, 0, 0, 0, 713, 109, 1, 0, 0, 0, 714, 715, 3, 116, 58, 0, 715, 716, 5, 0, 0, 1, 716, 111, 1, 0, 0, 0, 717, 774, 3, 160, 80, 0, 718, 719, 3, 160, 80, 0, 719, 720, 5, 130, 0, 0, 720, 721, 3, 160, 80, 0, 721, 728, 3, 112, 56, 0, 722, 723, 5, 116, 0, 0, 723, 724, 3, 160, 80, 0, 724, 725, 3, 112, 56, 0, 725, 727, 1, 0, 0, 0, 726, 722, 1, 0, 0, 0, 727, 730, 1, 0, 0, 0, 728, 726, 1, 0, 0, 0, 728, 729, 1, 0, 0, 0, 729, 732, 1, 0, 0, 0, 730, 728, 1, 0, 0, 0, 731, 733, 5, 116, 0, 0, 732, 731, 1, 0, 0, 0, 732, 733, 1, 0, 0, 0, 733, 734, 1, 0, 0, 0, 734, 735, 5, 149, 0, 0, 735, 774, 1, 0, 0, 0, 736, 737, 3, 160, 80, 0, 737, 738, 5, 130, 0, 0, 738, 743, 3, 162, 81, 0, 739, 740, 5, 116, 0, 0, 740, 742, 3, 162, 81, 0, 741, 739, 1, 0, 0, 0, 742, 745, 1, 0, 0, 0, 743, 741, 1, 0, 0, 0, 743, 744, 1, 0, 0, 0, 744, 747, 1, 0, 0, 0, 745, 743, 1, 0, 0, 0, 746, 748, 5, 116, 0, 0, 747, 746, 1, 0, 0, 0, 747, 748, 1, 0, 0, 0, 748, 749, 1, 0, 0, 0, 749, 750, 5, 149, 0, 0, 750, 774, 1, 0, 0, 0, 751, 752, 3, 160, 80, 0, 752, 753, 5, 130, 0, 0, 753, 758, 3, 112, 56, 0, 754, 755, 5, 116, 0, 0, 755, 757, 3, 112, 56, 0, 756, 754, 1, 0, 0, 0, 757, 760, 1, 0, 0, 0, 758, 756, 1, 0, 0, 0, 758, 759, 1, 0, 0, 0, 759, 762, 1, 0, 0, 0, 760, 758, 1, 0, 0, 0, 761, 763, 5, 116, 0, 0, 762, 761, 1, 0, 0, 0, 762, 763, 1, 0, 0, 0, 763, 764, 1, 0, 0, 0, 764, 765, 5, 149, 0, 0, 765, 774, 1, 0, 0, 0, 766, 767, 3, 160, 80, 0, 767, 769, 5, 130, 0, 0, 768, 770, 3, 114, 57, 0, 769, 768, 1, 0, 0, 0, 769, 770, 1, 0, 0, 0, 770, 771, 1, 0, 0, 0, 771, 772, 5, 149, 0, 0, 772, 774, 1, 0, 0, 0, 773, 717, 1, 0, 0, 0, 773, 718, 1, 0, 0, 0, 773, 736, 1, 0, 0, 0, 773, 751, 1, 0, 0, 0, 773, 766, 1, 0, 0, 0, 774, 113, 1, 0, 0, 0, 775, 780, 3, 116, 58, 0, 776, 777, 5, 116, 0, 0, 777, 779, 3, 116, 58, 0, 778, 776, 1, 0, 0, 0, 779, 782, 1, 0, 0, 0, 780, 778, 1, 0, 0, 0, 780, 781, 1, 0, 0, 0, 781, 784, 1, 0, 0, 0, 782, 780, 1, 0, 0, 0, 783, 785, 5, 116, 0, 0, 784, 783, 1, 0, 0, 0, 784, 785, 1, 0, 0, 0, 785, 115, 1, 0, 0, 0, 786, 787, 6, 58, -1, 0, 787, 789, 5, 12, 0, 0, 788, 790, 3, 116, 58, 0, 789, 788, 1, 0, 0, 0, 789, 790, 1, 0, 0, 0, 790, 796, 1, 0, 0, 0, 791, 792, 5, 98, 0, 0, 792, 793, 3, 116, 58, 0, 793, 794, 5, 83, 0, 0, 794, 795, 3, 116, 58, 0, 795, 797, 1, 0, 0, 0, 796, 791, 1, 0, 0, 0, 797, 798, 1, 0, 0, 0, 798, 796, 1, 0, 0, 0, 798, 799, 1, 0, 0, 0, 799, 802, 1, 0, 0, 0, 800, 801, 5, 25, 0, 0, 801, 803, 3, 116, 58, 0, 802, 800, 1, 0, 0, 0, 802, 803, 1, 0, 0, 0, 803, 804, 1, 0, 0, 0, 804, 805, 5, 26, 0, 0, 805, 936, 1, 0, 0, 0, 806, 807, 5, 13, 0, 0, 807, 808, 5, 130, 0, 0, 808, 809, 3, 116, 58, 0, 809, 810, 5, 6, 0, 0, 810, 811, 3, 112, 56, 0, 811, 812, 5, 149, 0, 0, 812, 936, 1, 0, 0, 0, 813, 814, 5, 20, 0, 0, 814, 936, 5, 110, 0, 0, 815, 816, 5, 45, 0, 0, 816, 817, 3, 116, 58, 0, 817, 818, 3, 152, 76, 0, 818, 936, 1, 0, 0, 0, 819, 820, 5, 82, 0, 0, 820, 821, 5, 130, 0, 0, 821, 822, 3, 116, 58, 0, 822, 823, 5, 34, 0, 0, 823, 826, 3, 116, 58, 0, 824, 825, 5, 33, 0, 0, 825, 827, 3, 116, 58, 0, 826, 824, 1, 0, 0, 0, 826, 827, 1, 0, 0, 0, 827, 828, 1, 0, 0, 0, 828, 829, 5, 149, 0, 0, 829, 936, 1, 0, 0, 0, 830, 831, 5, 86, 0, 0, 831, 936, 5, 110, 0, 0, 832, 833, 5, 91, 0, 0, 833, 834, 5, 130, 0, 0, 834, 835, 7, 9, 0, 0, 835, 836, 3, 166, 83, 0, 836, 837, 5, 34, 0, 0, 837, 838, 3, 116, 58, 0, 838, 839, 5, 149, 0, 0, 839, 936, 1, 0, 0, 0, 840, 841, 3, 160, 80, 0, 841, 843, 5, 130, 0, 0, 842, 844, 3, 114, 57, 0, 843, 842, 1, 0, 0, 0, 843, 844, 1, 0, 0, 0, 844, 845, 1, 0, 0, 0, 845, 846, 5, 149, 0, 0, 846, 855, 1, 0, 0, 0, 847, 849, 5, 130, 0, 0, 848, 850, 5, 24, 0, 0, 849, 848, 1, 0, 0, 0, 849, 850, 1, 0, 0, 0, 850, 852, 1, 0, 0, 0, 851, 853, 3, 118, 59, 0, 852, 851, 1, 0, 0, 0, 852, 853, 1, 0, 0, 0, 853, 854, 1, 0, 0, 0, 854, 856, 5, 149, 0, 0, 855, 847, 1, 0, 0, 0, 855, 856, 1, 0, 0, 0, 856, 857, 1, 0, 0, 0, 857, 858, 5, 66, 0, 0, 858, 859, 5, 130, 0, 0, 859, 860, 3, 98, 49, 0, 860, 861, 5, 149, 0, 0, 861, 936, 1, 0, 0, 0, 862, 863, 3, 160, 80, 0, 863, 865, 5, 130, 0, 0, 864, 866, 3, 114, 57, 0, 865, 864, 1, 0, 0, 0, 865, 866, 1, 0, 0, 0, 866, 867, 1, 0, 0, 0, 867, 868, 5, 149, 0, 0, 868, 877, 1, 0, 0, 0, 869, 871, 5, 130, 0, 0, 870, 872, 5, 24, 0, 0, 871, 870, 1, 0, 0, 0, 871, 872, 1, 0, 0, 0, 872, 874, 1, 0, 0, 0, 873, 875, 3, 118, 59, 0, 874, 873, 1, 0, 0, 0, 874, 875, 1, 0, 0, 0, 875, 876, 1, 0, 0, 0, 876, 878, 5, 149, 0, 0, 877, 869, 1, 0, 0, 0, 877, 878, 1, 0, 0, 0, 878, 879, 1, 0, 0, 0, 879, 880, 5, 66, 0, 0, 880, 881, 3, 160, 80, 0, 881, 936, 1, 0, 0, 0, 882, 888, 3, 160, 80, 0, 883, 885, 5, 130, 0, 0, 884, 886, 3, 114, 57, 0, 885, 884, 1, 0, 0, 0, 885, 886, 1, 0, 0, 0, 886, 887, 1, 0, 0, 0, 887, 889, 5, 149, 0, 0, 888, 883, 1, 0, 0, 0, 888, 889, 1, 0, 0, 0, 889, 890, 1, 0, 0, 0, 890, 892, 5, 130, 0, 0, 891, 893, 5, 24, 0, 0, 892, 891, 1, 0, 0, 0, 892, 893, 1, 0, 0, 0, 893, 895, 1, 0, 0, 0, 894, 896, 3, 118, 59, 0, 895, 894, 1, 0, 0, 0, 895, 896, 1, 0, 0, 0, 896, 897, 1, 0, 0, 0, 897, 898, 5, 149, 0, 0, 898, 936, 1, 0, 0, 0, 899, 936, 3, 124, 62, 0, 900, 936, 3, 168, 84, 0, 901, 936, 3, 150, 75, 0, 902, 903, 5, 118, 0, 0, 903, 936, 3, 116, 58, 19, 904, 905, 5, 58, 0, 0, 905, 936, 3, 116, 58, 13, 906, 907, 3, 140, 70, 0, 907, 908, 5, 120, 0, 0, 908, 910, 1, 0, 0, 0, 909, 906, 1, 0, 0, 0, 909, 910, 1, 0, 0, 0, 910, 911, 1, 0, 0, 0, 911, 936, 5, 112, 0, 0, 912, 913, 5, 130, 0, 0, 913, 914, 3, 44, 22, 0, 914, 915, 5, 149, 0, 0, 915, 936, 1, 0, 0, 0, 916, 917, 5, 130, 0, 0, 917, 918, 3, 116, 58, 0, 918, 919, 5, 149, 0, 0, 919, 936, 1, 0, 0, 0, 920, 921, 5, 130, 0, 0, 921, 922, 3, 114, 57, 0, 922, 923, 5, 149, 0, 0, 923, 936, 1, 0, 0, 0, 924, 926, 5, 129, 0, 0, 925, 927, 3, 114, 57, 0, 926, 925, 1, 0, 0, 0, 926, 927, 1, 0, 0, 0, 927, 928, 1, 0, 0, 0, 928, 936, 5, 148, 0, 0, 929, 931, 5, 128, 0, 0, 930, 932, 3, 40, 20, 0, 931, 930, 1, 0, 0, 0, 931, 932, 1, 0, 0, 0, 932, 933, 1, 0, 0, 0, 933, 936, 5, 147, 0, 0, 934, 936, 3, 132, 66, 0, 935, 786, 1, 0, 0, 0, 935, 806, 1, 0, 0, 0, 935, 813, 1, 0, 0, 0, 935, 815, 1, 0, 0, 0, 935, 819, 1, 0, 0, 0, 935, 830, 1, 0, 0, 0, 935, 832, 1, 0, 0, 0, 935, 840, 1, 0, 0, 0, 935, 862, 1, 0, 0, 0, 935, 882, 1, 0, 0, 0, 935, 899, 1, 0, 0, 0, 935, 900, 1, 0, 0, 0, 935, 901, 1, 0, 0, 0, 935, 902, 1, 0, 0, 0, 935, 904, 1, 0, 0, 0, 935, 909, 1, 0, 0, 0, 935, 912, 1, 0, 0, 0, 935, 916, 1, 0, 0, 0, 935, 920, 1, 0, 0, 0, 935, 924, 1, 0, 0, 0, 935, 929, 1, 0, 0, 0, 935, 934, 1, 0, 0, 0, 936, 1041, 1, 0, 0, 0, 937, 941, 10, 18, 0, 0, 938, 942, 5, 112, 0, 0, 939, 942, 5, 151, 0, 0, 940, 942, 5, 138, 0, 0, 941, 938, 1, 0, 0, 0, 941, 939, 1, 0, 0, 0, 941, 940, 1, 0, 0, 0, 942, 943, 1, 0, 0, 0, 943, 1040, 3, 116, 58, 19, 944, 948, 10, 17, 0, 0, 945, 949, 5, 139, 0, 0, 946, 949, 5, 118, 0, 0, 947, 949, 5, 117, 0, 0, 948, 945, 1, 0, 0, 0, 948, 946, 1, 0, 0, 0, 948, 947, 1, 0, 0, 0, 949, 950, 1, 0, 0, 0, 950, 1040, 3, 116, 58, 18, 951, 976, 10, 16, 0, 0, 952, 977, 5, 121, 0, 0, 953, 977, 5, 122, 0, 0, 954, 977, 5, 133, 0, 0, 955, 977, 5, 131, 0, 0, 956, 977, 5, 132, 0, 0, 957, 977, 5, 123, 0, 0, 958, 977, 5, 124, 0, 0, 959, 961, 5, 58, 0, 0, 960, 959, 1, 0, 0, 0, 960, 961, 1, 0, 0, 0, 961, 962, 1, 0, 0, 0, 962, 964, 5, 42, 0, 0, 963, 965, 5, 15, 0, 0, 964, 963, 1, 0, 0, 0, 964, 965, 1, 0, 0, 0, 965, 977, 1, 0, 0, 0, 966, 968, 5, 58, 0, 0, 967, 966, 1, 0, 0, 0, 967, 968, 1, 0, 0, 0, 968, 969, 1, 0, 0, 0, 969, 977, 7, 10, 0, 0, 970, 977, 5, 145, 0, 0, 971, 977, 5, 146, 0, 0, 972, 977, 5, 135, 0, 0, 973, 977, 5, 126, 0, 0, 974, 977, 5, 127, 0, 0, 975, 977, 5, 134, 0, 0, 976, 952, 1, 0, 0, 0, 976, 953, 1, 0, 0, 0, 976, 954, 1, 0, 0, 0, 976, 955, 1, 0, 0, 0, 976, 956, 1, 0, 0, 0, 976, 957, 1, 0, 0, 0, 976, 958, 1, 0, 0, 0, 976, 960, 1, 0, 0, 0, 976, 967, 1, 0, 0, 0, 976, 970, 1, 0, 0, 0, 976, 971, 1, 0, 0, 0, 976, 972, 1, 0, 0, 0, 976, 973, 1, 0, 0, 0, 976, 974, 1, 0, 0, 0, 976, 975, 1, 0, 0, 0, 977, 978, 1, 0, 0, 0, 978, 1040, 3, 116, 58, 17, 979, 980, 10, 14, 0, 0, 980, 981, 5, 137, 0, 0, 981, 1040, 3, 116, 58, 15, 982, 983, 10, 12, 0, 0, 983, 984, 5, 2, 0, 0, 984, 1040, 3, 116, 58, 13, 985, 986, 10, 11, 0, 0, 986, 987, 5, 63, 0, 0, 987, 1040, 3, 116, 58, 12, 988, 990, 10, 10, 0, 0, 989, 991, 5, 58, 0, 0, 990, 989, 1, 0, 0, 0, 990, 991, 1, 0, 0, 0, 991, 992, 1, 0, 0, 0, 992, 993, 5, 9, 0, 0, 993, 994, 3, 116, 58, 0, 994, 995, 5, 2, 0, 0, 995, 996, 3, 116, 58, 11, 996, 1040, 1, 0, 0, 0, 997, 998, 10, 9, 0, 0, 998, 999, 5, 140, 0, 0, 999, 1000, 3, 116, 58, 0, 1000, 1001, 5, 115, 0, 0, 1001, 1002, 3, 116, 58, 9, 1002, 1040, 1, 0, 0, 0, 1003, 1004, 10, 25, 0, 0, 1004, 1005, 5, 129, 0, 0, 1005, 1006, 3, 116, 58, 0, 1006, 1007, 5, 148, 0, 0, 1007, 1040, 1, 0, 0, 0, 1008, 1009, 10, 24, 0, 0, 1009, 1010, 5, 120, 0, 0, 1010, 1040, 5, 108, 0, 0, 1011, 1012, 10, 23, 0, 0, 1012, 1013, 5, 120, 0, 0, 1013, 1040, 3, 160, 80, 0, 1014, 1015, 10, 22, 0, 0, 1015, 1016, 5, 136, 0, 0, 1016, 1017, 5, 129, 0, 0, 1017, 1018, 3, 116, 58, 0, 1018, 1019, 5, 148, 0, 0, 1019, 1040, 1, 0, 0, 0, 1020, 1021, 10, 21, 0, 0, 1021, 1022, 5, 136, 0, 0, 1022, 1040, 5, 108, 0, 0, 1023, 1024, 10, 20, 0, 0, 1024, 1025, 5, 136, 0, 0, 1025, 1040, 3, 160, 80, 0, 1026, 1027, 10, 15, 0, 0, 1027, 1029, 5, 46, 0, 0, 1028, 1030, 5, 58, 0, 0, 1029, 1028, 1, 0, 0, 0, 1029, 1030, 1, 0, 0, 0, 1030, 1031, 1, 0, 0, 0, 1031, 1040, 5, 59, 0, 0, 1032, 1037, 10, 8, 0, 0, 1033, 1034, 5, 6, 0, 0, 1034, 1038, 3, 160, 80, 0, 1035, 1036, 5, 6, 0, 0, 1036, 1038, 5, 110, 0, 0, 1037, 1033, 1, 0, 0, 0, 1037, 1035, 1, 0, 0, 0, 1038, 1040, 1, 0, 0, 0, 1039, 937, 1, 0, 0, 0, 1039, 944, 1, 0, 0, 0, 1039, 951, 1, 0, 0, 0, 1039, 979, 1, 0, 0, 0, 1039, 982, 1, 0, 0, 0, 1039, 985, 1, 0, 0, 0, 1039, 988, 1, 0, 0, 0, 1039, 997, 1, 0, 0, 0, 1039, 1003, 1, 0, 0, 0, 1039, 1008, 1, 0, 0, 0, 1039, 1011, 1, 0, 0, 0, 1039, 1014, 1, 0, 0, 0, 1039, 1020, 1, 0, 0, 0, 1039, 1023, 1, 0, 0, 0, 1039, 1026, 1, 0, 0, 0, 1039, 1032, 1, 0, 0, 0, 1040, 1043, 1, 0, 0, 0, 1041, 1039, 1, 0, 0, 0, 1041, 1042, 1, 0, 0, 0, 1042, 117, 1, 0, 0, 0, 1043, 1041, 1, 0, 0, 0, 1044, 1049, 3, 120, 60, 0, 1045, 1046, 5, 116, 0, 0, 1046, 1048, 3, 120, 60, 0, 1047, 1045, 1, 0, 0, 0, 1048, 1051, 1, 0, 0, 0, 1049, 1047, 1, 0, 0, 0, 1049, 1050, 1, 0, 0, 0, 1050, 1053, 1, 0, 0, 0, 1051, 1049, 1, 0, 0, 0, 1052, 1054, 5, 116, 0, 0, 1053, 1052, 1, 0, 0, 0, 1053, 1054, 1, 0, 0, 0, 1054, 119, 1, 0, 0, 0, 1055, 1058, 3, 122, 61, 0, 1056, 1058, 3, 116, 58, 0, 1057, 1055, 1, 0, 0, 0, 1057, 1056, 1, 0, 0, 0, 1058, 121, 1, 0, 0, 0, 1059, 1060, 5, 130, 0, 0, 1060, 1065, 3, 160, 80, 0, 1061, 1062, 5, 116, 0, 0, 1062, 1064, 3, 160, 80, 0, 1063, 1061, 1, 0, 0, 0, 1064, 1067, 1, 0, 0, 0, 1065, 1063, 1, 0, 0, 0, 1065, 1066, 1, 0, 0, 0, 1066, 1069, 1, 0, 0, 0, 1067, 1065, 1, 0, 0, 0, 1068, 1070, 5, 116, 0, 0, 1069, 1068, 1, 0, 0, 0, 1069, 1070, 1, 0, 0, 0, 1070, 1071, 1, 0, 0, 0, 1071, 1072, 5, 149, 0, 0, 1072, 1085, 1, 0, 0, 0, 1073, 1078, 3, 160, 80, 0, 1074, 1075, 5, 116, 0, 0, 1075, 1077, 3, 160, 80, 0, 1076, 1074, 1, 0, 0, 0, 1077, 1080, 1, 0, 0, 0, 1078, 1076, 1, 0, 0, 0, 1078, 1079, 1, 0, 0, 0, 1079, 1082, 1, 0, 0, 0, 1080, 1078, 1, 0, 0, 0, 1081, 1083, 5, 116, 0, 0, 1082, 1081, 1, 0, 0, 0, 1082, 1083, 1, 0, 0, 0, 1083, 1085, 1, 0, 0, 0, 1084, 1059, 1, 0, 0, 0, 1084, 1073, 1, 0, 0, 0, 1085, 1086, 1, 0, 0, 0, 1086, 1087, 5, 111, 0, 0, 1087, 1088, 3, 116, 58, 0, 1088, 123, 1, 0, 0, 0, 1089, 1090, 5, 132, 0, 0, 1090, 1094, 3, 160, 80, 0, 1091, 1093, 3, 126, 63, 0, 1092, 1091, 1, 0, 0, 0, 1093, 1096, 1, 0, 0, 0, 1094, 1092, 1, 0, 0, 0, 1094, 1095, 1, 0, 0, 0, 1095, 1097, 1, 0, 0, 0, 1096, 1094, 1, 0, 0, 0, 1097, 1098, 5, 151, 0, 0, 1098, 1099, 5, 124, 0, 0, 1099, 1122, 1, 0, 0, 0, 1100, 1101, 5, 132, 0, 0, 1101, 1105, 3, 160, 80, 0, 1102, 1104, 3, 126, 63, 0, 1103, 1102, 1, 0, 0, 0, 1104, 1107, 1, 0, 0, 0, 1105, 1103, 1, 0, 0, 0, 1105, 1106, 1, 0, 0, 0, 1106, 1108, 1, 0, 0, 0, 1107, 1105, 1, 0, 0, 0, 1108, 1114, 5, 124, 0, 0, 1109, 1115, 3, 124, 62, 0, 1110, 1111, 5, 128, 0, 0, 1111, 1112, 3, 116, 58, 0, 1112, 1113, 5, 147, 0, 0, 1113, 1115, 1, 0, 0, 0, 1114, 1109, 1, 0, 0, 0, 1114, 1110, 1, 0, 0, 0, 1114, 1115, 1, 0, 0, 0, 1115, 1116, 1, 0, 0, 0, 1116, 1117, 5, 132, 0, 0, 1117, 1118, 5, 151, 0, 0, 1118, 1119, 3, 160, 80, 0, 1119, 1120, 5, 124, 0, 0, 1120, 1122, 1, 0, 0, 0, 1121, 1089, 1, 0, 0, 0, 1121, 1100, 1, 0, 0, 0, 1122, 125, 1, 0, 0, 0, 1123, 1124, 3, 160, 80, 0, 1124, 1125, 5, 122, 0, 0, 1125, 1126, 3, 166, 83, 0, 1126, 1135, 1, 0, 0, 0, 1127, 1128, 3, 160, 80, 0, 1128, 1129, 5, 122, 0, 0, 1129, 1130, 5, 128, 0, 0, 1130, 1131, 3, 116, 58, 0, 1131, 1132, 5, 147, 0, 0, 1132, 1135, 1, 0, 0, 0, 1133, 1135, 3, 160, 80, 0, 1134, 1123, 1, 0, 0, 0, 1134, 1127, 1, 0, 0, 0, 1134, 1133, 1, 0, 0, 0, 1135, 127, 1, 0, 0, 0, 1136, 1141, 3, 130, 65, 0, 1137, 1138, 5, 116, 0, 0, 1138, 1140, 3, 130, 65, 0, 1139, 1137, 1, 0, 0, 0, 1140, 1143, 1, 0, 0, 0, 1141, 1139, 1, 0, 0, 0, 1141, 1142, 1, 0, 0, 0, 1142, 1145, 1, 0, 0, 0, 1143, 1141, 1, 0, 0, 0, 1144, 1146, 5, 116, 0, 0, 1145, 1144, 1, 0, 0, 0, 1145, 1146, 1, 0, 0, 0, 1146, 129, 1, 0, 0, 0, 1147, 1148, 3, 160, 80, 0, 1148, 1149, 5, 6, 0, 0, 1149, 1150, 5, 130, 0, 0, 1150, 1151, 3, 44, 22, 0, 1151, 1152, 5, 149, 0, 0, 1152, 1158, 1, 0, 0, 0, 1153, 1154, 3, 116, 58, 0, 1154, 1155, 5, 6, 0, 0, 1155, 1156, 3, 160, 80, 0, 1156, 1158, 1, 0, 0, 0, 1157, 1147, 1, 0, 0, 0, 1157, 1153, 1, 0, 0, 0, 1158, 131, 1, 0, 0, 0, 1159, 1167, 3, 164, 82, 0, 1160, 1161, 3, 140, 70, 0, 1161, 1162, 5, 120, 0, 0, 1162, 1164, 1, 0, 0, 0, 1163, 1160, 1, 0, 0, 0, 1163, 1164, 1, 0, 0, 0, 1164, 1165, 1, 0, 0, 0, 1165, 1167, 3, 134, 67, 0, 1166, 1159, 1, 0, 0, 0, 1166, 1163, 1, 0, 0, 0, 1167, 133, 1, 0, 0, 0, 1168, 1173, 3, 160, 80, 0, 1169, 1170, 5, 120, 0, 0, 1170, 1172, 3, 160, 80, 0, 1171, 1169, 1, 0, 0, 0, 1172, 1175, 1, 0, 0, 0, 1173, 1171, 1, 0, 0, 0, 1173, 1174, 1, 0, 0, 0, 1174, 135, 1, 0, 0, 0, 1175, 1173, 1, 0, 0, 0, 1176, 1177, 6, 68, -1, 0, 1177, 1186, 3, 140, 70, 0, 1178, 1186, 3, 138, 69, 0, 1179, 1180, 5, 130, 0, 0, 1180, 1181, 3, 44, 22, 0, 1181, 1182, 5, 149, 0, 0, 1182, 1186, 1, 0, 0, 0, 1183, 1186, 3, 124, 62, 0, 1184, 1186, 3, 164, 82, 0, 1185, 1176, 1, 0, 0, 0, 1185, 1178, 1, 0, 0, 0, 1185, 1179, 1, 0, 0, 0, 1185, 1183, 1, 0, 0, 0, 1185, 1184, 1, 0, 0, 0, 1186, 1195, 1, 0, 0, 0, 1187, 1191, 10, 3, 0, 0, 1188, 1192, 3, 158, 79, 0, 1189, 1190, 5, 6, 0, 0, 1190, 1192, 3, 160, 80, 0, 1191, 1188, 1, 0, 0, 0, 1191, 1189, 1, 0, 0, 0, 1192, 1194, 1, 0, 0, 0, 1193, 1187, 1, 0, 0, 0, 1194, 1197, 1, 0, 0, 0, 1195, 1193, 1, 0, 0, 0, 1195, 1196, 1, 0, 0, 0, 1196, 137, 1, 0, 0, 0, 1197, 1195, 1, 0, 0, 0, 1198, 1199, 3, 160, 80, 0, 1199, 1201, 5, 130, 0, 0, 1200, 1202, 3, 142, 71, 0, 1201, 1200, 1, 0, 0, 0, 1201, 1202, 1, 0, 0, 0, 1202, 1203, 1, 0, 0, 0, 1203, 1204, 5, 149, 0, 0, 1204, 139, 1, 0, 0, 0, 1205, 1206, 3, 144, 72, 0, 1206, 1207, 5, 120, 0, 0, 1207, 1209, 1, 0, 0, 0, 1208, 1205, 1, 0, 0, 0, 1208, 1209, 1, 0, 0, 0, 1209, 1210, 1, 0, 0, 0, 1210, 1211, 3, 160, 80, 0, 1211, 141, 1, 0, 0, 0, 1212, 1217, 3, 116, 58, 0, 1213, 1214, 5, 116, 0, 0, 1214, 1216, 3, 116, 58, 0, 1215, 1213, 1, 0, 0, 0, 1216, 1219, 1, 0, 0, 0, 1217, 1215, 1, 0, 0, 0, 1217, 1218, 1, 0, 0, 0, 1218, 1221, 1, 0, 0, 0, 1219, 1217, 1, 0, 0, 0, 1220, 1222, 5, 116, 0, 0, 1221, 1220, 1, 0, 0, 0, 1221, 1222, 1, 0, 0, 0, 1222, 143, 1, 0, 0, 0, 1223, 1224, 3, 160, 80, 0, 1224, 145, 1, 0, 0, 0, 1225, 1234, 5, 106, 0, 0, 1226, 1227, 5, 120, 0, 0, 1227, 1234, 7, 11, 0, 0, 1228, 1229, 5, 108, 0, 0, 1229, 1231, 5, 120, 0, 0, 1230, 1232, 7, 11, 0, 0, 1231, 1230, 1, 0, 0, 0, 1231, 1232, 1, 0, 0, 0, 1232, 1234, 1, 0, 0, 0, 1233, 1225, 1, 0, 0, 0, 1233, 1226, 1, 0, 0, 0, 1233, 1228, 1, 0, 0, 0, 1234, 147, 1, 0, 0, 0, 1235, 1237, 7, 12, 0, 0, 1236, 1235, 1, 0, 0, 0, 1236, 1237, 1, 0, 0, 0, 1237, 1244, 1, 0, 0, 0, 1238, 1245, 3, 146, 73, 0, 1239, 1245, 5, 107, 0, 0, 1240, 1245, 5, 108, 0, 0, 1241, 1245, 5, 109, 0, 0, 1242, 1245, 5, 43, 0, 0, 1243, 1245, 5, 57, 0, 0, 1244, 1238, 1, 0, 0, 0, 1244, 1239, 1, 0, 0, 0, 1244, 1240, 1, 0, 0, 0, 1244, 1241, 1, 0, 0, 0, 1244, 1242, 1, 0, 0, 0, 1244, 1243, 1, 0, 0, 0, 1245, 149, 1, 0, 0, 0, 1246, 1250, 3, 148, 74, 0, 1247, 1250, 5, 110, 0, 0, 1248, 1250, 5, 59, 0, 0, 1249, 1246, 1, 0, 0, 0, 1249, 1247, 1, 0, 0, 0, 1249, 1248, 1, 0, 0, 0, 1250, 151, 1, 0, 0, 0, 1251, 1252, 7, 13, 0, 0, 1252, 153, 1, 0, 0, 0, 1253, 1254, 7, 14, 0, 0, 1254, 155, 1, 0, 0, 0, 1255, 1256, 7, 15, 0, 0, 1256, 157, 1, 0, 0, 0, 1257, 1260, 5, 105, 0, 0, 1258, 1260, 3, 156, 78, 0, 1259, 1257, 1, 0, 0, 0, 1259, 1258, 1, 0, 0, 0, 1260, 159, 1, 0, 0, 0, 1261, 1265, 5, 105, 0, 0, 1262, 1265, 3, 152, 76, 0, 1263, 1265, 3, 154, 77, 0, 1264, 1261, 1, 0, 0, 0, 1264, 1262, 1, 0, 0, 0, 1264, 1263, 1, 0, 0, 0, 1265, 161, 1, 0, 0, 0, 1266, 1267, 3, 166, 83, 0, 1267, 1268, 5, 122, 0, 0, 1268, 1269, 3, 148, 74, 0, 1269, 163, 1, 0, 0, 0, 1270, 1271, 5, 128, 0, 0, 1271, 1272, 3, 134, 67, 0, 1272, 1273, 5, 147, 0, 0, 1273, 165, 1, 0, 0, 0, 1274, 1277, 5, 110, 0, 0, 1275, 1277, 3, 168, 84, 0, 1276, 1274, 1, 0, 0, 0, 1276, 1275, 1, 0, 0, 0, 1277, 167, 1, 0, 0, 0, 1278, 1282, 5, 142, 0, 0, 1279, 1281, 3, 170, 85, 0, 1280, 1279, 1, 0, 0, 0, 1281, 1284, 1, 0, 0, 0, 1282, 1280, 1, 0, 0, 0, 1282, 1283, 1, 0, 0, 0, 1283, 1285, 1, 0, 0, 0, 1284, 1282, 1, 0, 0, 0, 1285, 1286, 5, 144, 0, 0, 1286, 169, 1, 0, 0, 0, 1287, 1288, 5, 157, 0, 0, 1288, 1289, 3, 116, 58, 0, 1289, 1290, 5, 147, 0, 0, 1290, 1293, 1, 0, 0, 0, 1291, 1293, 5, 156, 0, 0, 1292, 1287, 1, 0, 0, 0, 1292, 1291, 1, 0, 0, 0, 1293, 171, 1, 0, 0, 0, 1294, 1298, 5, 143, 0, 0, 1295, 1297, 3, 174, 87, 0, 1296, 1295, 1, 0, 0, 0, 1297, 1300, 1, 0, 0, 0, 1298, 1296, 1, 0, 0, 0, 1298, 1299, 1, 0, 0, 0, 1299, 1301, 1, 0, 0, 0, 1300, 1298, 1, 0, 0, 0, 1301, 1302, 5, 0, 0, 1, 1302, 173, 1, 0, 0, 0, 1303, 1304, 5, 159, 0, 0, 1304, 1305, 3, 116, 58, 0, 1305, 1306, 5, 147, 0, 0, 1306, 1309, 1, 0, 0, 0, 1307, 1309, 5, 158, 0, 0, 1308, 1303, 1, 0, 0, 0, 1308, 1307, 1, 0, 0, 0, 1309, 175, 1, 0, 0, 0, 168, 179, 186, 195, 202, 206, 220, 224, 227, 231, 234, 241, 245, 254, 259, 268, 276, 283, 287, 293, 298, 306, 313, 319, 331, 339, 353, 357, 362, 372, 381, 384, 388, 391, 395, 398, 401, 404, 407, 411, 415, 418, 421, 424, 428, 431, 440, 446, 467, 484, 501, 507, 513, 524, 526, 537, 540, 546, 554, 560, 562, 566, 571, 574, 577, 581, 585, 588, 590, 593, 597, 601, 604, 606, 608, 613, 624, 630, 637, 642, 646, 650, 656, 658, 665, 673, 676, 679, 698, 712, 728, 732, 743, 747, 758, 762, 769, 773, 780, 784, 789, 798, 802, 826, 843, 849, 852, 855, 865, 871, 874, 877, 885, 888, 892, 895, 909, 926, 931, 935, 941, 948, 960, 964, 967, 976, 990, 1029, 1037, 1039, 1041, 1049, 1053, 1057, 1065, 1069, 1078, 1082, 1084, 1094, 1105, 1114, 1121, 1134, 1141, 1145, 1157, 1163, 1166, 1173, 1185, 1191, 1195, 1201, 1208, 1217, 1221, 1231, 1233, 1236, 1244, 1249, 1259, 1264, 1276, 1282, 1292, 1298, 1308] \ No newline at end of file +[4, 1, 159, 1303, 2, 0, 7, 0, 2, 1, 7, 1, 2, 2, 7, 2, 2, 3, 7, 3, 2, 4, 7, 4, 2, 5, 7, 5, 2, 6, 7, 6, 2, 7, 7, 7, 2, 8, 7, 8, 2, 9, 7, 9, 2, 10, 7, 10, 2, 11, 7, 11, 2, 12, 7, 12, 2, 13, 7, 13, 2, 14, 7, 14, 2, 15, 7, 15, 2, 16, 7, 16, 2, 17, 7, 17, 2, 18, 7, 18, 2, 19, 7, 19, 2, 20, 7, 20, 2, 21, 7, 21, 2, 22, 7, 22, 2, 23, 7, 23, 2, 24, 7, 24, 2, 25, 7, 25, 2, 26, 7, 26, 2, 27, 7, 27, 2, 28, 7, 28, 2, 29, 7, 29, 2, 30, 7, 30, 2, 31, 7, 31, 2, 32, 7, 32, 2, 33, 7, 33, 2, 34, 7, 34, 2, 35, 7, 35, 2, 36, 7, 36, 2, 37, 7, 37, 2, 38, 7, 38, 2, 39, 7, 39, 2, 40, 7, 40, 2, 41, 7, 41, 2, 42, 7, 42, 2, 43, 7, 43, 2, 44, 7, 44, 2, 45, 7, 45, 2, 46, 7, 46, 2, 47, 7, 47, 2, 48, 7, 48, 2, 49, 7, 49, 2, 50, 7, 50, 2, 51, 7, 51, 2, 52, 7, 52, 2, 53, 7, 53, 2, 54, 7, 54, 2, 55, 7, 55, 2, 56, 7, 56, 2, 57, 7, 57, 2, 58, 7, 58, 2, 59, 7, 59, 2, 60, 7, 60, 2, 61, 7, 61, 2, 62, 7, 62, 2, 63, 7, 63, 2, 64, 7, 64, 2, 65, 7, 65, 2, 66, 7, 66, 2, 67, 7, 67, 2, 68, 7, 68, 2, 69, 7, 69, 2, 70, 7, 70, 2, 71, 7, 71, 2, 72, 7, 72, 2, 73, 7, 73, 2, 74, 7, 74, 2, 75, 7, 75, 2, 76, 7, 76, 2, 77, 7, 77, 2, 78, 7, 78, 2, 79, 7, 79, 2, 80, 7, 80, 2, 81, 7, 81, 2, 82, 7, 82, 2, 83, 7, 83, 2, 84, 7, 84, 2, 85, 7, 85, 1, 0, 5, 0, 174, 8, 0, 10, 0, 12, 0, 177, 9, 0, 1, 0, 1, 0, 1, 1, 1, 1, 3, 1, 183, 8, 1, 1, 2, 1, 2, 1, 3, 1, 3, 1, 3, 1, 3, 1, 3, 3, 3, 192, 8, 3, 1, 4, 1, 4, 1, 4, 5, 4, 197, 8, 4, 10, 4, 12, 4, 200, 9, 4, 1, 4, 3, 4, 203, 8, 4, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 3, 5, 217, 8, 5, 1, 6, 1, 6, 3, 6, 221, 8, 6, 1, 6, 3, 6, 224, 8, 6, 1, 7, 1, 7, 3, 7, 228, 8, 7, 1, 7, 3, 7, 231, 8, 7, 1, 8, 1, 8, 1, 8, 1, 8, 1, 8, 3, 8, 238, 8, 8, 1, 8, 1, 8, 3, 8, 242, 8, 8, 1, 8, 1, 8, 1, 9, 1, 9, 1, 9, 5, 9, 249, 8, 9, 10, 9, 12, 9, 252, 9, 9, 1, 9, 1, 9, 3, 9, 256, 8, 9, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 3, 10, 265, 8, 10, 1, 11, 1, 11, 1, 11, 1, 11, 1, 11, 1, 11, 3, 11, 273, 8, 11, 1, 12, 1, 12, 1, 12, 1, 12, 1, 12, 3, 12, 280, 8, 12, 1, 12, 1, 12, 3, 12, 284, 8, 12, 1, 12, 1, 12, 1, 12, 1, 12, 3, 12, 290, 8, 12, 1, 12, 1, 12, 1, 12, 3, 12, 295, 8, 12, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 3, 13, 303, 8, 13, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 3, 13, 310, 8, 13, 1, 14, 1, 14, 1, 14, 1, 14, 3, 14, 316, 8, 14, 1, 14, 1, 14, 1, 14, 1, 15, 1, 15, 1, 15, 1, 15, 1, 15, 1, 16, 1, 16, 3, 16, 328, 8, 16, 1, 17, 1, 17, 1, 18, 1, 18, 5, 18, 334, 8, 18, 10, 18, 12, 18, 337, 9, 18, 1, 18, 1, 18, 1, 19, 1, 19, 1, 19, 1, 19, 1, 20, 1, 20, 1, 20, 5, 20, 348, 8, 20, 10, 20, 12, 20, 351, 9, 20, 1, 20, 3, 20, 354, 8, 20, 1, 21, 1, 21, 1, 21, 3, 21, 359, 8, 21, 1, 21, 1, 21, 1, 22, 1, 22, 1, 22, 1, 22, 5, 22, 367, 8, 22, 10, 22, 12, 22, 370, 9, 22, 1, 23, 1, 23, 1, 23, 1, 23, 1, 23, 1, 23, 3, 23, 378, 8, 23, 1, 24, 3, 24, 381, 8, 24, 1, 24, 1, 24, 3, 24, 385, 8, 24, 1, 24, 3, 24, 388, 8, 24, 1, 24, 1, 24, 3, 24, 392, 8, 24, 1, 24, 3, 24, 395, 8, 24, 1, 24, 3, 24, 398, 8, 24, 1, 24, 3, 24, 401, 8, 24, 1, 24, 3, 24, 404, 8, 24, 1, 24, 1, 24, 3, 24, 408, 8, 24, 1, 24, 1, 24, 3, 24, 412, 8, 24, 1, 24, 3, 24, 415, 8, 24, 1, 24, 3, 24, 418, 8, 24, 1, 24, 3, 24, 421, 8, 24, 1, 24, 1, 24, 3, 24, 425, 8, 24, 1, 24, 3, 24, 428, 8, 24, 1, 25, 1, 25, 1, 25, 1, 26, 1, 26, 1, 26, 1, 26, 3, 26, 437, 8, 26, 1, 27, 1, 27, 1, 27, 1, 28, 3, 28, 443, 8, 28, 1, 28, 1, 28, 1, 28, 1, 28, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 5, 29, 462, 8, 29, 10, 29, 12, 29, 465, 9, 29, 1, 30, 1, 30, 1, 30, 1, 31, 1, 31, 1, 31, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 3, 32, 481, 8, 32, 1, 33, 1, 33, 1, 33, 1, 34, 1, 34, 1, 34, 1, 34, 1, 35, 1, 35, 1, 35, 1, 35, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 498, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 504, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 510, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 521, 8, 36, 3, 36, 523, 8, 36, 1, 37, 1, 37, 1, 37, 1, 38, 1, 38, 1, 38, 1, 39, 1, 39, 1, 39, 3, 39, 534, 8, 39, 1, 39, 3, 39, 537, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 3, 39, 543, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 1, 39, 1, 39, 3, 39, 551, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 5, 39, 557, 8, 39, 10, 39, 12, 39, 560, 9, 39, 1, 40, 3, 40, 563, 8, 40, 1, 40, 1, 40, 1, 40, 3, 40, 568, 8, 40, 1, 40, 3, 40, 571, 8, 40, 1, 40, 3, 40, 574, 8, 40, 1, 40, 1, 40, 3, 40, 578, 8, 40, 1, 40, 1, 40, 3, 40, 582, 8, 40, 1, 40, 3, 40, 585, 8, 40, 3, 40, 587, 8, 40, 1, 40, 3, 40, 590, 8, 40, 1, 40, 1, 40, 3, 40, 594, 8, 40, 1, 40, 1, 40, 3, 40, 598, 8, 40, 1, 40, 3, 40, 601, 8, 40, 3, 40, 603, 8, 40, 3, 40, 605, 8, 40, 1, 41, 1, 41, 1, 41, 3, 41, 610, 8, 41, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 3, 42, 621, 8, 42, 1, 43, 1, 43, 1, 43, 1, 43, 3, 43, 627, 8, 43, 1, 44, 1, 44, 1, 44, 5, 44, 632, 8, 44, 10, 44, 12, 44, 635, 9, 44, 1, 45, 1, 45, 3, 45, 639, 8, 45, 1, 45, 1, 45, 3, 45, 643, 8, 45, 1, 45, 1, 45, 3, 45, 647, 8, 45, 1, 46, 1, 46, 1, 46, 1, 46, 3, 46, 653, 8, 46, 3, 46, 655, 8, 46, 1, 47, 1, 47, 1, 47, 5, 47, 660, 8, 47, 10, 47, 12, 47, 663, 9, 47, 1, 48, 1, 48, 1, 48, 1, 48, 1, 49, 3, 49, 670, 8, 49, 1, 49, 3, 49, 673, 8, 49, 1, 49, 3, 49, 676, 8, 49, 1, 50, 1, 50, 1, 50, 1, 50, 1, 51, 1, 51, 1, 51, 1, 51, 1, 52, 1, 52, 1, 52, 1, 53, 1, 53, 1, 53, 1, 53, 1, 53, 1, 53, 3, 53, 695, 8, 53, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 3, 54, 709, 8, 54, 1, 55, 1, 55, 1, 55, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 723, 8, 56, 10, 56, 12, 56, 726, 9, 56, 1, 56, 3, 56, 729, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 738, 8, 56, 10, 56, 12, 56, 741, 9, 56, 1, 56, 3, 56, 744, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 753, 8, 56, 10, 56, 12, 56, 756, 9, 56, 1, 56, 3, 56, 759, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 3, 56, 766, 8, 56, 1, 56, 1, 56, 3, 56, 770, 8, 56, 1, 57, 1, 57, 1, 57, 5, 57, 775, 8, 57, 10, 57, 12, 57, 778, 9, 57, 1, 57, 3, 57, 781, 8, 57, 1, 58, 1, 58, 1, 58, 3, 58, 786, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 4, 58, 793, 8, 58, 11, 58, 12, 58, 794, 1, 58, 1, 58, 3, 58, 799, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 823, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 840, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 846, 8, 58, 1, 58, 3, 58, 849, 8, 58, 1, 58, 3, 58, 852, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 862, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 868, 8, 58, 1, 58, 3, 58, 871, 8, 58, 1, 58, 3, 58, 874, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 882, 8, 58, 1, 58, 3, 58, 885, 8, 58, 1, 58, 1, 58, 3, 58, 889, 8, 58, 1, 58, 3, 58, 892, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 906, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 923, 8, 58, 1, 58, 1, 58, 1, 58, 3, 58, 928, 8, 58, 1, 58, 1, 58, 1, 58, 3, 58, 933, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 939, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 946, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 958, 8, 58, 1, 58, 1, 58, 3, 58, 962, 8, 58, 1, 58, 3, 58, 965, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 974, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 988, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 1004, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 1033, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 1041, 8, 58, 5, 58, 1043, 8, 58, 10, 58, 12, 58, 1046, 9, 58, 1, 59, 1, 59, 1, 59, 1, 59, 5, 59, 1052, 8, 59, 10, 59, 12, 59, 1055, 9, 59, 1, 59, 3, 59, 1058, 8, 59, 1, 59, 1, 59, 1, 59, 1, 59, 1, 59, 5, 59, 1065, 8, 59, 10, 59, 12, 59, 1068, 9, 59, 1, 59, 3, 59, 1071, 8, 59, 1, 59, 1, 59, 3, 59, 1075, 8, 59, 1, 59, 1, 59, 1, 59, 3, 59, 1080, 8, 59, 1, 60, 1, 60, 1, 60, 5, 60, 1085, 8, 60, 10, 60, 12, 60, 1088, 9, 60, 1, 60, 1, 60, 1, 60, 1, 60, 1, 60, 1, 60, 5, 60, 1096, 8, 60, 10, 60, 12, 60, 1099, 9, 60, 1, 60, 1, 60, 1, 60, 1, 60, 1, 60, 1, 60, 3, 60, 1107, 8, 60, 1, 60, 1, 60, 1, 60, 1, 60, 1, 60, 3, 60, 1114, 8, 60, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 3, 61, 1127, 8, 61, 1, 62, 1, 62, 1, 62, 5, 62, 1132, 8, 62, 10, 62, 12, 62, 1135, 9, 62, 1, 62, 3, 62, 1138, 8, 62, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 3, 63, 1150, 8, 63, 1, 64, 1, 64, 1, 64, 1, 64, 3, 64, 1156, 8, 64, 1, 64, 3, 64, 1159, 8, 64, 1, 65, 1, 65, 1, 65, 5, 65, 1164, 8, 65, 10, 65, 12, 65, 1167, 9, 65, 1, 66, 1, 66, 1, 66, 1, 66, 1, 66, 1, 66, 1, 66, 1, 66, 1, 66, 3, 66, 1178, 8, 66, 1, 66, 1, 66, 1, 66, 1, 66, 3, 66, 1184, 8, 66, 5, 66, 1186, 8, 66, 10, 66, 12, 66, 1189, 9, 66, 1, 67, 1, 67, 1, 67, 3, 67, 1194, 8, 67, 1, 67, 1, 67, 1, 68, 1, 68, 1, 68, 3, 68, 1201, 8, 68, 1, 68, 1, 68, 1, 69, 1, 69, 1, 69, 5, 69, 1208, 8, 69, 10, 69, 12, 69, 1211, 9, 69, 1, 69, 3, 69, 1214, 8, 69, 1, 70, 1, 70, 1, 71, 1, 71, 1, 71, 1, 71, 1, 71, 1, 71, 3, 71, 1224, 8, 71, 3, 71, 1226, 8, 71, 1, 72, 3, 72, 1229, 8, 72, 1, 72, 1, 72, 1, 72, 1, 72, 1, 72, 1, 72, 3, 72, 1237, 8, 72, 1, 73, 1, 73, 1, 73, 3, 73, 1242, 8, 73, 1, 74, 1, 74, 1, 75, 1, 75, 1, 76, 1, 76, 1, 77, 1, 77, 3, 77, 1252, 8, 77, 1, 78, 1, 78, 1, 78, 3, 78, 1257, 8, 78, 1, 79, 1, 79, 1, 79, 1, 79, 1, 80, 1, 80, 1, 80, 1, 80, 1, 81, 1, 81, 3, 81, 1269, 8, 81, 1, 82, 1, 82, 5, 82, 1273, 8, 82, 10, 82, 12, 82, 1276, 9, 82, 1, 82, 1, 82, 1, 83, 1, 83, 1, 83, 1, 83, 1, 83, 3, 83, 1285, 8, 83, 1, 84, 1, 84, 5, 84, 1289, 8, 84, 10, 84, 12, 84, 1292, 9, 84, 1, 84, 1, 84, 1, 85, 1, 85, 1, 85, 1, 85, 1, 85, 3, 85, 1301, 8, 85, 1, 85, 0, 3, 78, 116, 132, 86, 0, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28, 30, 32, 34, 36, 38, 40, 42, 44, 46, 48, 50, 52, 54, 56, 58, 60, 62, 64, 66, 68, 70, 72, 74, 76, 78, 80, 82, 84, 86, 88, 90, 92, 94, 96, 98, 100, 102, 104, 106, 108, 110, 112, 114, 116, 118, 120, 122, 124, 126, 128, 130, 132, 134, 136, 138, 140, 142, 144, 146, 148, 150, 152, 154, 156, 158, 160, 162, 164, 166, 168, 170, 0, 16, 2, 0, 18, 18, 74, 74, 2, 0, 44, 44, 51, 51, 3, 0, 1, 1, 4, 4, 8, 8, 4, 0, 1, 1, 3, 4, 8, 8, 80, 80, 2, 0, 51, 51, 73, 73, 2, 0, 1, 1, 4, 4, 2, 0, 7, 7, 22, 23, 2, 0, 30, 30, 49, 49, 2, 0, 71, 71, 76, 76, 3, 0, 10, 10, 50, 50, 90, 90, 2, 0, 41, 41, 53, 53, 1, 0, 107, 108, 2, 0, 118, 118, 139, 139, 7, 0, 21, 21, 38, 38, 55, 56, 70, 70, 78, 78, 97, 97, 103, 103, 16, 0, 1, 13, 15, 20, 22, 28, 30, 30, 32, 37, 39, 42, 44, 51, 53, 54, 58, 58, 60, 69, 71, 77, 79, 83, 85, 92, 94, 96, 98, 99, 101, 102, 4, 0, 20, 20, 30, 30, 39, 39, 48, 48, 1475, 0, 175, 1, 0, 0, 0, 2, 182, 1, 0, 0, 0, 4, 184, 1, 0, 0, 0, 6, 186, 1, 0, 0, 0, 8, 193, 1, 0, 0, 0, 10, 216, 1, 0, 0, 0, 12, 218, 1, 0, 0, 0, 14, 225, 1, 0, 0, 0, 16, 232, 1, 0, 0, 0, 18, 245, 1, 0, 0, 0, 20, 257, 1, 0, 0, 0, 22, 266, 1, 0, 0, 0, 24, 274, 1, 0, 0, 0, 26, 296, 1, 0, 0, 0, 28, 311, 1, 0, 0, 0, 30, 320, 1, 0, 0, 0, 32, 325, 1, 0, 0, 0, 34, 329, 1, 0, 0, 0, 36, 331, 1, 0, 0, 0, 38, 340, 1, 0, 0, 0, 40, 344, 1, 0, 0, 0, 42, 358, 1, 0, 0, 0, 44, 362, 1, 0, 0, 0, 46, 377, 1, 0, 0, 0, 48, 380, 1, 0, 0, 0, 50, 429, 1, 0, 0, 0, 52, 432, 1, 0, 0, 0, 54, 438, 1, 0, 0, 0, 56, 442, 1, 0, 0, 0, 58, 448, 1, 0, 0, 0, 60, 466, 1, 0, 0, 0, 62, 469, 1, 0, 0, 0, 64, 472, 1, 0, 0, 0, 66, 482, 1, 0, 0, 0, 68, 485, 1, 0, 0, 0, 70, 489, 1, 0, 0, 0, 72, 522, 1, 0, 0, 0, 74, 524, 1, 0, 0, 0, 76, 527, 1, 0, 0, 0, 78, 542, 1, 0, 0, 0, 80, 604, 1, 0, 0, 0, 82, 609, 1, 0, 0, 0, 84, 620, 1, 0, 0, 0, 86, 622, 1, 0, 0, 0, 88, 628, 1, 0, 0, 0, 90, 636, 1, 0, 0, 0, 92, 654, 1, 0, 0, 0, 94, 656, 1, 0, 0, 0, 96, 664, 1, 0, 0, 0, 98, 669, 1, 0, 0, 0, 100, 677, 1, 0, 0, 0, 102, 681, 1, 0, 0, 0, 104, 685, 1, 0, 0, 0, 106, 694, 1, 0, 0, 0, 108, 708, 1, 0, 0, 0, 110, 710, 1, 0, 0, 0, 112, 769, 1, 0, 0, 0, 114, 771, 1, 0, 0, 0, 116, 932, 1, 0, 0, 0, 118, 1074, 1, 0, 0, 0, 120, 1113, 1, 0, 0, 0, 122, 1126, 1, 0, 0, 0, 124, 1128, 1, 0, 0, 0, 126, 1149, 1, 0, 0, 0, 128, 1158, 1, 0, 0, 0, 130, 1160, 1, 0, 0, 0, 132, 1177, 1, 0, 0, 0, 134, 1190, 1, 0, 0, 0, 136, 1200, 1, 0, 0, 0, 138, 1204, 1, 0, 0, 0, 140, 1215, 1, 0, 0, 0, 142, 1225, 1, 0, 0, 0, 144, 1228, 1, 0, 0, 0, 146, 1241, 1, 0, 0, 0, 148, 1243, 1, 0, 0, 0, 150, 1245, 1, 0, 0, 0, 152, 1247, 1, 0, 0, 0, 154, 1251, 1, 0, 0, 0, 156, 1256, 1, 0, 0, 0, 158, 1258, 1, 0, 0, 0, 160, 1262, 1, 0, 0, 0, 162, 1268, 1, 0, 0, 0, 164, 1270, 1, 0, 0, 0, 166, 1284, 1, 0, 0, 0, 168, 1286, 1, 0, 0, 0, 170, 1300, 1, 0, 0, 0, 172, 174, 3, 2, 1, 0, 173, 172, 1, 0, 0, 0, 174, 177, 1, 0, 0, 0, 175, 173, 1, 0, 0, 0, 175, 176, 1, 0, 0, 0, 176, 178, 1, 0, 0, 0, 177, 175, 1, 0, 0, 0, 178, 179, 5, 0, 0, 1, 179, 1, 1, 0, 0, 0, 180, 183, 3, 6, 3, 0, 181, 183, 3, 10, 5, 0, 182, 180, 1, 0, 0, 0, 182, 181, 1, 0, 0, 0, 183, 3, 1, 0, 0, 0, 184, 185, 3, 116, 58, 0, 185, 5, 1, 0, 0, 0, 186, 187, 5, 52, 0, 0, 187, 191, 3, 156, 78, 0, 188, 189, 5, 115, 0, 0, 189, 190, 5, 122, 0, 0, 190, 192, 3, 4, 2, 0, 191, 188, 1, 0, 0, 0, 191, 192, 1, 0, 0, 0, 192, 7, 1, 0, 0, 0, 193, 198, 3, 156, 78, 0, 194, 195, 5, 116, 0, 0, 195, 197, 3, 156, 78, 0, 196, 194, 1, 0, 0, 0, 197, 200, 1, 0, 0, 0, 198, 196, 1, 0, 0, 0, 198, 199, 1, 0, 0, 0, 199, 202, 1, 0, 0, 0, 200, 198, 1, 0, 0, 0, 201, 203, 5, 116, 0, 0, 202, 201, 1, 0, 0, 0, 202, 203, 1, 0, 0, 0, 203, 9, 1, 0, 0, 0, 204, 217, 3, 12, 6, 0, 205, 217, 3, 14, 7, 0, 206, 217, 3, 18, 9, 0, 207, 217, 3, 20, 10, 0, 208, 217, 3, 22, 11, 0, 209, 217, 3, 26, 13, 0, 210, 217, 3, 24, 12, 0, 211, 217, 3, 28, 14, 0, 212, 217, 3, 30, 15, 0, 213, 217, 3, 36, 18, 0, 214, 217, 3, 32, 16, 0, 215, 217, 3, 34, 17, 0, 216, 204, 1, 0, 0, 0, 216, 205, 1, 0, 0, 0, 216, 206, 1, 0, 0, 0, 216, 207, 1, 0, 0, 0, 216, 208, 1, 0, 0, 0, 216, 209, 1, 0, 0, 0, 216, 210, 1, 0, 0, 0, 216, 211, 1, 0, 0, 0, 216, 212, 1, 0, 0, 0, 216, 213, 1, 0, 0, 0, 216, 214, 1, 0, 0, 0, 216, 215, 1, 0, 0, 0, 217, 11, 1, 0, 0, 0, 218, 220, 5, 72, 0, 0, 219, 221, 3, 4, 2, 0, 220, 219, 1, 0, 0, 0, 220, 221, 1, 0, 0, 0, 221, 223, 1, 0, 0, 0, 222, 224, 5, 150, 0, 0, 223, 222, 1, 0, 0, 0, 223, 224, 1, 0, 0, 0, 224, 13, 1, 0, 0, 0, 225, 227, 5, 84, 0, 0, 226, 228, 3, 4, 2, 0, 227, 226, 1, 0, 0, 0, 227, 228, 1, 0, 0, 0, 228, 230, 1, 0, 0, 0, 229, 231, 5, 150, 0, 0, 230, 229, 1, 0, 0, 0, 230, 231, 1, 0, 0, 0, 231, 15, 1, 0, 0, 0, 232, 241, 5, 14, 0, 0, 233, 234, 5, 130, 0, 0, 234, 237, 3, 156, 78, 0, 235, 236, 5, 115, 0, 0, 236, 238, 3, 156, 78, 0, 237, 235, 1, 0, 0, 0, 237, 238, 1, 0, 0, 0, 238, 239, 1, 0, 0, 0, 239, 240, 5, 149, 0, 0, 240, 242, 1, 0, 0, 0, 241, 233, 1, 0, 0, 0, 241, 242, 1, 0, 0, 0, 242, 243, 1, 0, 0, 0, 243, 244, 3, 36, 18, 0, 244, 17, 1, 0, 0, 0, 245, 246, 5, 93, 0, 0, 246, 250, 3, 36, 18, 0, 247, 249, 3, 16, 8, 0, 248, 247, 1, 0, 0, 0, 249, 252, 1, 0, 0, 0, 250, 248, 1, 0, 0, 0, 250, 251, 1, 0, 0, 0, 251, 255, 1, 0, 0, 0, 252, 250, 1, 0, 0, 0, 253, 254, 5, 29, 0, 0, 254, 256, 3, 36, 18, 0, 255, 253, 1, 0, 0, 0, 255, 256, 1, 0, 0, 0, 256, 19, 1, 0, 0, 0, 257, 258, 5, 40, 0, 0, 258, 259, 5, 130, 0, 0, 259, 260, 3, 4, 2, 0, 260, 261, 5, 149, 0, 0, 261, 264, 3, 10, 5, 0, 262, 263, 5, 25, 0, 0, 263, 265, 3, 10, 5, 0, 264, 262, 1, 0, 0, 0, 264, 265, 1, 0, 0, 0, 265, 21, 1, 0, 0, 0, 266, 267, 5, 100, 0, 0, 267, 268, 5, 130, 0, 0, 268, 269, 3, 4, 2, 0, 269, 270, 5, 149, 0, 0, 270, 272, 3, 10, 5, 0, 271, 273, 5, 150, 0, 0, 272, 271, 1, 0, 0, 0, 272, 273, 1, 0, 0, 0, 273, 23, 1, 0, 0, 0, 274, 275, 5, 33, 0, 0, 275, 279, 5, 130, 0, 0, 276, 280, 3, 6, 3, 0, 277, 280, 3, 30, 15, 0, 278, 280, 3, 4, 2, 0, 279, 276, 1, 0, 0, 0, 279, 277, 1, 0, 0, 0, 279, 278, 1, 0, 0, 0, 279, 280, 1, 0, 0, 0, 280, 281, 1, 0, 0, 0, 281, 283, 5, 150, 0, 0, 282, 284, 3, 4, 2, 0, 283, 282, 1, 0, 0, 0, 283, 284, 1, 0, 0, 0, 284, 285, 1, 0, 0, 0, 285, 289, 5, 150, 0, 0, 286, 290, 3, 6, 3, 0, 287, 290, 3, 30, 15, 0, 288, 290, 3, 4, 2, 0, 289, 286, 1, 0, 0, 0, 289, 287, 1, 0, 0, 0, 289, 288, 1, 0, 0, 0, 289, 290, 1, 0, 0, 0, 290, 291, 1, 0, 0, 0, 291, 292, 5, 149, 0, 0, 292, 294, 3, 10, 5, 0, 293, 295, 5, 150, 0, 0, 294, 293, 1, 0, 0, 0, 294, 295, 1, 0, 0, 0, 295, 25, 1, 0, 0, 0, 296, 297, 5, 33, 0, 0, 297, 298, 5, 130, 0, 0, 298, 299, 5, 52, 0, 0, 299, 302, 3, 156, 78, 0, 300, 301, 5, 116, 0, 0, 301, 303, 3, 156, 78, 0, 302, 300, 1, 0, 0, 0, 302, 303, 1, 0, 0, 0, 303, 304, 1, 0, 0, 0, 304, 305, 5, 42, 0, 0, 305, 306, 3, 4, 2, 0, 306, 307, 5, 149, 0, 0, 307, 309, 3, 10, 5, 0, 308, 310, 5, 150, 0, 0, 309, 308, 1, 0, 0, 0, 309, 310, 1, 0, 0, 0, 310, 27, 1, 0, 0, 0, 311, 312, 5, 31, 0, 0, 312, 313, 3, 156, 78, 0, 313, 315, 5, 130, 0, 0, 314, 316, 3, 8, 4, 0, 315, 314, 1, 0, 0, 0, 315, 316, 1, 0, 0, 0, 316, 317, 1, 0, 0, 0, 317, 318, 5, 149, 0, 0, 318, 319, 3, 36, 18, 0, 319, 29, 1, 0, 0, 0, 320, 321, 3, 4, 2, 0, 321, 322, 5, 115, 0, 0, 322, 323, 5, 122, 0, 0, 323, 324, 3, 4, 2, 0, 324, 31, 1, 0, 0, 0, 325, 327, 3, 4, 2, 0, 326, 328, 5, 150, 0, 0, 327, 326, 1, 0, 0, 0, 327, 328, 1, 0, 0, 0, 328, 33, 1, 0, 0, 0, 329, 330, 5, 150, 0, 0, 330, 35, 1, 0, 0, 0, 331, 335, 5, 128, 0, 0, 332, 334, 3, 2, 1, 0, 333, 332, 1, 0, 0, 0, 334, 337, 1, 0, 0, 0, 335, 333, 1, 0, 0, 0, 335, 336, 1, 0, 0, 0, 336, 338, 1, 0, 0, 0, 337, 335, 1, 0, 0, 0, 338, 339, 5, 147, 0, 0, 339, 37, 1, 0, 0, 0, 340, 341, 3, 4, 2, 0, 341, 342, 5, 115, 0, 0, 342, 343, 3, 4, 2, 0, 343, 39, 1, 0, 0, 0, 344, 349, 3, 38, 19, 0, 345, 346, 5, 116, 0, 0, 346, 348, 3, 38, 19, 0, 347, 345, 1, 0, 0, 0, 348, 351, 1, 0, 0, 0, 349, 347, 1, 0, 0, 0, 349, 350, 1, 0, 0, 0, 350, 353, 1, 0, 0, 0, 351, 349, 1, 0, 0, 0, 352, 354, 5, 116, 0, 0, 353, 352, 1, 0, 0, 0, 353, 354, 1, 0, 0, 0, 354, 41, 1, 0, 0, 0, 355, 359, 3, 44, 22, 0, 356, 359, 3, 48, 24, 0, 357, 359, 3, 120, 60, 0, 358, 355, 1, 0, 0, 0, 358, 356, 1, 0, 0, 0, 358, 357, 1, 0, 0, 0, 359, 360, 1, 0, 0, 0, 360, 361, 5, 0, 0, 1, 361, 43, 1, 0, 0, 0, 362, 368, 3, 46, 23, 0, 363, 364, 5, 95, 0, 0, 364, 365, 5, 1, 0, 0, 365, 367, 3, 46, 23, 0, 366, 363, 1, 0, 0, 0, 367, 370, 1, 0, 0, 0, 368, 366, 1, 0, 0, 0, 368, 369, 1, 0, 0, 0, 369, 45, 1, 0, 0, 0, 370, 368, 1, 0, 0, 0, 371, 378, 3, 48, 24, 0, 372, 373, 5, 130, 0, 0, 373, 374, 3, 44, 22, 0, 374, 375, 5, 149, 0, 0, 375, 378, 1, 0, 0, 0, 376, 378, 3, 160, 80, 0, 377, 371, 1, 0, 0, 0, 377, 372, 1, 0, 0, 0, 377, 376, 1, 0, 0, 0, 378, 47, 1, 0, 0, 0, 379, 381, 3, 50, 25, 0, 380, 379, 1, 0, 0, 0, 380, 381, 1, 0, 0, 0, 381, 382, 1, 0, 0, 0, 382, 384, 5, 79, 0, 0, 383, 385, 5, 24, 0, 0, 384, 383, 1, 0, 0, 0, 384, 385, 1, 0, 0, 0, 385, 387, 1, 0, 0, 0, 386, 388, 3, 52, 26, 0, 387, 386, 1, 0, 0, 0, 387, 388, 1, 0, 0, 0, 388, 389, 1, 0, 0, 0, 389, 391, 3, 114, 57, 0, 390, 392, 3, 54, 27, 0, 391, 390, 1, 0, 0, 0, 391, 392, 1, 0, 0, 0, 392, 394, 1, 0, 0, 0, 393, 395, 3, 56, 28, 0, 394, 393, 1, 0, 0, 0, 394, 395, 1, 0, 0, 0, 395, 397, 1, 0, 0, 0, 396, 398, 3, 60, 30, 0, 397, 396, 1, 0, 0, 0, 397, 398, 1, 0, 0, 0, 398, 400, 1, 0, 0, 0, 399, 401, 3, 62, 31, 0, 400, 399, 1, 0, 0, 0, 400, 401, 1, 0, 0, 0, 401, 403, 1, 0, 0, 0, 402, 404, 3, 64, 32, 0, 403, 402, 1, 0, 0, 0, 403, 404, 1, 0, 0, 0, 404, 407, 1, 0, 0, 0, 405, 406, 5, 102, 0, 0, 406, 408, 7, 0, 0, 0, 407, 405, 1, 0, 0, 0, 407, 408, 1, 0, 0, 0, 408, 411, 1, 0, 0, 0, 409, 410, 5, 102, 0, 0, 410, 412, 5, 89, 0, 0, 411, 409, 1, 0, 0, 0, 411, 412, 1, 0, 0, 0, 412, 414, 1, 0, 0, 0, 413, 415, 3, 66, 33, 0, 414, 413, 1, 0, 0, 0, 414, 415, 1, 0, 0, 0, 415, 417, 1, 0, 0, 0, 416, 418, 3, 58, 29, 0, 417, 416, 1, 0, 0, 0, 417, 418, 1, 0, 0, 0, 418, 420, 1, 0, 0, 0, 419, 421, 3, 68, 34, 0, 420, 419, 1, 0, 0, 0, 420, 421, 1, 0, 0, 0, 421, 424, 1, 0, 0, 0, 422, 425, 3, 72, 36, 0, 423, 425, 3, 74, 37, 0, 424, 422, 1, 0, 0, 0, 424, 423, 1, 0, 0, 0, 424, 425, 1, 0, 0, 0, 425, 427, 1, 0, 0, 0, 426, 428, 3, 76, 38, 0, 427, 426, 1, 0, 0, 0, 427, 428, 1, 0, 0, 0, 428, 49, 1, 0, 0, 0, 429, 430, 5, 102, 0, 0, 430, 431, 3, 124, 62, 0, 431, 51, 1, 0, 0, 0, 432, 433, 5, 88, 0, 0, 433, 436, 5, 108, 0, 0, 434, 435, 5, 102, 0, 0, 435, 437, 5, 85, 0, 0, 436, 434, 1, 0, 0, 0, 436, 437, 1, 0, 0, 0, 437, 53, 1, 0, 0, 0, 438, 439, 5, 34, 0, 0, 439, 440, 3, 78, 39, 0, 440, 55, 1, 0, 0, 0, 441, 443, 7, 1, 0, 0, 442, 441, 1, 0, 0, 0, 442, 443, 1, 0, 0, 0, 443, 444, 1, 0, 0, 0, 444, 445, 5, 5, 0, 0, 445, 446, 5, 47, 0, 0, 446, 447, 3, 114, 57, 0, 447, 57, 1, 0, 0, 0, 448, 449, 5, 101, 0, 0, 449, 450, 3, 156, 78, 0, 450, 451, 5, 6, 0, 0, 451, 452, 5, 130, 0, 0, 452, 453, 3, 98, 49, 0, 453, 463, 5, 149, 0, 0, 454, 455, 5, 116, 0, 0, 455, 456, 3, 156, 78, 0, 456, 457, 5, 6, 0, 0, 457, 458, 5, 130, 0, 0, 458, 459, 3, 98, 49, 0, 459, 460, 5, 149, 0, 0, 460, 462, 1, 0, 0, 0, 461, 454, 1, 0, 0, 0, 462, 465, 1, 0, 0, 0, 463, 461, 1, 0, 0, 0, 463, 464, 1, 0, 0, 0, 464, 59, 1, 0, 0, 0, 465, 463, 1, 0, 0, 0, 466, 467, 5, 69, 0, 0, 467, 468, 3, 116, 58, 0, 468, 61, 1, 0, 0, 0, 469, 470, 5, 99, 0, 0, 470, 471, 3, 116, 58, 0, 471, 63, 1, 0, 0, 0, 472, 473, 5, 36, 0, 0, 473, 480, 5, 11, 0, 0, 474, 475, 7, 0, 0, 0, 475, 476, 5, 130, 0, 0, 476, 477, 3, 114, 57, 0, 477, 478, 5, 149, 0, 0, 478, 481, 1, 0, 0, 0, 479, 481, 3, 114, 57, 0, 480, 474, 1, 0, 0, 0, 480, 479, 1, 0, 0, 0, 481, 65, 1, 0, 0, 0, 482, 483, 5, 37, 0, 0, 483, 484, 3, 116, 58, 0, 484, 67, 1, 0, 0, 0, 485, 486, 5, 64, 0, 0, 486, 487, 5, 11, 0, 0, 487, 488, 3, 88, 44, 0, 488, 69, 1, 0, 0, 0, 489, 490, 5, 64, 0, 0, 490, 491, 5, 11, 0, 0, 491, 492, 3, 114, 57, 0, 492, 71, 1, 0, 0, 0, 493, 494, 5, 54, 0, 0, 494, 497, 3, 116, 58, 0, 495, 496, 5, 116, 0, 0, 496, 498, 3, 116, 58, 0, 497, 495, 1, 0, 0, 0, 497, 498, 1, 0, 0, 0, 498, 503, 1, 0, 0, 0, 499, 500, 5, 102, 0, 0, 500, 504, 5, 85, 0, 0, 501, 502, 5, 11, 0, 0, 502, 504, 3, 114, 57, 0, 503, 499, 1, 0, 0, 0, 503, 501, 1, 0, 0, 0, 503, 504, 1, 0, 0, 0, 504, 523, 1, 0, 0, 0, 505, 506, 5, 54, 0, 0, 506, 509, 3, 116, 58, 0, 507, 508, 5, 102, 0, 0, 508, 510, 5, 85, 0, 0, 509, 507, 1, 0, 0, 0, 509, 510, 1, 0, 0, 0, 510, 511, 1, 0, 0, 0, 511, 512, 5, 61, 0, 0, 512, 513, 3, 116, 58, 0, 513, 523, 1, 0, 0, 0, 514, 515, 5, 54, 0, 0, 515, 516, 3, 116, 58, 0, 516, 517, 5, 61, 0, 0, 517, 520, 3, 116, 58, 0, 518, 519, 5, 11, 0, 0, 519, 521, 3, 114, 57, 0, 520, 518, 1, 0, 0, 0, 520, 521, 1, 0, 0, 0, 521, 523, 1, 0, 0, 0, 522, 493, 1, 0, 0, 0, 522, 505, 1, 0, 0, 0, 522, 514, 1, 0, 0, 0, 523, 73, 1, 0, 0, 0, 524, 525, 5, 61, 0, 0, 525, 526, 3, 116, 58, 0, 526, 75, 1, 0, 0, 0, 527, 528, 5, 81, 0, 0, 528, 529, 3, 94, 47, 0, 529, 77, 1, 0, 0, 0, 530, 531, 6, 39, -1, 0, 531, 533, 3, 132, 66, 0, 532, 534, 5, 28, 0, 0, 533, 532, 1, 0, 0, 0, 533, 534, 1, 0, 0, 0, 534, 536, 1, 0, 0, 0, 535, 537, 3, 86, 43, 0, 536, 535, 1, 0, 0, 0, 536, 537, 1, 0, 0, 0, 537, 543, 1, 0, 0, 0, 538, 539, 5, 130, 0, 0, 539, 540, 3, 78, 39, 0, 540, 541, 5, 149, 0, 0, 541, 543, 1, 0, 0, 0, 542, 530, 1, 0, 0, 0, 542, 538, 1, 0, 0, 0, 543, 558, 1, 0, 0, 0, 544, 545, 10, 3, 0, 0, 545, 546, 3, 82, 41, 0, 546, 547, 3, 78, 39, 4, 547, 557, 1, 0, 0, 0, 548, 550, 10, 4, 0, 0, 549, 551, 3, 80, 40, 0, 550, 549, 1, 0, 0, 0, 550, 551, 1, 0, 0, 0, 551, 552, 1, 0, 0, 0, 552, 553, 5, 47, 0, 0, 553, 554, 3, 78, 39, 0, 554, 555, 3, 84, 42, 0, 555, 557, 1, 0, 0, 0, 556, 544, 1, 0, 0, 0, 556, 548, 1, 0, 0, 0, 557, 560, 1, 0, 0, 0, 558, 556, 1, 0, 0, 0, 558, 559, 1, 0, 0, 0, 559, 79, 1, 0, 0, 0, 560, 558, 1, 0, 0, 0, 561, 563, 7, 2, 0, 0, 562, 561, 1, 0, 0, 0, 562, 563, 1, 0, 0, 0, 563, 564, 1, 0, 0, 0, 564, 571, 5, 44, 0, 0, 565, 567, 5, 44, 0, 0, 566, 568, 7, 2, 0, 0, 567, 566, 1, 0, 0, 0, 567, 568, 1, 0, 0, 0, 568, 571, 1, 0, 0, 0, 569, 571, 7, 2, 0, 0, 570, 562, 1, 0, 0, 0, 570, 565, 1, 0, 0, 0, 570, 569, 1, 0, 0, 0, 571, 605, 1, 0, 0, 0, 572, 574, 7, 3, 0, 0, 573, 572, 1, 0, 0, 0, 573, 574, 1, 0, 0, 0, 574, 575, 1, 0, 0, 0, 575, 577, 7, 4, 0, 0, 576, 578, 5, 65, 0, 0, 577, 576, 1, 0, 0, 0, 577, 578, 1, 0, 0, 0, 578, 587, 1, 0, 0, 0, 579, 581, 7, 4, 0, 0, 580, 582, 5, 65, 0, 0, 581, 580, 1, 0, 0, 0, 581, 582, 1, 0, 0, 0, 582, 584, 1, 0, 0, 0, 583, 585, 7, 3, 0, 0, 584, 583, 1, 0, 0, 0, 584, 585, 1, 0, 0, 0, 585, 587, 1, 0, 0, 0, 586, 573, 1, 0, 0, 0, 586, 579, 1, 0, 0, 0, 587, 605, 1, 0, 0, 0, 588, 590, 7, 5, 0, 0, 589, 588, 1, 0, 0, 0, 589, 590, 1, 0, 0, 0, 590, 591, 1, 0, 0, 0, 591, 593, 5, 35, 0, 0, 592, 594, 5, 65, 0, 0, 593, 592, 1, 0, 0, 0, 593, 594, 1, 0, 0, 0, 594, 603, 1, 0, 0, 0, 595, 597, 5, 35, 0, 0, 596, 598, 5, 65, 0, 0, 597, 596, 1, 0, 0, 0, 597, 598, 1, 0, 0, 0, 598, 600, 1, 0, 0, 0, 599, 601, 7, 5, 0, 0, 600, 599, 1, 0, 0, 0, 600, 601, 1, 0, 0, 0, 601, 603, 1, 0, 0, 0, 602, 589, 1, 0, 0, 0, 602, 595, 1, 0, 0, 0, 603, 605, 1, 0, 0, 0, 604, 570, 1, 0, 0, 0, 604, 586, 1, 0, 0, 0, 604, 602, 1, 0, 0, 0, 605, 81, 1, 0, 0, 0, 606, 607, 5, 17, 0, 0, 607, 610, 5, 47, 0, 0, 608, 610, 5, 116, 0, 0, 609, 606, 1, 0, 0, 0, 609, 608, 1, 0, 0, 0, 610, 83, 1, 0, 0, 0, 611, 612, 5, 62, 0, 0, 612, 621, 3, 114, 57, 0, 613, 614, 5, 96, 0, 0, 614, 615, 5, 130, 0, 0, 615, 616, 3, 114, 57, 0, 616, 617, 5, 149, 0, 0, 617, 621, 1, 0, 0, 0, 618, 619, 5, 96, 0, 0, 619, 621, 3, 114, 57, 0, 620, 611, 1, 0, 0, 0, 620, 613, 1, 0, 0, 0, 620, 618, 1, 0, 0, 0, 621, 85, 1, 0, 0, 0, 622, 623, 5, 77, 0, 0, 623, 626, 3, 92, 46, 0, 624, 625, 5, 61, 0, 0, 625, 627, 3, 92, 46, 0, 626, 624, 1, 0, 0, 0, 626, 627, 1, 0, 0, 0, 627, 87, 1, 0, 0, 0, 628, 633, 3, 90, 45, 0, 629, 630, 5, 116, 0, 0, 630, 632, 3, 90, 45, 0, 631, 629, 1, 0, 0, 0, 632, 635, 1, 0, 0, 0, 633, 631, 1, 0, 0, 0, 633, 634, 1, 0, 0, 0, 634, 89, 1, 0, 0, 0, 635, 633, 1, 0, 0, 0, 636, 638, 3, 116, 58, 0, 637, 639, 7, 6, 0, 0, 638, 637, 1, 0, 0, 0, 638, 639, 1, 0, 0, 0, 639, 642, 1, 0, 0, 0, 640, 641, 5, 60, 0, 0, 641, 643, 7, 7, 0, 0, 642, 640, 1, 0, 0, 0, 642, 643, 1, 0, 0, 0, 643, 646, 1, 0, 0, 0, 644, 645, 5, 16, 0, 0, 645, 647, 5, 110, 0, 0, 646, 644, 1, 0, 0, 0, 646, 647, 1, 0, 0, 0, 647, 91, 1, 0, 0, 0, 648, 655, 3, 160, 80, 0, 649, 652, 3, 144, 72, 0, 650, 651, 5, 151, 0, 0, 651, 653, 3, 144, 72, 0, 652, 650, 1, 0, 0, 0, 652, 653, 1, 0, 0, 0, 653, 655, 1, 0, 0, 0, 654, 648, 1, 0, 0, 0, 654, 649, 1, 0, 0, 0, 655, 93, 1, 0, 0, 0, 656, 661, 3, 96, 48, 0, 657, 658, 5, 116, 0, 0, 658, 660, 3, 96, 48, 0, 659, 657, 1, 0, 0, 0, 660, 663, 1, 0, 0, 0, 661, 659, 1, 0, 0, 0, 661, 662, 1, 0, 0, 0, 662, 95, 1, 0, 0, 0, 663, 661, 1, 0, 0, 0, 664, 665, 3, 156, 78, 0, 665, 666, 5, 122, 0, 0, 666, 667, 3, 146, 73, 0, 667, 97, 1, 0, 0, 0, 668, 670, 3, 100, 50, 0, 669, 668, 1, 0, 0, 0, 669, 670, 1, 0, 0, 0, 670, 672, 1, 0, 0, 0, 671, 673, 3, 102, 51, 0, 672, 671, 1, 0, 0, 0, 672, 673, 1, 0, 0, 0, 673, 675, 1, 0, 0, 0, 674, 676, 3, 104, 52, 0, 675, 674, 1, 0, 0, 0, 675, 676, 1, 0, 0, 0, 676, 99, 1, 0, 0, 0, 677, 678, 5, 67, 0, 0, 678, 679, 5, 11, 0, 0, 679, 680, 3, 114, 57, 0, 680, 101, 1, 0, 0, 0, 681, 682, 5, 64, 0, 0, 682, 683, 5, 11, 0, 0, 683, 684, 3, 88, 44, 0, 684, 103, 1, 0, 0, 0, 685, 686, 7, 8, 0, 0, 686, 687, 3, 106, 53, 0, 687, 105, 1, 0, 0, 0, 688, 695, 3, 108, 54, 0, 689, 690, 5, 9, 0, 0, 690, 691, 3, 108, 54, 0, 691, 692, 5, 2, 0, 0, 692, 693, 3, 108, 54, 0, 693, 695, 1, 0, 0, 0, 694, 688, 1, 0, 0, 0, 694, 689, 1, 0, 0, 0, 695, 107, 1, 0, 0, 0, 696, 697, 5, 19, 0, 0, 697, 709, 5, 75, 0, 0, 698, 699, 5, 94, 0, 0, 699, 709, 5, 68, 0, 0, 700, 701, 5, 94, 0, 0, 701, 709, 5, 32, 0, 0, 702, 703, 3, 144, 72, 0, 703, 704, 5, 68, 0, 0, 704, 709, 1, 0, 0, 0, 705, 706, 3, 144, 72, 0, 706, 707, 5, 32, 0, 0, 707, 709, 1, 0, 0, 0, 708, 696, 1, 0, 0, 0, 708, 698, 1, 0, 0, 0, 708, 700, 1, 0, 0, 0, 708, 702, 1, 0, 0, 0, 708, 705, 1, 0, 0, 0, 709, 109, 1, 0, 0, 0, 710, 711, 3, 116, 58, 0, 711, 712, 5, 0, 0, 1, 712, 111, 1, 0, 0, 0, 713, 770, 3, 156, 78, 0, 714, 715, 3, 156, 78, 0, 715, 716, 5, 130, 0, 0, 716, 717, 3, 156, 78, 0, 717, 724, 3, 112, 56, 0, 718, 719, 5, 116, 0, 0, 719, 720, 3, 156, 78, 0, 720, 721, 3, 112, 56, 0, 721, 723, 1, 0, 0, 0, 722, 718, 1, 0, 0, 0, 723, 726, 1, 0, 0, 0, 724, 722, 1, 0, 0, 0, 724, 725, 1, 0, 0, 0, 725, 728, 1, 0, 0, 0, 726, 724, 1, 0, 0, 0, 727, 729, 5, 116, 0, 0, 728, 727, 1, 0, 0, 0, 728, 729, 1, 0, 0, 0, 729, 730, 1, 0, 0, 0, 730, 731, 5, 149, 0, 0, 731, 770, 1, 0, 0, 0, 732, 733, 3, 156, 78, 0, 733, 734, 5, 130, 0, 0, 734, 739, 3, 158, 79, 0, 735, 736, 5, 116, 0, 0, 736, 738, 3, 158, 79, 0, 737, 735, 1, 0, 0, 0, 738, 741, 1, 0, 0, 0, 739, 737, 1, 0, 0, 0, 739, 740, 1, 0, 0, 0, 740, 743, 1, 0, 0, 0, 741, 739, 1, 0, 0, 0, 742, 744, 5, 116, 0, 0, 743, 742, 1, 0, 0, 0, 743, 744, 1, 0, 0, 0, 744, 745, 1, 0, 0, 0, 745, 746, 5, 149, 0, 0, 746, 770, 1, 0, 0, 0, 747, 748, 3, 156, 78, 0, 748, 749, 5, 130, 0, 0, 749, 754, 3, 112, 56, 0, 750, 751, 5, 116, 0, 0, 751, 753, 3, 112, 56, 0, 752, 750, 1, 0, 0, 0, 753, 756, 1, 0, 0, 0, 754, 752, 1, 0, 0, 0, 754, 755, 1, 0, 0, 0, 755, 758, 1, 0, 0, 0, 756, 754, 1, 0, 0, 0, 757, 759, 5, 116, 0, 0, 758, 757, 1, 0, 0, 0, 758, 759, 1, 0, 0, 0, 759, 760, 1, 0, 0, 0, 760, 761, 5, 149, 0, 0, 761, 770, 1, 0, 0, 0, 762, 763, 3, 156, 78, 0, 763, 765, 5, 130, 0, 0, 764, 766, 3, 114, 57, 0, 765, 764, 1, 0, 0, 0, 765, 766, 1, 0, 0, 0, 766, 767, 1, 0, 0, 0, 767, 768, 5, 149, 0, 0, 768, 770, 1, 0, 0, 0, 769, 713, 1, 0, 0, 0, 769, 714, 1, 0, 0, 0, 769, 732, 1, 0, 0, 0, 769, 747, 1, 0, 0, 0, 769, 762, 1, 0, 0, 0, 770, 113, 1, 0, 0, 0, 771, 776, 3, 116, 58, 0, 772, 773, 5, 116, 0, 0, 773, 775, 3, 116, 58, 0, 774, 772, 1, 0, 0, 0, 775, 778, 1, 0, 0, 0, 776, 774, 1, 0, 0, 0, 776, 777, 1, 0, 0, 0, 777, 780, 1, 0, 0, 0, 778, 776, 1, 0, 0, 0, 779, 781, 5, 116, 0, 0, 780, 779, 1, 0, 0, 0, 780, 781, 1, 0, 0, 0, 781, 115, 1, 0, 0, 0, 782, 783, 6, 58, -1, 0, 783, 785, 5, 12, 0, 0, 784, 786, 3, 116, 58, 0, 785, 784, 1, 0, 0, 0, 785, 786, 1, 0, 0, 0, 786, 792, 1, 0, 0, 0, 787, 788, 5, 98, 0, 0, 788, 789, 3, 116, 58, 0, 789, 790, 5, 83, 0, 0, 790, 791, 3, 116, 58, 0, 791, 793, 1, 0, 0, 0, 792, 787, 1, 0, 0, 0, 793, 794, 1, 0, 0, 0, 794, 792, 1, 0, 0, 0, 794, 795, 1, 0, 0, 0, 795, 798, 1, 0, 0, 0, 796, 797, 5, 25, 0, 0, 797, 799, 3, 116, 58, 0, 798, 796, 1, 0, 0, 0, 798, 799, 1, 0, 0, 0, 799, 800, 1, 0, 0, 0, 800, 801, 5, 26, 0, 0, 801, 933, 1, 0, 0, 0, 802, 803, 5, 13, 0, 0, 803, 804, 5, 130, 0, 0, 804, 805, 3, 116, 58, 0, 805, 806, 5, 6, 0, 0, 806, 807, 3, 112, 56, 0, 807, 808, 5, 149, 0, 0, 808, 933, 1, 0, 0, 0, 809, 810, 5, 20, 0, 0, 810, 933, 5, 110, 0, 0, 811, 812, 5, 45, 0, 0, 812, 813, 3, 116, 58, 0, 813, 814, 3, 148, 74, 0, 814, 933, 1, 0, 0, 0, 815, 816, 5, 82, 0, 0, 816, 817, 5, 130, 0, 0, 817, 818, 3, 116, 58, 0, 818, 819, 5, 34, 0, 0, 819, 822, 3, 116, 58, 0, 820, 821, 5, 33, 0, 0, 821, 823, 3, 116, 58, 0, 822, 820, 1, 0, 0, 0, 822, 823, 1, 0, 0, 0, 823, 824, 1, 0, 0, 0, 824, 825, 5, 149, 0, 0, 825, 933, 1, 0, 0, 0, 826, 827, 5, 86, 0, 0, 827, 933, 5, 110, 0, 0, 828, 829, 5, 91, 0, 0, 829, 830, 5, 130, 0, 0, 830, 831, 7, 9, 0, 0, 831, 832, 3, 162, 81, 0, 832, 833, 5, 34, 0, 0, 833, 834, 3, 116, 58, 0, 834, 835, 5, 149, 0, 0, 835, 933, 1, 0, 0, 0, 836, 837, 3, 156, 78, 0, 837, 839, 5, 130, 0, 0, 838, 840, 3, 114, 57, 0, 839, 838, 1, 0, 0, 0, 839, 840, 1, 0, 0, 0, 840, 841, 1, 0, 0, 0, 841, 842, 5, 149, 0, 0, 842, 851, 1, 0, 0, 0, 843, 845, 5, 130, 0, 0, 844, 846, 5, 24, 0, 0, 845, 844, 1, 0, 0, 0, 845, 846, 1, 0, 0, 0, 846, 848, 1, 0, 0, 0, 847, 849, 3, 114, 57, 0, 848, 847, 1, 0, 0, 0, 848, 849, 1, 0, 0, 0, 849, 850, 1, 0, 0, 0, 850, 852, 5, 149, 0, 0, 851, 843, 1, 0, 0, 0, 851, 852, 1, 0, 0, 0, 852, 853, 1, 0, 0, 0, 853, 854, 5, 66, 0, 0, 854, 855, 5, 130, 0, 0, 855, 856, 3, 98, 49, 0, 856, 857, 5, 149, 0, 0, 857, 933, 1, 0, 0, 0, 858, 859, 3, 156, 78, 0, 859, 861, 5, 130, 0, 0, 860, 862, 3, 114, 57, 0, 861, 860, 1, 0, 0, 0, 861, 862, 1, 0, 0, 0, 862, 863, 1, 0, 0, 0, 863, 864, 5, 149, 0, 0, 864, 873, 1, 0, 0, 0, 865, 867, 5, 130, 0, 0, 866, 868, 5, 24, 0, 0, 867, 866, 1, 0, 0, 0, 867, 868, 1, 0, 0, 0, 868, 870, 1, 0, 0, 0, 869, 871, 3, 114, 57, 0, 870, 869, 1, 0, 0, 0, 870, 871, 1, 0, 0, 0, 871, 872, 1, 0, 0, 0, 872, 874, 5, 149, 0, 0, 873, 865, 1, 0, 0, 0, 873, 874, 1, 0, 0, 0, 874, 875, 1, 0, 0, 0, 875, 876, 5, 66, 0, 0, 876, 877, 3, 156, 78, 0, 877, 933, 1, 0, 0, 0, 878, 884, 3, 156, 78, 0, 879, 881, 5, 130, 0, 0, 880, 882, 3, 114, 57, 0, 881, 880, 1, 0, 0, 0, 881, 882, 1, 0, 0, 0, 882, 883, 1, 0, 0, 0, 883, 885, 5, 149, 0, 0, 884, 879, 1, 0, 0, 0, 884, 885, 1, 0, 0, 0, 885, 886, 1, 0, 0, 0, 886, 888, 5, 130, 0, 0, 887, 889, 5, 24, 0, 0, 888, 887, 1, 0, 0, 0, 888, 889, 1, 0, 0, 0, 889, 891, 1, 0, 0, 0, 890, 892, 3, 114, 57, 0, 891, 890, 1, 0, 0, 0, 891, 892, 1, 0, 0, 0, 892, 893, 1, 0, 0, 0, 893, 894, 5, 149, 0, 0, 894, 933, 1, 0, 0, 0, 895, 933, 3, 120, 60, 0, 896, 933, 3, 164, 82, 0, 897, 933, 3, 146, 73, 0, 898, 899, 5, 118, 0, 0, 899, 933, 3, 116, 58, 20, 900, 901, 5, 58, 0, 0, 901, 933, 3, 116, 58, 14, 902, 903, 3, 136, 68, 0, 903, 904, 5, 120, 0, 0, 904, 906, 1, 0, 0, 0, 905, 902, 1, 0, 0, 0, 905, 906, 1, 0, 0, 0, 906, 907, 1, 0, 0, 0, 907, 933, 5, 112, 0, 0, 908, 909, 5, 130, 0, 0, 909, 910, 3, 44, 22, 0, 910, 911, 5, 149, 0, 0, 911, 933, 1, 0, 0, 0, 912, 913, 5, 130, 0, 0, 913, 914, 3, 116, 58, 0, 914, 915, 5, 149, 0, 0, 915, 933, 1, 0, 0, 0, 916, 917, 5, 130, 0, 0, 917, 918, 3, 114, 57, 0, 918, 919, 5, 149, 0, 0, 919, 933, 1, 0, 0, 0, 920, 922, 5, 129, 0, 0, 921, 923, 3, 114, 57, 0, 922, 921, 1, 0, 0, 0, 922, 923, 1, 0, 0, 0, 923, 924, 1, 0, 0, 0, 924, 933, 5, 148, 0, 0, 925, 927, 5, 128, 0, 0, 926, 928, 3, 40, 20, 0, 927, 926, 1, 0, 0, 0, 927, 928, 1, 0, 0, 0, 928, 929, 1, 0, 0, 0, 929, 933, 5, 147, 0, 0, 930, 933, 3, 118, 59, 0, 931, 933, 3, 128, 64, 0, 932, 782, 1, 0, 0, 0, 932, 802, 1, 0, 0, 0, 932, 809, 1, 0, 0, 0, 932, 811, 1, 0, 0, 0, 932, 815, 1, 0, 0, 0, 932, 826, 1, 0, 0, 0, 932, 828, 1, 0, 0, 0, 932, 836, 1, 0, 0, 0, 932, 858, 1, 0, 0, 0, 932, 878, 1, 0, 0, 0, 932, 895, 1, 0, 0, 0, 932, 896, 1, 0, 0, 0, 932, 897, 1, 0, 0, 0, 932, 898, 1, 0, 0, 0, 932, 900, 1, 0, 0, 0, 932, 905, 1, 0, 0, 0, 932, 908, 1, 0, 0, 0, 932, 912, 1, 0, 0, 0, 932, 916, 1, 0, 0, 0, 932, 920, 1, 0, 0, 0, 932, 925, 1, 0, 0, 0, 932, 930, 1, 0, 0, 0, 932, 931, 1, 0, 0, 0, 933, 1044, 1, 0, 0, 0, 934, 938, 10, 19, 0, 0, 935, 939, 5, 112, 0, 0, 936, 939, 5, 151, 0, 0, 937, 939, 5, 138, 0, 0, 938, 935, 1, 0, 0, 0, 938, 936, 1, 0, 0, 0, 938, 937, 1, 0, 0, 0, 939, 940, 1, 0, 0, 0, 940, 1043, 3, 116, 58, 20, 941, 945, 10, 18, 0, 0, 942, 946, 5, 139, 0, 0, 943, 946, 5, 118, 0, 0, 944, 946, 5, 117, 0, 0, 945, 942, 1, 0, 0, 0, 945, 943, 1, 0, 0, 0, 945, 944, 1, 0, 0, 0, 946, 947, 1, 0, 0, 0, 947, 1043, 3, 116, 58, 19, 948, 973, 10, 17, 0, 0, 949, 974, 5, 121, 0, 0, 950, 974, 5, 122, 0, 0, 951, 974, 5, 133, 0, 0, 952, 974, 5, 131, 0, 0, 953, 974, 5, 132, 0, 0, 954, 974, 5, 123, 0, 0, 955, 974, 5, 124, 0, 0, 956, 958, 5, 58, 0, 0, 957, 956, 1, 0, 0, 0, 957, 958, 1, 0, 0, 0, 958, 959, 1, 0, 0, 0, 959, 961, 5, 42, 0, 0, 960, 962, 5, 15, 0, 0, 961, 960, 1, 0, 0, 0, 961, 962, 1, 0, 0, 0, 962, 974, 1, 0, 0, 0, 963, 965, 5, 58, 0, 0, 964, 963, 1, 0, 0, 0, 964, 965, 1, 0, 0, 0, 965, 966, 1, 0, 0, 0, 966, 974, 7, 10, 0, 0, 967, 974, 5, 145, 0, 0, 968, 974, 5, 146, 0, 0, 969, 974, 5, 135, 0, 0, 970, 974, 5, 126, 0, 0, 971, 974, 5, 127, 0, 0, 972, 974, 5, 134, 0, 0, 973, 949, 1, 0, 0, 0, 973, 950, 1, 0, 0, 0, 973, 951, 1, 0, 0, 0, 973, 952, 1, 0, 0, 0, 973, 953, 1, 0, 0, 0, 973, 954, 1, 0, 0, 0, 973, 955, 1, 0, 0, 0, 973, 957, 1, 0, 0, 0, 973, 964, 1, 0, 0, 0, 973, 967, 1, 0, 0, 0, 973, 968, 1, 0, 0, 0, 973, 969, 1, 0, 0, 0, 973, 970, 1, 0, 0, 0, 973, 971, 1, 0, 0, 0, 973, 972, 1, 0, 0, 0, 974, 975, 1, 0, 0, 0, 975, 1043, 3, 116, 58, 18, 976, 977, 10, 15, 0, 0, 977, 978, 5, 137, 0, 0, 978, 1043, 3, 116, 58, 16, 979, 980, 10, 13, 0, 0, 980, 981, 5, 2, 0, 0, 981, 1043, 3, 116, 58, 14, 982, 983, 10, 12, 0, 0, 983, 984, 5, 63, 0, 0, 984, 1043, 3, 116, 58, 13, 985, 987, 10, 11, 0, 0, 986, 988, 5, 58, 0, 0, 987, 986, 1, 0, 0, 0, 987, 988, 1, 0, 0, 0, 988, 989, 1, 0, 0, 0, 989, 990, 5, 9, 0, 0, 990, 991, 3, 116, 58, 0, 991, 992, 5, 2, 0, 0, 992, 993, 3, 116, 58, 12, 993, 1043, 1, 0, 0, 0, 994, 995, 10, 10, 0, 0, 995, 996, 5, 140, 0, 0, 996, 997, 3, 116, 58, 0, 997, 998, 5, 115, 0, 0, 998, 999, 3, 116, 58, 10, 999, 1043, 1, 0, 0, 0, 1000, 1001, 10, 30, 0, 0, 1001, 1003, 5, 130, 0, 0, 1002, 1004, 3, 114, 57, 0, 1003, 1002, 1, 0, 0, 0, 1003, 1004, 1, 0, 0, 0, 1004, 1005, 1, 0, 0, 0, 1005, 1043, 5, 149, 0, 0, 1006, 1007, 10, 26, 0, 0, 1007, 1008, 5, 129, 0, 0, 1008, 1009, 3, 116, 58, 0, 1009, 1010, 5, 148, 0, 0, 1010, 1043, 1, 0, 0, 0, 1011, 1012, 10, 25, 0, 0, 1012, 1013, 5, 120, 0, 0, 1013, 1043, 5, 108, 0, 0, 1014, 1015, 10, 24, 0, 0, 1015, 1016, 5, 120, 0, 0, 1016, 1043, 3, 156, 78, 0, 1017, 1018, 10, 23, 0, 0, 1018, 1019, 5, 136, 0, 0, 1019, 1020, 5, 129, 0, 0, 1020, 1021, 3, 116, 58, 0, 1021, 1022, 5, 148, 0, 0, 1022, 1043, 1, 0, 0, 0, 1023, 1024, 10, 22, 0, 0, 1024, 1025, 5, 136, 0, 0, 1025, 1043, 5, 108, 0, 0, 1026, 1027, 10, 21, 0, 0, 1027, 1028, 5, 136, 0, 0, 1028, 1043, 3, 156, 78, 0, 1029, 1030, 10, 16, 0, 0, 1030, 1032, 5, 46, 0, 0, 1031, 1033, 5, 58, 0, 0, 1032, 1031, 1, 0, 0, 0, 1032, 1033, 1, 0, 0, 0, 1033, 1034, 1, 0, 0, 0, 1034, 1043, 5, 59, 0, 0, 1035, 1040, 10, 9, 0, 0, 1036, 1037, 5, 6, 0, 0, 1037, 1041, 3, 156, 78, 0, 1038, 1039, 5, 6, 0, 0, 1039, 1041, 5, 110, 0, 0, 1040, 1036, 1, 0, 0, 0, 1040, 1038, 1, 0, 0, 0, 1041, 1043, 1, 0, 0, 0, 1042, 934, 1, 0, 0, 0, 1042, 941, 1, 0, 0, 0, 1042, 948, 1, 0, 0, 0, 1042, 976, 1, 0, 0, 0, 1042, 979, 1, 0, 0, 0, 1042, 982, 1, 0, 0, 0, 1042, 985, 1, 0, 0, 0, 1042, 994, 1, 0, 0, 0, 1042, 1000, 1, 0, 0, 0, 1042, 1006, 1, 0, 0, 0, 1042, 1011, 1, 0, 0, 0, 1042, 1014, 1, 0, 0, 0, 1042, 1017, 1, 0, 0, 0, 1042, 1023, 1, 0, 0, 0, 1042, 1026, 1, 0, 0, 0, 1042, 1029, 1, 0, 0, 0, 1042, 1035, 1, 0, 0, 0, 1043, 1046, 1, 0, 0, 0, 1044, 1042, 1, 0, 0, 0, 1044, 1045, 1, 0, 0, 0, 1045, 117, 1, 0, 0, 0, 1046, 1044, 1, 0, 0, 0, 1047, 1048, 5, 130, 0, 0, 1048, 1053, 3, 156, 78, 0, 1049, 1050, 5, 116, 0, 0, 1050, 1052, 3, 156, 78, 0, 1051, 1049, 1, 0, 0, 0, 1052, 1055, 1, 0, 0, 0, 1053, 1051, 1, 0, 0, 0, 1053, 1054, 1, 0, 0, 0, 1054, 1057, 1, 0, 0, 0, 1055, 1053, 1, 0, 0, 0, 1056, 1058, 5, 116, 0, 0, 1057, 1056, 1, 0, 0, 0, 1057, 1058, 1, 0, 0, 0, 1058, 1059, 1, 0, 0, 0, 1059, 1060, 5, 149, 0, 0, 1060, 1075, 1, 0, 0, 0, 1061, 1066, 3, 156, 78, 0, 1062, 1063, 5, 116, 0, 0, 1063, 1065, 3, 156, 78, 0, 1064, 1062, 1, 0, 0, 0, 1065, 1068, 1, 0, 0, 0, 1066, 1064, 1, 0, 0, 0, 1066, 1067, 1, 0, 0, 0, 1067, 1070, 1, 0, 0, 0, 1068, 1066, 1, 0, 0, 0, 1069, 1071, 5, 116, 0, 0, 1070, 1069, 1, 0, 0, 0, 1070, 1071, 1, 0, 0, 0, 1071, 1075, 1, 0, 0, 0, 1072, 1073, 5, 130, 0, 0, 1073, 1075, 5, 149, 0, 0, 1074, 1047, 1, 0, 0, 0, 1074, 1061, 1, 0, 0, 0, 1074, 1072, 1, 0, 0, 0, 1075, 1076, 1, 0, 0, 0, 1076, 1079, 5, 111, 0, 0, 1077, 1080, 3, 116, 58, 0, 1078, 1080, 3, 36, 18, 0, 1079, 1077, 1, 0, 0, 0, 1079, 1078, 1, 0, 0, 0, 1080, 119, 1, 0, 0, 0, 1081, 1082, 5, 132, 0, 0, 1082, 1086, 3, 156, 78, 0, 1083, 1085, 3, 122, 61, 0, 1084, 1083, 1, 0, 0, 0, 1085, 1088, 1, 0, 0, 0, 1086, 1084, 1, 0, 0, 0, 1086, 1087, 1, 0, 0, 0, 1087, 1089, 1, 0, 0, 0, 1088, 1086, 1, 0, 0, 0, 1089, 1090, 5, 151, 0, 0, 1090, 1091, 5, 124, 0, 0, 1091, 1114, 1, 0, 0, 0, 1092, 1093, 5, 132, 0, 0, 1093, 1097, 3, 156, 78, 0, 1094, 1096, 3, 122, 61, 0, 1095, 1094, 1, 0, 0, 0, 1096, 1099, 1, 0, 0, 0, 1097, 1095, 1, 0, 0, 0, 1097, 1098, 1, 0, 0, 0, 1098, 1100, 1, 0, 0, 0, 1099, 1097, 1, 0, 0, 0, 1100, 1106, 5, 124, 0, 0, 1101, 1107, 3, 120, 60, 0, 1102, 1103, 5, 128, 0, 0, 1103, 1104, 3, 116, 58, 0, 1104, 1105, 5, 147, 0, 0, 1105, 1107, 1, 0, 0, 0, 1106, 1101, 1, 0, 0, 0, 1106, 1102, 1, 0, 0, 0, 1106, 1107, 1, 0, 0, 0, 1107, 1108, 1, 0, 0, 0, 1108, 1109, 5, 132, 0, 0, 1109, 1110, 5, 151, 0, 0, 1110, 1111, 3, 156, 78, 0, 1111, 1112, 5, 124, 0, 0, 1112, 1114, 1, 0, 0, 0, 1113, 1081, 1, 0, 0, 0, 1113, 1092, 1, 0, 0, 0, 1114, 121, 1, 0, 0, 0, 1115, 1116, 3, 156, 78, 0, 1116, 1117, 5, 122, 0, 0, 1117, 1118, 3, 162, 81, 0, 1118, 1127, 1, 0, 0, 0, 1119, 1120, 3, 156, 78, 0, 1120, 1121, 5, 122, 0, 0, 1121, 1122, 5, 128, 0, 0, 1122, 1123, 3, 116, 58, 0, 1123, 1124, 5, 147, 0, 0, 1124, 1127, 1, 0, 0, 0, 1125, 1127, 3, 156, 78, 0, 1126, 1115, 1, 0, 0, 0, 1126, 1119, 1, 0, 0, 0, 1126, 1125, 1, 0, 0, 0, 1127, 123, 1, 0, 0, 0, 1128, 1133, 3, 126, 63, 0, 1129, 1130, 5, 116, 0, 0, 1130, 1132, 3, 126, 63, 0, 1131, 1129, 1, 0, 0, 0, 1132, 1135, 1, 0, 0, 0, 1133, 1131, 1, 0, 0, 0, 1133, 1134, 1, 0, 0, 0, 1134, 1137, 1, 0, 0, 0, 1135, 1133, 1, 0, 0, 0, 1136, 1138, 5, 116, 0, 0, 1137, 1136, 1, 0, 0, 0, 1137, 1138, 1, 0, 0, 0, 1138, 125, 1, 0, 0, 0, 1139, 1140, 3, 156, 78, 0, 1140, 1141, 5, 6, 0, 0, 1141, 1142, 5, 130, 0, 0, 1142, 1143, 3, 44, 22, 0, 1143, 1144, 5, 149, 0, 0, 1144, 1150, 1, 0, 0, 0, 1145, 1146, 3, 116, 58, 0, 1146, 1147, 5, 6, 0, 0, 1147, 1148, 3, 156, 78, 0, 1148, 1150, 1, 0, 0, 0, 1149, 1139, 1, 0, 0, 0, 1149, 1145, 1, 0, 0, 0, 1150, 127, 1, 0, 0, 0, 1151, 1159, 3, 160, 80, 0, 1152, 1153, 3, 136, 68, 0, 1153, 1154, 5, 120, 0, 0, 1154, 1156, 1, 0, 0, 0, 1155, 1152, 1, 0, 0, 0, 1155, 1156, 1, 0, 0, 0, 1156, 1157, 1, 0, 0, 0, 1157, 1159, 3, 130, 65, 0, 1158, 1151, 1, 0, 0, 0, 1158, 1155, 1, 0, 0, 0, 1159, 129, 1, 0, 0, 0, 1160, 1165, 3, 156, 78, 0, 1161, 1162, 5, 120, 0, 0, 1162, 1164, 3, 156, 78, 0, 1163, 1161, 1, 0, 0, 0, 1164, 1167, 1, 0, 0, 0, 1165, 1163, 1, 0, 0, 0, 1165, 1166, 1, 0, 0, 0, 1166, 131, 1, 0, 0, 0, 1167, 1165, 1, 0, 0, 0, 1168, 1169, 6, 66, -1, 0, 1169, 1178, 3, 136, 68, 0, 1170, 1178, 3, 134, 67, 0, 1171, 1172, 5, 130, 0, 0, 1172, 1173, 3, 44, 22, 0, 1173, 1174, 5, 149, 0, 0, 1174, 1178, 1, 0, 0, 0, 1175, 1178, 3, 120, 60, 0, 1176, 1178, 3, 160, 80, 0, 1177, 1168, 1, 0, 0, 0, 1177, 1170, 1, 0, 0, 0, 1177, 1171, 1, 0, 0, 0, 1177, 1175, 1, 0, 0, 0, 1177, 1176, 1, 0, 0, 0, 1178, 1187, 1, 0, 0, 0, 1179, 1183, 10, 3, 0, 0, 1180, 1184, 3, 154, 77, 0, 1181, 1182, 5, 6, 0, 0, 1182, 1184, 3, 156, 78, 0, 1183, 1180, 1, 0, 0, 0, 1183, 1181, 1, 0, 0, 0, 1184, 1186, 1, 0, 0, 0, 1185, 1179, 1, 0, 0, 0, 1186, 1189, 1, 0, 0, 0, 1187, 1185, 1, 0, 0, 0, 1187, 1188, 1, 0, 0, 0, 1188, 133, 1, 0, 0, 0, 1189, 1187, 1, 0, 0, 0, 1190, 1191, 3, 156, 78, 0, 1191, 1193, 5, 130, 0, 0, 1192, 1194, 3, 138, 69, 0, 1193, 1192, 1, 0, 0, 0, 1193, 1194, 1, 0, 0, 0, 1194, 1195, 1, 0, 0, 0, 1195, 1196, 5, 149, 0, 0, 1196, 135, 1, 0, 0, 0, 1197, 1198, 3, 140, 70, 0, 1198, 1199, 5, 120, 0, 0, 1199, 1201, 1, 0, 0, 0, 1200, 1197, 1, 0, 0, 0, 1200, 1201, 1, 0, 0, 0, 1201, 1202, 1, 0, 0, 0, 1202, 1203, 3, 156, 78, 0, 1203, 137, 1, 0, 0, 0, 1204, 1209, 3, 116, 58, 0, 1205, 1206, 5, 116, 0, 0, 1206, 1208, 3, 116, 58, 0, 1207, 1205, 1, 0, 0, 0, 1208, 1211, 1, 0, 0, 0, 1209, 1207, 1, 0, 0, 0, 1209, 1210, 1, 0, 0, 0, 1210, 1213, 1, 0, 0, 0, 1211, 1209, 1, 0, 0, 0, 1212, 1214, 5, 116, 0, 0, 1213, 1212, 1, 0, 0, 0, 1213, 1214, 1, 0, 0, 0, 1214, 139, 1, 0, 0, 0, 1215, 1216, 3, 156, 78, 0, 1216, 141, 1, 0, 0, 0, 1217, 1226, 5, 106, 0, 0, 1218, 1219, 5, 120, 0, 0, 1219, 1226, 7, 11, 0, 0, 1220, 1221, 5, 108, 0, 0, 1221, 1223, 5, 120, 0, 0, 1222, 1224, 7, 11, 0, 0, 1223, 1222, 1, 0, 0, 0, 1223, 1224, 1, 0, 0, 0, 1224, 1226, 1, 0, 0, 0, 1225, 1217, 1, 0, 0, 0, 1225, 1218, 1, 0, 0, 0, 1225, 1220, 1, 0, 0, 0, 1226, 143, 1, 0, 0, 0, 1227, 1229, 7, 12, 0, 0, 1228, 1227, 1, 0, 0, 0, 1228, 1229, 1, 0, 0, 0, 1229, 1236, 1, 0, 0, 0, 1230, 1237, 3, 142, 71, 0, 1231, 1237, 5, 107, 0, 0, 1232, 1237, 5, 108, 0, 0, 1233, 1237, 5, 109, 0, 0, 1234, 1237, 5, 43, 0, 0, 1235, 1237, 5, 57, 0, 0, 1236, 1230, 1, 0, 0, 0, 1236, 1231, 1, 0, 0, 0, 1236, 1232, 1, 0, 0, 0, 1236, 1233, 1, 0, 0, 0, 1236, 1234, 1, 0, 0, 0, 1236, 1235, 1, 0, 0, 0, 1237, 145, 1, 0, 0, 0, 1238, 1242, 3, 144, 72, 0, 1239, 1242, 5, 110, 0, 0, 1240, 1242, 5, 59, 0, 0, 1241, 1238, 1, 0, 0, 0, 1241, 1239, 1, 0, 0, 0, 1241, 1240, 1, 0, 0, 0, 1242, 147, 1, 0, 0, 0, 1243, 1244, 7, 13, 0, 0, 1244, 149, 1, 0, 0, 0, 1245, 1246, 7, 14, 0, 0, 1246, 151, 1, 0, 0, 0, 1247, 1248, 7, 15, 0, 0, 1248, 153, 1, 0, 0, 0, 1249, 1252, 5, 105, 0, 0, 1250, 1252, 3, 152, 76, 0, 1251, 1249, 1, 0, 0, 0, 1251, 1250, 1, 0, 0, 0, 1252, 155, 1, 0, 0, 0, 1253, 1257, 5, 105, 0, 0, 1254, 1257, 3, 148, 74, 0, 1255, 1257, 3, 150, 75, 0, 1256, 1253, 1, 0, 0, 0, 1256, 1254, 1, 0, 0, 0, 1256, 1255, 1, 0, 0, 0, 1257, 157, 1, 0, 0, 0, 1258, 1259, 3, 162, 81, 0, 1259, 1260, 5, 122, 0, 0, 1260, 1261, 3, 144, 72, 0, 1261, 159, 1, 0, 0, 0, 1262, 1263, 5, 128, 0, 0, 1263, 1264, 3, 130, 65, 0, 1264, 1265, 5, 147, 0, 0, 1265, 161, 1, 0, 0, 0, 1266, 1269, 5, 110, 0, 0, 1267, 1269, 3, 164, 82, 0, 1268, 1266, 1, 0, 0, 0, 1268, 1267, 1, 0, 0, 0, 1269, 163, 1, 0, 0, 0, 1270, 1274, 5, 142, 0, 0, 1271, 1273, 3, 166, 83, 0, 1272, 1271, 1, 0, 0, 0, 1273, 1276, 1, 0, 0, 0, 1274, 1272, 1, 0, 0, 0, 1274, 1275, 1, 0, 0, 0, 1275, 1277, 1, 0, 0, 0, 1276, 1274, 1, 0, 0, 0, 1277, 1278, 5, 144, 0, 0, 1278, 165, 1, 0, 0, 0, 1279, 1280, 5, 157, 0, 0, 1280, 1281, 3, 116, 58, 0, 1281, 1282, 5, 147, 0, 0, 1282, 1285, 1, 0, 0, 0, 1283, 1285, 5, 156, 0, 0, 1284, 1279, 1, 0, 0, 0, 1284, 1283, 1, 0, 0, 0, 1285, 167, 1, 0, 0, 0, 1286, 1290, 5, 143, 0, 0, 1287, 1289, 3, 170, 85, 0, 1288, 1287, 1, 0, 0, 0, 1289, 1292, 1, 0, 0, 0, 1290, 1288, 1, 0, 0, 0, 1290, 1291, 1, 0, 0, 0, 1291, 1293, 1, 0, 0, 0, 1292, 1290, 1, 0, 0, 0, 1293, 1294, 5, 0, 0, 1, 1294, 169, 1, 0, 0, 0, 1295, 1296, 5, 159, 0, 0, 1296, 1297, 3, 116, 58, 0, 1297, 1298, 5, 147, 0, 0, 1298, 1301, 1, 0, 0, 0, 1299, 1301, 5, 158, 0, 0, 1300, 1295, 1, 0, 0, 0, 1300, 1299, 1, 0, 0, 0, 1301, 171, 1, 0, 0, 0, 167, 175, 182, 191, 198, 202, 216, 220, 223, 227, 230, 237, 241, 250, 255, 264, 272, 279, 283, 289, 294, 302, 309, 315, 327, 335, 349, 353, 358, 368, 377, 380, 384, 387, 391, 394, 397, 400, 403, 407, 411, 414, 417, 420, 424, 427, 436, 442, 463, 480, 497, 503, 509, 520, 522, 533, 536, 542, 550, 556, 558, 562, 567, 570, 573, 577, 581, 584, 586, 589, 593, 597, 600, 602, 604, 609, 620, 626, 633, 638, 642, 646, 652, 654, 661, 669, 672, 675, 694, 708, 724, 728, 739, 743, 754, 758, 765, 769, 776, 780, 785, 794, 798, 822, 839, 845, 848, 851, 861, 867, 870, 873, 881, 884, 888, 891, 905, 922, 927, 932, 938, 945, 957, 961, 964, 973, 987, 1003, 1032, 1040, 1042, 1044, 1053, 1057, 1066, 1070, 1074, 1079, 1086, 1097, 1106, 1113, 1126, 1133, 1137, 1149, 1155, 1158, 1165, 1177, 1183, 1187, 1193, 1200, 1209, 1213, 1223, 1225, 1228, 1236, 1241, 1251, 1256, 1268, 1274, 1284, 1290, 1300] \ No newline at end of file diff --git a/hogql_parser/HogQLParserBaseVisitor.cpp b/hogql_parser/HogQLParserBaseVisitor.cpp index 03bf38add14d5..b5a8af7e0f7aa 100644 --- a/hogql_parser/HogQLParserBaseVisitor.cpp +++ b/hogql_parser/HogQLParserBaseVisitor.cpp @@ -1,5 +1,5 @@ -// Generated from HogQLParser.g4 by ANTLR 4.13.1 +// Generated from HogQLParser.g4 by ANTLR 4.13.2 #include "HogQLParserBaseVisitor.h" diff --git a/hogql_parser/HogQLParserBaseVisitor.h b/hogql_parser/HogQLParserBaseVisitor.h index 7329e835cad57..80b4d5d487765 100644 --- a/hogql_parser/HogQLParserBaseVisitor.h +++ b/hogql_parser/HogQLParserBaseVisitor.h @@ -1,5 +1,5 @@ -// Generated from HogQLParser.g4 by ANTLR 4.13.1 +// Generated from HogQLParser.g4 by ANTLR 4.13.2 #pragma once @@ -375,6 +375,10 @@ class HogQLParserBaseVisitor : public HogQLParserVisitor { return visitChildren(ctx); } + virtual std::any visitColumnExprCall(HogQLParser::ColumnExprCallContext *ctx) override { + return visitChildren(ctx); + } + virtual std::any visitColumnExprArrayAccess(HogQLParser::ColumnExprArrayAccessContext *ctx) override { return visitChildren(ctx); } @@ -427,23 +431,19 @@ class HogQLParserBaseVisitor : public HogQLParserVisitor { return visitChildren(ctx); } - virtual std::any visitColumnExprIdentifier(HogQLParser::ColumnExprIdentifierContext *ctx) override { - return visitChildren(ctx); - } - - virtual std::any visitColumnExprFunction(HogQLParser::ColumnExprFunctionContext *ctx) override { + virtual std::any visitColumnExprLambda(HogQLParser::ColumnExprLambdaContext *ctx) override { return visitChildren(ctx); } - virtual std::any visitColumnExprAsterisk(HogQLParser::ColumnExprAsteriskContext *ctx) override { + virtual std::any visitColumnExprIdentifier(HogQLParser::ColumnExprIdentifierContext *ctx) override { return visitChildren(ctx); } - virtual std::any visitColumnArgList(HogQLParser::ColumnArgListContext *ctx) override { + virtual std::any visitColumnExprFunction(HogQLParser::ColumnExprFunctionContext *ctx) override { return visitChildren(ctx); } - virtual std::any visitColumnArgExpr(HogQLParser::ColumnArgExprContext *ctx) override { + virtual std::any visitColumnExprAsterisk(HogQLParser::ColumnExprAsteriskContext *ctx) override { return visitChildren(ctx); } diff --git a/hogql_parser/HogQLParserVisitor.cpp b/hogql_parser/HogQLParserVisitor.cpp index 82a2f14e4ec57..07a511ac187fc 100644 --- a/hogql_parser/HogQLParserVisitor.cpp +++ b/hogql_parser/HogQLParserVisitor.cpp @@ -1,5 +1,5 @@ -// Generated from HogQLParser.g4 by ANTLR 4.13.1 +// Generated from HogQLParser.g4 by ANTLR 4.13.2 #include "HogQLParserVisitor.h" diff --git a/hogql_parser/HogQLParserVisitor.h b/hogql_parser/HogQLParserVisitor.h index 8e4259a8209b1..0b9e1797ad836 100644 --- a/hogql_parser/HogQLParserVisitor.h +++ b/hogql_parser/HogQLParserVisitor.h @@ -1,5 +1,5 @@ -// Generated from HogQLParser.g4 by ANTLR 4.13.1 +// Generated from HogQLParser.g4 by ANTLR 4.13.2 #pragma once @@ -199,6 +199,8 @@ class HogQLParserVisitor : public antlr4::tree::AbstractParseTreeVisitor { virtual std::any visitColumnExprTuple(HogQLParser::ColumnExprTupleContext *context) = 0; + virtual std::any visitColumnExprCall(HogQLParser::ColumnExprCallContext *context) = 0; + virtual std::any visitColumnExprArrayAccess(HogQLParser::ColumnExprArrayAccessContext *context) = 0; virtual std::any visitColumnExprBetween(HogQLParser::ColumnExprBetweenContext *context) = 0; @@ -225,16 +227,14 @@ class HogQLParserVisitor : public antlr4::tree::AbstractParseTreeVisitor { virtual std::any visitColumnExprWinFunction(HogQLParser::ColumnExprWinFunctionContext *context) = 0; + virtual std::any visitColumnExprLambda(HogQLParser::ColumnExprLambdaContext *context) = 0; + virtual std::any visitColumnExprIdentifier(HogQLParser::ColumnExprIdentifierContext *context) = 0; virtual std::any visitColumnExprFunction(HogQLParser::ColumnExprFunctionContext *context) = 0; virtual std::any visitColumnExprAsterisk(HogQLParser::ColumnExprAsteriskContext *context) = 0; - virtual std::any visitColumnArgList(HogQLParser::ColumnArgListContext *context) = 0; - - virtual std::any visitColumnArgExpr(HogQLParser::ColumnArgExprContext *context) = 0; - virtual std::any visitColumnLambdaExpr(HogQLParser::ColumnLambdaExprContext *context) = 0; virtual std::any visitHogqlxTagElementClosed(HogQLParser::HogqlxTagElementClosedContext *context) = 0; diff --git a/hogql_parser/parser.cpp b/hogql_parser/parser.cpp index 54a1eeedd8e29..409f2fdb46f7b 100644 --- a/hogql_parser/parser.cpp +++ b/hogql_parser/parser.cpp @@ -2151,13 +2151,13 @@ class HogQLParseTreeConverter : public HogQLParserBaseVisitor { VISIT(ColumnExprNot) { RETURN_NEW_AST_NODE("Not", "{s:N}", "expr", visitAsPyObject(ctx->columnExpr())); } VISIT(ColumnExprWinFunctionTarget) { - auto column_expr_list_ctx = ctx->columnExprList(); + auto column_expr_list_ctx = ctx->columnExprs; string name = visitAsString(ctx->identifier(0)); string over_identifier = visitAsString(ctx->identifier(1)); PyObject* exprs = visitAsPyObjectOrEmptyList(column_expr_list_ctx); PyObject* args; try { - args = visitAsPyObjectOrEmptyList(ctx->columnArgList()); + args = visitAsPyObjectOrEmptyList(ctx->columnArgList); } catch (...) { Py_DECREF(exprs); throw; @@ -2170,11 +2170,11 @@ class HogQLParseTreeConverter : public HogQLParserBaseVisitor { VISIT(ColumnExprWinFunction) { string identifier = visitAsString(ctx->identifier()); - auto column_expr_list_ctx = ctx->columnExprList(); + auto column_expr_list_ctx = ctx->columnExprs; PyObject* exprs = visitAsPyObjectOrEmptyList(column_expr_list_ctx); PyObject* args; try { - args = visitAsPyObjectOrEmptyList(ctx->columnArgList()); + args = visitAsPyObjectOrEmptyList(ctx->columnArgList); } catch (...) { Py_DECREF(exprs); throw; @@ -2197,10 +2197,18 @@ class HogQLParseTreeConverter : public HogQLParserBaseVisitor { VISIT(ColumnExprFunction) { string name = visitAsString(ctx->identifier()); - PyObject* params = visitAsPyObjectOrNone(ctx->columnExprList()); + + // if two LPARENs ()(), make sure the first one is at least an empty list + PyObject* params; + if (ctx->LPAREN(1)) { + params = visitAsPyObjectOrEmptyList(ctx->columnExprs); + } else { + params = visitAsPyObjectOrNone(ctx->columnExprs); + } + PyObject* args; try { - args = visitAsPyObjectOrEmptyList(ctx->columnArgList()); + args = visitAsPyObjectOrEmptyList(ctx->columnArgList); } catch (...) { Py_DECREF(params); throw; @@ -2223,10 +2231,18 @@ class HogQLParseTreeConverter : public HogQLParserBaseVisitor { VISIT(ColumnExprTagElement) { return visit(ctx->hogqlxTagElement()); } - VISIT(ColumnArgList) { return visitPyListOfObjects(ctx->columnArgExpr()); } - VISIT(ColumnLambdaExpr) { - PyObject* expr = visitAsPyObject(ctx->columnExpr()); + PyObject* expr; + auto column_expr_ctx = ctx->columnExpr(); + auto block_ctx = ctx->block(); + if (!column_expr_ctx && !block_ctx) { + throw ParsingError("ColumnLambdaExpr must have either a columnExpr or a block"); + } + if (column_expr_ctx) { + expr = visitAsPyObject(column_expr_ctx); + } else { + expr = visitAsPyObject(block_ctx); + } PyObject* args; try { args = X_PyList_FromStrings(visitAsVectorOfStrings(ctx->identifier())); @@ -2575,6 +2591,18 @@ class HogQLParseTreeConverter : public HogQLParserBaseVisitor { RETURN_NEW_AST_NODE("Call", "{s:s, s:[NN]}", "name", "ifNull", "args", value, fallback); } + VISIT(ColumnExprCall) { + PyObject* expr = visitAsPyObject(ctx->columnExpr()); + PyObject* args; + try { + args = visitAsPyObjectOrEmptyList(ctx->columnExprList()); + } catch (...) { + Py_DECREF(expr); + throw; + } + RETURN_NEW_AST_NODE("ExprCall", "{s:N, s:N}", "expr", expr, "args", args); + } + VISIT(ColumnExprTemplateString) { return visit(ctx->templateString()); } VISIT(String) { diff --git a/hogql_parser/setup.py b/hogql_parser/setup.py index 1eb433f6d1d9a..ba14c773be681 100644 --- a/hogql_parser/setup.py +++ b/hogql_parser/setup.py @@ -32,7 +32,7 @@ setup( name="hogql_parser", - version="1.0.36", + version="1.0.38", url="https://github.com/PostHog/posthog/tree/master/hogql_parser", author="PostHog Inc.", author_email="hey@posthog.com", diff --git a/latest_migrations.manifest b/latest_migrations.manifest index 323c6228e1202..af30ae6589b0d 100644 --- a/latest_migrations.manifest +++ b/latest_migrations.manifest @@ -5,7 +5,7 @@ contenttypes: 0002_remove_content_type_name ee: 0016_rolemembership_organization_member otp_static: 0002_throttling otp_totp: 0002_auto_20190420_0723 -posthog: 0459_convert_personsnode_insights_to_actorsquery +posthog: 0460_alertconfiguration_threshold_alertsubscription_and_more sessions: 0001_initial social_django: 0010_uid_db_index two_factor: 0007_auto_20201201_1019 diff --git a/mypy-baseline.txt b/mypy-baseline.txt index bf03484babdbf..4ec481c6b94fa 100644 --- a/mypy-baseline.txt +++ b/mypy-baseline.txt @@ -141,36 +141,23 @@ posthog/hogql_queries/utils/query_date_range.py:0: error: Incompatible default f posthog/hogql_queries/utils/query_date_range.py:0: note: PEP 484 prohibits implicit Optional. Accordingly, mypy has changed its default to no_implicit_optional=True posthog/hogql_queries/utils/query_date_range.py:0: note: Use https://github.com/hauntsaninja/no_implicit_optional to automatically upgrade your codebase posthog/hogql_queries/utils/query_date_range.py:0: error: Item "None" of "IntervalType | None" has no attribute "name" [union-attr] -posthog/hogql/resolver.py:0: error: Argument 1 of "visit" is incompatible with supertype "Visitor"; supertype defines the argument type as "AST | None" [override] -posthog/hogql/resolver.py:0: note: This violates the Liskov substitution principle -posthog/hogql/resolver.py:0: note: See https://mypy.readthedocs.io/en/stable/common_issues.html#incompatible-overrides posthog/hogql/resolver.py:0: error: List comprehension has incompatible type List[SelectQueryType | None]; expected List[SelectQueryType] [misc] -posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "Expr", variable has type "JoinExpr | None") [assignment] posthog/hogql/resolver.py:0: error: Need type annotation for "columns_with_visible_alias" (hint: "columns_with_visible_alias: dict[, ] = ...") [var-annotated] -posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "Type | None", target has type "Type") [assignment] -posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "Type | None", target has type "Type") [assignment] -posthog/hogql/resolver.py:0: error: List comprehension has incompatible type List[Expr]; expected List[OrderExpr] [misc] -posthog/hogql/resolver.py:0: error: Value expression in dictionary comprehension has incompatible type "Expr"; expected type "WindowExpr" [misc] posthog/hogql/resolver.py:0: error: Statement is unreachable [unreachable] posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "Expr", variable has type "SelectQuery | SelectUnionQuery | Field | None") [assignment] posthog/hogql/resolver.py:0: error: Item "None" of "Database | None" has no attribute "get_table" [union-attr] -posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "Expr", variable has type "JoinExpr") [assignment] posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "TableType", variable has type "LazyTableType") [assignment] posthog/hogql/resolver.py:0: error: Argument "table_type" to "TableAliasType" has incompatible type "LazyTableType"; expected "TableType" [arg-type] posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "LazyTableType", variable has type "TableAliasType") [assignment] posthog/hogql/resolver.py:0: error: Argument 1 to "clone_expr" has incompatible type "SelectQuery | SelectUnionQuery | Field | None"; expected "Expr" [arg-type] -posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "Expr", variable has type "JoinExpr | None") [assignment] posthog/hogql/resolver.py:0: error: Statement is unreachable [unreachable] posthog/hogql/resolver.py:0: error: Item "None" of "JoinExpr | None" has no attribute "join_type" [union-attr] -posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "Expr", variable has type "SampleExpr | None") [assignment] posthog/hogql/resolver.py:0: error: Argument "select_query_type" to "SelectViewType" has incompatible type "SelectQueryType | None"; expected "SelectQueryType | SelectUnionQueryType" [arg-type] posthog/hogql/resolver.py:0: error: Item "None" of "SelectQuery | SelectUnionQuery | Field | None" has no attribute "type" [union-attr] posthog/hogql/resolver.py:0: error: Argument "select_query_type" to "SelectQueryAliasType" has incompatible type "Type | Any | None"; expected "SelectQueryType | SelectUnionQueryType" [arg-type] posthog/hogql/resolver.py:0: error: Item "None" of "SelectQuery | SelectUnionQuery | Field | None" has no attribute "type" [union-attr] posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "Type | Any | None", variable has type "BaseTableType | SelectUnionQueryType | SelectQueryType | SelectQueryAliasType | SelectViewType | None") [assignment] posthog/hogql/resolver.py:0: error: Argument 1 to "append" of "list" has incompatible type "BaseTableType | SelectUnionQueryType | SelectQueryType | SelectQueryAliasType | SelectViewType | None"; expected "SelectQueryType | SelectUnionQueryType" [arg-type] -posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "Expr", variable has type "JoinExpr | None") [assignment] -posthog/hogql/resolver.py:0: error: Incompatible types in assignment (expression has type "Expr", variable has type "SampleExpr | None") [assignment] posthog/hogql/resolver.py:0: error: Statement is unreachable [unreachable] posthog/hogql/resolver.py:0: error: Statement is unreachable [unreachable] posthog/hogql/resolver.py:0: error: Item "None" of "Type | None" has no attribute "resolve_constant_type" [union-attr] @@ -277,8 +264,6 @@ posthog/demo/matrix/taxonomy_inference.py:0: error: Name "timezone.datetime" is posthog/demo/matrix/matrix.py:0: error: Name "timezone.datetime" is not defined [name-defined] posthog/demo/matrix/matrix.py:0: error: Name "timezone.datetime" is not defined [name-defined] posthog/demo/matrix/matrix.py:0: error: Name "timezone.datetime" is not defined [name-defined] -posthog/api/utils.py:0: error: Incompatible types in assignment (expression has type "type[EventDefinition]", variable has type "type[EnterpriseEventDefinition]") [assignment] -posthog/api/utils.py:0: error: Argument 1 to "UUID" has incompatible type "int | str"; expected "str | None" [arg-type] posthog/api/shared.py:0: error: Incompatible return value type (got "int | None", expected "Level | None") [return-value] ee/billing/quota_limiting.py:0: error: Argument 2 to "feature_enabled" has incompatible type "UUID"; expected "str" [arg-type] ee/billing/quota_limiting.py:0: error: List comprehension has incompatible type List[int]; expected List[str] [misc] @@ -306,7 +291,6 @@ posthog/tasks/email.py:0: error: Module "django.utils.timezone" does not explici posthog/tasks/email.py:0: error: Argument "email" to "add_recipient" of "EmailMessage" has incompatible type "str | None"; expected "str" [arg-type] posthog/tasks/email.py:0: error: Argument 1 to "capture" has incompatible type "str | None"; expected "str" [arg-type] posthog/tasks/email.py:0: error: Incompatible types in assignment (expression has type "Team | None", variable has type "Team") [assignment] -posthog/api/routing.py:0: error: Incompatible return value type (got "Project | None", expected "Project") [return-value] posthog/api/documentation.py:0: error: Signature of "run_validation" incompatible with supertype "Field" [override] posthog/api/documentation.py:0: note: Superclass: posthog/api/documentation.py:0: note: def run_validation(self, data: Any = ...) -> Any @@ -326,21 +310,24 @@ posthog/models/property/util.py:0: error: Argument 3 to "format_cohort_subquery" posthog/models/property/util.py:0: error: Argument 1 to "append" of "list" has incompatible type "str | int"; expected "str" [arg-type] posthog/models/property/util.py:0: error: Argument 1 to "append" of "list" has incompatible type "str | int"; expected "str" [arg-type] posthog/models/property/util.py:0: error: Argument 1 to "append" of "list" has incompatible type "str | int"; expected "str" [arg-type] -posthog/api/organization.py:0: error: Incompatible return value type (got "int | None", expected "Level | None") [return-value] +posthog/api/utils.py:0: error: Incompatible types in assignment (expression has type "type[EventDefinition]", variable has type "type[EnterpriseEventDefinition]") [assignment] +posthog/api/utils.py:0: error: Argument 1 to "UUID" has incompatible type "int | str"; expected "str | None" [arg-type] posthog/api/email_verification.py:0: error: Argument 2 to "feature_enabled" has incompatible type "UUID"; expected "str" [arg-type] -posthog/api/capture.py:0: error: Module has no attribute "utc" [attr-defined] posthog/queries/trends/util.py:0: error: Argument 1 to "translate_hogql" has incompatible type "str | None"; expected "str" [arg-type] posthog/hogql/property.py:0: error: Incompatible type for lookup 'id': (got "str | int | list[str]", expected "str | int") [misc] posthog/hogql/property.py:0: error: Incompatible type for lookup 'pk': (got "str | float", expected "str | int") [misc] -posthog/api/action.py:0: error: Argument 1 to has incompatible type "*tuple[str, ...]"; expected "type[BaseRenderer]" [arg-type] +posthog/api/routing.py:0: error: Incompatible return value type (got "Project | None", expected "Project") [return-value] +posthog/api/capture.py:0: error: Module has no attribute "utc" [attr-defined] posthog/hogql/filters.py:0: error: Incompatible default for argument "team" (default has type "None", argument has type "Team") [assignment] posthog/hogql/filters.py:0: note: PEP 484 prohibits implicit Optional. Accordingly, mypy has changed its default to no_implicit_optional=True posthog/hogql/filters.py:0: note: Use https://github.com/hauntsaninja/no_implicit_optional to automatically upgrade your codebase +posthog/api/organization.py:0: error: Incompatible return value type (got "int | None", expected "Level | None") [return-value] posthog/hogql/query.py:0: error: Incompatible types in assignment (expression has type "None", variable has type "str | SelectQuery | SelectUnionQuery") [assignment] posthog/hogql/query.py:0: error: Incompatible types in assignment (expression has type "Expr", variable has type "SelectQuery | SelectUnionQuery") [assignment] posthog/hogql/query.py:0: error: Argument 1 to "get_default_limit_for_context" has incompatible type "LimitContext | None"; expected "LimitContext" [arg-type] posthog/hogql/query.py:0: error: "SelectQuery" has no attribute "select_queries" [attr-defined] posthog/hogql/query.py:0: error: Subclass of "SelectQuery" and "SelectUnionQuery" cannot exist: would have incompatible method signatures [unreachable] +posthog/api/action.py:0: error: Argument 1 to has incompatible type "*tuple[str, ...]"; expected "type[BaseRenderer]" [arg-type] posthog/queries/person_query.py:0: error: Incompatible type for lookup 'pk': (got "str | int | list[str]", expected "str | int") [misc] posthog/queries/event_query/event_query.py:0: error: Incompatible type for lookup 'pk': (got "str | int | list[str]", expected "str | int") [misc] posthog/hogql_queries/sessions_timeline_query_runner.py:0: error: Statement is unreachable [unreachable] @@ -806,24 +793,6 @@ posthog/queries/app_metrics/historical_exports.py:0: error: Argument 1 to "loads posthog/api/test/test_decide.py:0: error: Item "None" of "User | None" has no attribute "toolbar_mode" [union-attr] posthog/api/test/test_decide.py:0: error: Item "None" of "User | None" has no attribute "save" [union-attr] posthog/api/test/test_authentication.py:0: error: Module has no attribute "utc" [attr-defined] -posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "organization" [union-attr] -posthog/api/plugin.py:0: error: Argument "organization_id" to "log_activity" has incompatible type "UUID | Any"; expected "UUIDT | None" [arg-type] -posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "id" [union-attr] -posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "organization" [union-attr] -posthog/api/plugin.py:0: error: Argument "organization_id" to "log_activity" has incompatible type "UUID | Any"; expected "UUIDT | None" [arg-type] -posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "id" [union-attr] -posthog/api/plugin.py:0: error: Incompatible types in assignment (expression has type "str | None", variable has type "str | int | Combinable") [assignment] -posthog/api/plugin.py:0: error: Incompatible types in assignment (expression has type "str | None", variable has type "str | int | Combinable") [assignment] -posthog/api/plugin.py:0: error: Incompatible types in assignment (expression has type "int | None", variable has type "float | int | str | Combinable") [assignment] -posthog/api/plugin.py:0: error: Item "None" of "IO[Any] | None" has no attribute "read" [union-attr] -posthog/api/plugin.py:0: error: Incompatible type for "file_name" of "PluginAttachment" (got "str | None", expected "str | int | Combinable") [misc] -posthog/api/plugin.py:0: error: Incompatible type for "file_size" of "PluginAttachment" (got "int | None", expected "float | int | str | Combinable") [misc] -posthog/api/plugin.py:0: error: Item "None" of "IO[Any] | None" has no attribute "read" [union-attr] -posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "organization" [union-attr] -posthog/api/plugin.py:0: error: Argument "organization_id" to "log_activity" has incompatible type "UUID | Any"; expected "UUIDT | None" [arg-type] -posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "id" [union-attr] -posthog/api/plugin.py:0: error: Argument "organization_id" to "log_activity" has incompatible type "UUID"; expected "UUIDT | None" [arg-type] -posthog/api/plugin.py:0: error: Argument "organization_id" to "log_activity" has incompatible type "UUID"; expected "UUIDT | None" [arg-type] posthog/admin/admins/plugin_config_admin.py:0: error: Item "None" of "Team | None" has no attribute "name" [union-attr] posthog/warehouse/api/external_data_schema.py:0: error: Incompatible return value type (got "str | None", expected "SyncType | None") [return-value] posthog/warehouse/api/external_data_schema.py:0: error: Argument 1 to "get_sql_schemas_for_source_type" has incompatible type "str"; expected "Type" [arg-type] @@ -846,16 +815,30 @@ posthog/temporal/data_imports/workflow_activities/import_data.py:0: error: Argum posthog/temporal/data_imports/workflow_activities/import_data.py:0: error: Argument "source_type" to "sql_source_for_type" has incompatible type "str"; expected "Type" [arg-type] posthog/migrations/0237_remove_timezone_from_teams.py:0: error: Argument 2 to "RunPython" has incompatible type "Callable[[Migration, Any], None]"; expected "_CodeCallable | None" [arg-type] posthog/migrations/0228_fix_tile_layouts.py:0: error: Argument 2 to "RunPython" has incompatible type "Callable[[Migration, Any], None]"; expected "_CodeCallable | None" [arg-type] -posthog/api/plugin_log_entry.py:0: error: Name "timezone.datetime" is not defined [name-defined] -posthog/api/plugin_log_entry.py:0: error: Module "django.utils.timezone" does not explicitly export attribute "datetime" [attr-defined] -posthog/api/plugin_log_entry.py:0: error: Name "timezone.datetime" is not defined [name-defined] -posthog/api/plugin_log_entry.py:0: error: Module "django.utils.timezone" does not explicitly export attribute "datetime" [attr-defined] posthog/warehouse/external_data_source/source.py:0: error: Incompatible types in assignment (expression has type "int", target has type "str") [assignment] posthog/warehouse/external_data_source/source.py:0: error: Incompatible types in assignment (expression has type "int", target has type "str") [assignment] posthog/warehouse/external_data_source/source.py:0: error: Incompatible types in assignment (expression has type "dict[str, Collection[str]]", variable has type "StripeSourcePayload") [assignment] posthog/warehouse/external_data_source/source.py:0: error: Argument 1 to "_create_source" has incompatible type "StripeSourcePayload"; expected "dict[Any, Any]" [arg-type] posthog/temporal/tests/batch_exports/test_redshift_batch_export_workflow.py:0: error: Incompatible types in assignment (expression has type "str | int", variable has type "int") [assignment] posthog/api/sharing.py:0: error: Item "None" of "list[Any] | None" has no attribute "__iter__" (not iterable) [union-attr] +posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "organization" [union-attr] +posthog/api/plugin.py:0: error: Argument "organization_id" to "log_activity" has incompatible type "UUID | Any"; expected "UUIDT | None" [arg-type] +posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "id" [union-attr] +posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "organization" [union-attr] +posthog/api/plugin.py:0: error: Argument "organization_id" to "log_activity" has incompatible type "UUID | Any"; expected "UUIDT | None" [arg-type] +posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "id" [union-attr] +posthog/api/plugin.py:0: error: Incompatible types in assignment (expression has type "str | None", variable has type "str | int | Combinable") [assignment] +posthog/api/plugin.py:0: error: Incompatible types in assignment (expression has type "str | None", variable has type "str | int | Combinable") [assignment] +posthog/api/plugin.py:0: error: Incompatible types in assignment (expression has type "int | None", variable has type "float | int | str | Combinable") [assignment] +posthog/api/plugin.py:0: error: Item "None" of "IO[Any] | None" has no attribute "read" [union-attr] +posthog/api/plugin.py:0: error: Incompatible type for "file_name" of "PluginAttachment" (got "str | None", expected "str | int | Combinable") [misc] +posthog/api/plugin.py:0: error: Incompatible type for "file_size" of "PluginAttachment" (got "int | None", expected "float | int | str | Combinable") [misc] +posthog/api/plugin.py:0: error: Item "None" of "IO[Any] | None" has no attribute "read" [union-attr] +posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "organization" [union-attr] +posthog/api/plugin.py:0: error: Argument "organization_id" to "log_activity" has incompatible type "UUID | Any"; expected "UUIDT | None" [arg-type] +posthog/api/plugin.py:0: error: Item "None" of "Team | None" has no attribute "id" [union-attr] +posthog/api/plugin.py:0: error: Argument "organization_id" to "log_activity" has incompatible type "UUID"; expected "UUIDT | None" [arg-type] +posthog/api/plugin.py:0: error: Argument "organization_id" to "log_activity" has incompatible type "UUID"; expected "UUIDT | None" [arg-type] posthog/temporal/tests/external_data/test_external_data_job.py:0: error: Invalid index type "str" for "dict[Type, Sequence[str]]"; expected type "Type" [index] posthog/temporal/tests/external_data/test_external_data_job.py:0: error: Invalid index type "str" for "dict[Type, Sequence[str]]"; expected type "Type" [index] posthog/temporal/tests/external_data/test_external_data_job.py:0: error: Invalid index type "str" for "dict[Type, Sequence[str]]"; expected type "Type" [index] @@ -867,21 +850,13 @@ posthog/api/test/batch_exports/conftest.py:0: note: def run(self) -> Coroutine[A posthog/api/test/batch_exports/conftest.py:0: note: Subclass: posthog/api/test/batch_exports/conftest.py:0: note: def run(self, loop: Any) -> Any posthog/api/test/batch_exports/conftest.py:0: error: Argument "activities" to "ThreadedWorker" has incompatible type "list[function]"; expected "Sequence[Callable[..., Any]]" [arg-type] +posthog/api/plugin_log_entry.py:0: error: Name "timezone.datetime" is not defined [name-defined] +posthog/api/plugin_log_entry.py:0: error: Module "django.utils.timezone" does not explicitly export attribute "datetime" [attr-defined] +posthog/api/plugin_log_entry.py:0: error: Name "timezone.datetime" is not defined [name-defined] +posthog/api/plugin_log_entry.py:0: error: Module "django.utils.timezone" does not explicitly export attribute "datetime" [attr-defined] posthog/temporal/tests/data_imports/test_end_to_end.py:0: error: Unused "type: ignore" comment [unused-ignore] posthog/api/test/test_team.py:0: error: "HttpResponse" has no attribute "json" [attr-defined] posthog/api/test/test_team.py:0: error: "HttpResponse" has no attribute "json" [attr-defined] -posthog/api/test/test_capture.py:0: error: Statement is unreachable [unreachable] -posthog/api/test/test_capture.py:0: error: Incompatible return value type (got "_MonkeyPatchedWSGIResponse", expected "HttpResponse") [return-value] -posthog/api/test/test_capture.py:0: error: Module has no attribute "utc" [attr-defined] -posthog/api/test/test_capture.py:0: error: Unpacked dict entry 0 has incompatible type "Collection[str]"; expected "SupportsKeysAndGetItem[str, dict[Never, Never]]" [dict-item] -posthog/api/test/test_capture.py:0: error: Unpacked dict entry 0 has incompatible type "Collection[str]"; expected "SupportsKeysAndGetItem[str, dict[Never, Never]]" [dict-item] -posthog/api/test/test_capture.py:0: error: Unpacked dict entry 0 has incompatible type "Collection[str]"; expected "SupportsKeysAndGetItem[str, dict[Never, Never]]" [dict-item] -posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] -posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] -posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] -posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] -posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] -posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] posthog/test/test_middleware.py:0: error: Incompatible types in assignment (expression has type "_MonkeyPatchedWSGIResponse", variable has type "_MonkeyPatchedResponse") [assignment] posthog/management/commands/test/test_create_batch_export_from_app.py:0: error: Incompatible return value type (got "dict[str, Collection[str]]", expected "dict[str, str]") [return-value] posthog/management/commands/test/test_create_batch_export_from_app.py:0: error: Incompatible types in assignment (expression has type "dict[str, Collection[str]]", variable has type "dict[str, str]") [assignment] @@ -917,6 +892,18 @@ posthog/management/commands/test/test_create_batch_export_from_app.py:0: error: posthog/management/commands/test/test_create_batch_export_from_app.py:0: note: Possible overload variants: posthog/management/commands/test/test_create_batch_export_from_app.py:0: note: def __getitem__(self, SupportsIndex, /) -> str posthog/management/commands/test/test_create_batch_export_from_app.py:0: note: def __getitem__(self, slice, /) -> list[str] +posthog/api/test/test_capture.py:0: error: Statement is unreachable [unreachable] +posthog/api/test/test_capture.py:0: error: Incompatible return value type (got "_MonkeyPatchedWSGIResponse", expected "HttpResponse") [return-value] +posthog/api/test/test_capture.py:0: error: Module has no attribute "utc" [attr-defined] +posthog/api/test/test_capture.py:0: error: Unpacked dict entry 0 has incompatible type "Collection[str]"; expected "SupportsKeysAndGetItem[str, dict[Never, Never]]" [dict-item] +posthog/api/test/test_capture.py:0: error: Unpacked dict entry 0 has incompatible type "Collection[str]"; expected "SupportsKeysAndGetItem[str, dict[Never, Never]]" [dict-item] +posthog/api/test/test_capture.py:0: error: Unpacked dict entry 0 has incompatible type "Collection[str]"; expected "SupportsKeysAndGetItem[str, dict[Never, Never]]" [dict-item] +posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] +posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] +posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] +posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] +posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] +posthog/api/test/test_capture.py:0: error: Dict entry 0 has incompatible type "str": "float"; expected "str": "int" [dict-item] posthog/api/test/batch_exports/test_update.py:0: error: Unsupported target for indexed assignment ("Collection[str]") [index] posthog/api/test/batch_exports/test_update.py:0: error: Unsupported target for indexed assignment ("Collection[str]") [index] posthog/api/test/batch_exports/test_update.py:0: error: Dict entry 1 has incompatible type "str": "dict[str, Collection[str]]"; expected "str": "str" [dict-item] diff --git a/posthog/api/__init__.py b/posthog/api/__init__.py index 1e6af969cbafd..9d61824e95d13 100644 --- a/posthog/api/__init__.py +++ b/posthog/api/__init__.py @@ -383,6 +383,20 @@ def api_not_found(request): ["team_id", "insight_id"], ) +project_insights_router.register( + "thresholds", + alert.ThresholdViewSet, + "project_insight_thresholds", + ["team_id", "insight_id"], +) + +project_insights_router.register( + "alerts", + alert.AlertViewSet, + "project_insight_alerts", + ["team_id", "insight_id"], +) + project_session_recordings_router.register( r"sharing", sharing.SharingConfigurationViewSet, diff --git a/posthog/api/alert.py b/posthog/api/alert.py index 8560b0304e198..0a655a8bd3d9d 100644 --- a/posthog/api/alert.py +++ b/posthog/api/alert.py @@ -1,37 +1,208 @@ from rest_framework import serializers, viewsets from rest_framework.exceptions import ValidationError +from rest_framework.response import Response from django.db.models import QuerySet from posthog.api.routing import TeamAndOrgViewSetMixin -from posthog.models.alert import Alert +from posthog.api.shared import UserBasicSerializer +from posthog.models import User +from posthog.models.alert import ( + AlertConfiguration, + AlertCheck, + Threshold, + AlertSubscription, + are_alerts_supported_for_insight, +) + + +class ThresholdSerializer(serializers.ModelSerializer): + class Meta: + model = Threshold + fields = [ + "id", + "created_at", + "name", + "configuration", + ] + read_only_fields = [ + "id", + "created_at", + ] + + def validate(self, data): + instance = Threshold(**data) + instance.clean() + return data + + +class AlertCheckSerializer(serializers.ModelSerializer): + targets_notified = serializers.SerializerMethodField() + + class Meta: + model = AlertCheck + fields = [ + "id", + "created_at", + "calculated_value", + "state", + "targets_notified", + ] + read_only_fields = fields + + def get_targets_notified(self, instance: AlertCheck) -> bool: + return instance.targets_notified != {} + + +class AlertSubscriptionSerializer(serializers.ModelSerializer): + user = serializers.PrimaryKeyRelatedField(queryset=User.objects.filter(is_active=True), required=True) + + class Meta: + model = AlertSubscription + fields = ["id", "user", "alert_configuration"] + read_only_fields = ["id", "alert_configuration"] + + def validate(self, data): + user: User = data["user"] + alert_configuration = data["alert_configuration"] + + if not user.teams.filter(pk=alert_configuration.team_id).exists(): + raise serializers.ValidationError("User does not belong to the same organization as the alert's team.") + + return data class AlertSerializer(serializers.ModelSerializer): + created_by = UserBasicSerializer(read_only=True) + checks = AlertCheckSerializer(many=True, read_only=True) + threshold = ThresholdSerializer() + subscribed_users = serializers.PrimaryKeyRelatedField( + queryset=User.objects.filter(is_active=True), + many=True, + required=True, + write_only=True, + allow_empty=False, + ) + class Meta: - model = Alert + model = AlertConfiguration fields = [ "id", + "created_by", + "created_at", "insight", "name", - "target_value", - "anomaly_condition", + "subscribed_users", + "threshold", + "condition", + "state", + "enabled", + "last_notified_at", + "checks", + ] + read_only_fields = [ + "id", + "created_at", + "state", + "last_notified_at", ] - read_only_fields = ["id"] - def create(self, validated_data: dict) -> Alert: + def to_representation(self, instance): + data = super().to_representation(instance) + data["subscribed_users"] = UserBasicSerializer(instance.subscribed_users.all(), many=True, read_only=True).data + return data + + def add_threshold(self, threshold_data, validated_data): + threshold_instance = Threshold.objects.create( + **threshold_data, + team_id=self.context["team_id"], + created_by=self.context["request"].user, + insight_id=validated_data["insight"].id, + ) + return threshold_instance + + def create(self, validated_data: dict) -> AlertConfiguration: validated_data["team_id"] = self.context["team_id"] - instance: Alert = super().create(validated_data) + validated_data["created_by"] = self.context["request"].user + subscribed_users = validated_data.pop("subscribed_users") + threshold_data = validated_data.pop("threshold", None) + + if threshold_data: + threshold_instance = self.add_threshold(threshold_data, validated_data) + validated_data["threshold"] = threshold_instance + + instance: AlertConfiguration = super().create(validated_data) + + for user in subscribed_users: + AlertSubscription.objects.create( + user=user, alert_configuration=instance, created_by=self.context["request"].user + ) + return instance + def update(self, instance, validated_data): + conditions_or_threshold_changed = False + + threshold_data = validated_data.pop("threshold", None) + if threshold_data is not None: + if threshold_data == {}: + instance.threshold = None + conditions_or_threshold_changed = True + elif instance.threshold: + previous_threshold_configuration = instance.threshold.configuration + threshold_instance = instance.threshold + for key, value in threshold_data.items(): + setattr(threshold_instance, key, value) + threshold_instance.save() + if previous_threshold_configuration != threshold_instance.configuration: + conditions_or_threshold_changed = True + else: + threshold_instance = self.add_threshold(threshold_data, validated_data) + validated_data["threshold"] = threshold_instance + conditions_or_threshold_changed = True + + subscribed_users = validated_data.pop("subscribed_users", None) + if subscribed_users is not None: + AlertSubscription.objects.filter(alert_configuration=instance).exclude(user__in=subscribed_users).delete() + for user in subscribed_users: + AlertSubscription.objects.get_or_create( + user=user, alert_configuration=instance, defaults={"created_by": self.context["request"].user} + ) + + if conditions_or_threshold_changed: + # If anything changed we set inactive, so it's firing and notifying with the new settings + instance.state = "inactive" + + return super().update(instance, validated_data) + + def validate_insight(self, value): + if value and not are_alerts_supported_for_insight(value): + raise ValidationError("Alerts are not supported for this insight.") + return value + + def validate_subscribed_users(self, value): + for user in value: + if not user.teams.filter(pk=self.context["team_id"]).exists(): + raise ValidationError("User does not belong to the same organization as the alert's team.") + return value + def validate(self, attrs): if attrs.get("insight") and attrs["insight"].team.id != self.context["team_id"]: raise ValidationError({"insight": ["This insight does not belong to your team."]}) + + if attrs.get("enabled") is not False and ( + AlertConfiguration.objects.filter(team_id=self.context["team_id"], enabled=True).count() + >= AlertConfiguration.ALERTS_PER_TEAM + ): + raise ValidationError( + {"alert": [f"Your team has reached the limit of {AlertConfiguration.ALERTS_PER_TEAM} enabled alerts."]} + ) + return attrs class AlertViewSet(TeamAndOrgViewSetMixin, viewsets.ModelViewSet): scope_object = "INTERNAL" - queryset = Alert.objects.all() + queryset = AlertConfiguration.objects.all().order_by("-created_at") serializer_class = AlertSerializer def safely_get_queryset(self, queryset) -> QuerySet: @@ -39,3 +210,23 @@ def safely_get_queryset(self, queryset) -> QuerySet: if "insight" in filters: queryset = queryset.filter(insight_id=filters["insight"]) return queryset + + def retrieve(self, request, *args, **kwargs): + instance = self.get_object() + instance.checks = instance.alertcheck_set.all().order_by("-created_at")[:5] + serializer = self.get_serializer(instance) + return Response(serializer.data) + + +class ThresholdWithAlertSerializer(ThresholdSerializer): + alerts = AlertSerializer(many=True, read_only=True, source="alertconfiguration_set") + + class Meta(ThresholdSerializer.Meta): + fields = [*ThresholdSerializer.Meta.fields, "alerts"] + read_only_fields = [*ThresholdSerializer.Meta.read_only_fields, "alerts"] + + +class ThresholdViewSet(TeamAndOrgViewSetMixin, viewsets.ReadOnlyModelViewSet): + scope_object = "INTERNAL" + queryset = Threshold.objects.all() + serializer_class = ThresholdWithAlertSerializer diff --git a/posthog/api/insight.py b/posthog/api/insight.py index 0073c2e67a563..6bf91e1649672 100644 --- a/posthog/api/insight.py +++ b/posthog/api/insight.py @@ -406,7 +406,7 @@ def update(self, instance: Insight, validated_data: dict, **kwargs) -> Insight: updated_insight = super().update(instance, validated_data) if not are_alerts_supported_for_insight(updated_insight): - instance.alert_set.all().delete() + instance.alertconfiguration_set.all().delete() self._log_insight_update(before_update, dashboards_before_change, updated_insight) diff --git a/posthog/api/test/test_alert.py b/posthog/api/test/test_alert.py index 87eab7245829d..49d502256294c 100644 --- a/posthog/api/test/test_alert.py +++ b/posthog/api/test/test_alert.py @@ -28,20 +28,33 @@ def setUp(self): def test_create_and_delete_alert(self) -> None: creation_request = { "insight": self.insight["id"], - "target_value": "test@posthog.com", + "subscribed_users": [ + self.user.id, + ], "name": "alert name", - "anomaly_condition": {}, + "threshold": {"configuration": {}}, } response = self.client.post(f"/api/projects/{self.team.id}/alerts", creation_request) expected_alert_json = { + "condition": {}, + "created_at": mock.ANY, + "created_by": mock.ANY, + "enabled": True, "id": mock.ANY, - "insight": self.insight["id"], - "target_value": "test@posthog.com", + "insight": mock.ANY, + "last_notified_at": None, "name": "alert name", - "anomaly_condition": {}, + "subscribed_users": mock.ANY, + "state": "inactive", + "threshold": { + "configuration": {}, + "created_at": mock.ANY, + "id": mock.ANY, + "name": "", + }, } - assert response.status_code == status.HTTP_201_CREATED + assert response.status_code == status.HTTP_201_CREATED, response.content assert response.json() == expected_alert_json alerts = self.client.get(f"/api/projects/{self.team.id}/alerts") @@ -55,9 +68,11 @@ def test_create_and_delete_alert(self) -> None: def test_incorrect_creation(self) -> None: creation_request = { - "target_value": "test@posthog.com", + "subscribed_users": [ + self.user.id, + ], + "threshold": {"configuration": {}}, "name": "alert name", - "anomaly_condition": {}, } response = self.client.post(f"/api/projects/{self.team.id}/alerts", creation_request) assert response.status_code == status.HTTP_400_BAD_REQUEST @@ -71,9 +86,11 @@ def test_incorrect_creation(self) -> None: ).json() creation_request = { "insight": str(another_team_insight["id"]), - "target_value": "test@posthog.com", + "subscribed_users": [ + self.user.id, + ], + "threshold": {"configuration": {}}, "name": "alert name", - "anomaly_condition": {}, } response = self.client.post(f"/api/projects/{self.team.id}/alerts", creation_request) assert response.status_code == status.HTTP_400_BAD_REQUEST @@ -81,9 +98,11 @@ def test_incorrect_creation(self) -> None: def test_create_and_list_alert(self) -> None: creation_request = { "insight": self.insight["id"], - "target_value": "test@posthog.com", + "subscribed_users": [ + self.user.id, + ], + "threshold": {"configuration": {}}, "name": "alert name", - "anomaly_condition": {}, } alert = self.client.post(f"/api/projects/{self.team.id}/alerts", creation_request).json() @@ -99,15 +118,35 @@ def test_create_and_list_alert(self) -> None: assert list_for_another_insight.status_code == status.HTTP_200_OK assert len(list_for_another_insight.json()["results"]) == 0 + def test_alert_limit(self) -> None: + with mock.patch("posthog.api.alert.AlertConfiguration.ALERTS_PER_TEAM") as alert_limit: + alert_limit.__get__ = mock.Mock(return_value=1) + + creation_request = { + "insight": self.insight["id"], + "subscribed_users": [ + self.user.id, + ], + "threshold": {"configuration": {}}, + "name": "alert name", + } + self.client.post(f"/api/projects/{self.team.id}/alerts", creation_request) + + alert_2 = self.client.post(f"/api/projects/{self.team.id}/alerts", creation_request).json() + + assert alert_2["code"] == "invalid_input" + def test_alert_is_deleted_on_insight_update(self) -> None: another_insight = self.client.post( f"/api/projects/{self.team.id}/insights", data=self.default_insight_data ).json() creation_request = { "insight": another_insight["id"], - "target_value": "test@posthog.com", + "subscribed_users": [ + self.user.id, + ], + "threshold": {"configuration": {}}, "name": "alert name", - "anomaly_condition": {}, } alert = self.client.post(f"/api/projects/{self.team.id}/alerts", creation_request).json() diff --git a/posthog/hogql/ast.py b/posthog/hogql/ast.py index 575085ca1ed45..5758fbe891187 100644 --- a/posthog/hogql/ast.py +++ b/posthog/hogql/ast.py @@ -683,7 +683,7 @@ class Tuple(Expr): @dataclass(kw_only=True) class Lambda(Expr): args: list[str] - expr: Expr + expr: Expr | Block @dataclass(kw_only=True) @@ -718,6 +718,12 @@ class Call(Expr): distinct: bool = False +@dataclass(kw_only=True) +class ExprCall(Expr): + expr: Expr + args: list[Expr] + + @dataclass(kw_only=True) class JoinConstraint(Expr): expr: Expr diff --git a/posthog/hogql/grammar/HogQLLexer.py b/posthog/hogql/grammar/HogQLLexer.py index 1ce68bc24a643..4647d13b29108 100644 --- a/posthog/hogql/grammar/HogQLLexer.py +++ b/posthog/hogql/grammar/HogQLLexer.py @@ -1,4 +1,4 @@ -# Generated from HogQLLexer.g4 by ANTLR 4.13.1 +# Generated from HogQLLexer.g4 by ANTLR 4.13.2 from antlr4 import * from io import StringIO import sys @@ -830,7 +830,7 @@ class HogQLLexer(Lexer): def __init__(self, input=None, output:TextIO = sys.stdout): super().__init__(input, output) - self.checkVersion("4.13.1") + self.checkVersion("4.13.2") self._interp = LexerATNSimulator(self, self.atn, self.decisionsToDFA, PredictionContextCache()) self._actions = None self._predicates = None diff --git a/posthog/hogql/grammar/HogQLParser.g4 b/posthog/hogql/grammar/HogQLParser.g4 index 4a6e187643877..6e86a6f86afd9 100644 --- a/posthog/hogql/grammar/HogQLParser.g4 +++ b/posthog/hogql/grammar/HogQLParser.g4 @@ -151,9 +151,10 @@ columnExpr | SUBSTRING LPAREN columnExpr FROM columnExpr (FOR columnExpr)? RPAREN # ColumnExprSubstring | TIMESTAMP STRING_LITERAL # ColumnExprTimestamp | TRIM LPAREN (BOTH | LEADING | TRAILING) string FROM columnExpr RPAREN # ColumnExprTrim - | identifier (LPAREN columnExprList? RPAREN) (LPAREN DISTINCT? columnArgList? RPAREN)? OVER LPAREN windowExpr RPAREN # ColumnExprWinFunction - | identifier (LPAREN columnExprList? RPAREN) (LPAREN DISTINCT? columnArgList? RPAREN)? OVER identifier # ColumnExprWinFunctionTarget - | identifier (LPAREN columnExprList? RPAREN)? LPAREN DISTINCT? columnArgList? RPAREN # ColumnExprFunction + | identifier (LPAREN columnExprs=columnExprList? RPAREN) (LPAREN DISTINCT? columnArgList=columnExprList? RPAREN)? OVER LPAREN windowExpr RPAREN # ColumnExprWinFunction + | identifier (LPAREN columnExprs=columnExprList? RPAREN) (LPAREN DISTINCT? columnArgList=columnExprList? RPAREN)? OVER identifier # ColumnExprWinFunctionTarget + | identifier (LPAREN columnExprs=columnExprList? RPAREN)? LPAREN DISTINCT? columnArgList=columnExprList? RPAREN # ColumnExprFunction + | columnExpr LPAREN columnExprList? RPAREN # ColumnExprCall | hogqlxTagElement # ColumnExprTagElement | templateString # ColumnExprTemplateString | literal # ColumnExprLiteral @@ -198,24 +199,23 @@ columnExpr // TODO(ilezhankin): `BETWEEN a AND b AND c` is parsed in a wrong way: `BETWEEN (a AND b) AND c` | columnExpr NOT? BETWEEN columnExpr AND columnExpr # ColumnExprBetween | columnExpr QUERY columnExpr COLON columnExpr # ColumnExprTernaryOp - | columnExpr (AS identifier | AS STRING_LITERAL) # ColumnExprAlias - + | columnExpr (AS identifier | AS STRING_LITERAL) # ColumnExprAlias | (tableIdentifier DOT)? ASTERISK # ColumnExprAsterisk // single-column only | LPAREN selectUnionStmt RPAREN # ColumnExprSubquery // single-column only | LPAREN columnExpr RPAREN # ColumnExprParens // single-column only | LPAREN columnExprList RPAREN # ColumnExprTuple | LBRACKET columnExprList? RBRACKET # ColumnExprArray | LBRACE (kvPairList)? RBRACE # ColumnExprDict + | columnLambdaExpr # ColumnExprLambda | columnIdentifier # ColumnExprIdentifier ; -columnArgList: columnArgExpr (COMMA columnArgExpr)* COMMA?; -columnArgExpr: columnLambdaExpr | columnExpr; columnLambdaExpr: ( LPAREN identifier (COMMA identifier)* COMMA? RPAREN | identifier (COMMA identifier)* COMMA? + | LPAREN RPAREN ) - ARROW columnExpr + ARROW (columnExpr | block) ; diff --git a/posthog/hogql/grammar/HogQLParser.interp b/posthog/hogql/grammar/HogQLParser.interp index 183059ec0fd79..b965cfbb577c7 100644 --- a/posthog/hogql/grammar/HogQLParser.interp +++ b/posthog/hogql/grammar/HogQLParser.interp @@ -382,8 +382,6 @@ expr columnTypeExpr columnExprList columnExpr -columnArgList -columnArgExpr columnLambdaExpr hogqlxTagElement hogqlxTagAttribute @@ -414,4 +412,4 @@ stringContentsFull atn: -[4, 1, 159, 1311, 2, 0, 7, 0, 2, 1, 7, 1, 2, 2, 7, 2, 2, 3, 7, 3, 2, 4, 7, 4, 2, 5, 7, 5, 2, 6, 7, 6, 2, 7, 7, 7, 2, 8, 7, 8, 2, 9, 7, 9, 2, 10, 7, 10, 2, 11, 7, 11, 2, 12, 7, 12, 2, 13, 7, 13, 2, 14, 7, 14, 2, 15, 7, 15, 2, 16, 7, 16, 2, 17, 7, 17, 2, 18, 7, 18, 2, 19, 7, 19, 2, 20, 7, 20, 2, 21, 7, 21, 2, 22, 7, 22, 2, 23, 7, 23, 2, 24, 7, 24, 2, 25, 7, 25, 2, 26, 7, 26, 2, 27, 7, 27, 2, 28, 7, 28, 2, 29, 7, 29, 2, 30, 7, 30, 2, 31, 7, 31, 2, 32, 7, 32, 2, 33, 7, 33, 2, 34, 7, 34, 2, 35, 7, 35, 2, 36, 7, 36, 2, 37, 7, 37, 2, 38, 7, 38, 2, 39, 7, 39, 2, 40, 7, 40, 2, 41, 7, 41, 2, 42, 7, 42, 2, 43, 7, 43, 2, 44, 7, 44, 2, 45, 7, 45, 2, 46, 7, 46, 2, 47, 7, 47, 2, 48, 7, 48, 2, 49, 7, 49, 2, 50, 7, 50, 2, 51, 7, 51, 2, 52, 7, 52, 2, 53, 7, 53, 2, 54, 7, 54, 2, 55, 7, 55, 2, 56, 7, 56, 2, 57, 7, 57, 2, 58, 7, 58, 2, 59, 7, 59, 2, 60, 7, 60, 2, 61, 7, 61, 2, 62, 7, 62, 2, 63, 7, 63, 2, 64, 7, 64, 2, 65, 7, 65, 2, 66, 7, 66, 2, 67, 7, 67, 2, 68, 7, 68, 2, 69, 7, 69, 2, 70, 7, 70, 2, 71, 7, 71, 2, 72, 7, 72, 2, 73, 7, 73, 2, 74, 7, 74, 2, 75, 7, 75, 2, 76, 7, 76, 2, 77, 7, 77, 2, 78, 7, 78, 2, 79, 7, 79, 2, 80, 7, 80, 2, 81, 7, 81, 2, 82, 7, 82, 2, 83, 7, 83, 2, 84, 7, 84, 2, 85, 7, 85, 2, 86, 7, 86, 2, 87, 7, 87, 1, 0, 5, 0, 178, 8, 0, 10, 0, 12, 0, 181, 9, 0, 1, 0, 1, 0, 1, 1, 1, 1, 3, 1, 187, 8, 1, 1, 2, 1, 2, 1, 3, 1, 3, 1, 3, 1, 3, 1, 3, 3, 3, 196, 8, 3, 1, 4, 1, 4, 1, 4, 5, 4, 201, 8, 4, 10, 4, 12, 4, 204, 9, 4, 1, 4, 3, 4, 207, 8, 4, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 3, 5, 221, 8, 5, 1, 6, 1, 6, 3, 6, 225, 8, 6, 1, 6, 3, 6, 228, 8, 6, 1, 7, 1, 7, 3, 7, 232, 8, 7, 1, 7, 3, 7, 235, 8, 7, 1, 8, 1, 8, 1, 8, 1, 8, 1, 8, 3, 8, 242, 8, 8, 1, 8, 1, 8, 3, 8, 246, 8, 8, 1, 8, 1, 8, 1, 9, 1, 9, 1, 9, 5, 9, 253, 8, 9, 10, 9, 12, 9, 256, 9, 9, 1, 9, 1, 9, 3, 9, 260, 8, 9, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 3, 10, 269, 8, 10, 1, 11, 1, 11, 1, 11, 1, 11, 1, 11, 1, 11, 3, 11, 277, 8, 11, 1, 12, 1, 12, 1, 12, 1, 12, 1, 12, 3, 12, 284, 8, 12, 1, 12, 1, 12, 3, 12, 288, 8, 12, 1, 12, 1, 12, 1, 12, 1, 12, 3, 12, 294, 8, 12, 1, 12, 1, 12, 1, 12, 3, 12, 299, 8, 12, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 3, 13, 307, 8, 13, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 3, 13, 314, 8, 13, 1, 14, 1, 14, 1, 14, 1, 14, 3, 14, 320, 8, 14, 1, 14, 1, 14, 1, 14, 1, 15, 1, 15, 1, 15, 1, 15, 1, 15, 1, 16, 1, 16, 3, 16, 332, 8, 16, 1, 17, 1, 17, 1, 18, 1, 18, 5, 18, 338, 8, 18, 10, 18, 12, 18, 341, 9, 18, 1, 18, 1, 18, 1, 19, 1, 19, 1, 19, 1, 19, 1, 20, 1, 20, 1, 20, 5, 20, 352, 8, 20, 10, 20, 12, 20, 355, 9, 20, 1, 20, 3, 20, 358, 8, 20, 1, 21, 1, 21, 1, 21, 3, 21, 363, 8, 21, 1, 21, 1, 21, 1, 22, 1, 22, 1, 22, 1, 22, 5, 22, 371, 8, 22, 10, 22, 12, 22, 374, 9, 22, 1, 23, 1, 23, 1, 23, 1, 23, 1, 23, 1, 23, 3, 23, 382, 8, 23, 1, 24, 3, 24, 385, 8, 24, 1, 24, 1, 24, 3, 24, 389, 8, 24, 1, 24, 3, 24, 392, 8, 24, 1, 24, 1, 24, 3, 24, 396, 8, 24, 1, 24, 3, 24, 399, 8, 24, 1, 24, 3, 24, 402, 8, 24, 1, 24, 3, 24, 405, 8, 24, 1, 24, 3, 24, 408, 8, 24, 1, 24, 1, 24, 3, 24, 412, 8, 24, 1, 24, 1, 24, 3, 24, 416, 8, 24, 1, 24, 3, 24, 419, 8, 24, 1, 24, 3, 24, 422, 8, 24, 1, 24, 3, 24, 425, 8, 24, 1, 24, 1, 24, 3, 24, 429, 8, 24, 1, 24, 3, 24, 432, 8, 24, 1, 25, 1, 25, 1, 25, 1, 26, 1, 26, 1, 26, 1, 26, 3, 26, 441, 8, 26, 1, 27, 1, 27, 1, 27, 1, 28, 3, 28, 447, 8, 28, 1, 28, 1, 28, 1, 28, 1, 28, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 5, 29, 466, 8, 29, 10, 29, 12, 29, 469, 9, 29, 1, 30, 1, 30, 1, 30, 1, 31, 1, 31, 1, 31, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 3, 32, 485, 8, 32, 1, 33, 1, 33, 1, 33, 1, 34, 1, 34, 1, 34, 1, 34, 1, 35, 1, 35, 1, 35, 1, 35, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 502, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 508, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 514, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 525, 8, 36, 3, 36, 527, 8, 36, 1, 37, 1, 37, 1, 37, 1, 38, 1, 38, 1, 38, 1, 39, 1, 39, 1, 39, 3, 39, 538, 8, 39, 1, 39, 3, 39, 541, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 3, 39, 547, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 1, 39, 1, 39, 3, 39, 555, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 5, 39, 561, 8, 39, 10, 39, 12, 39, 564, 9, 39, 1, 40, 3, 40, 567, 8, 40, 1, 40, 1, 40, 1, 40, 3, 40, 572, 8, 40, 1, 40, 3, 40, 575, 8, 40, 1, 40, 3, 40, 578, 8, 40, 1, 40, 1, 40, 3, 40, 582, 8, 40, 1, 40, 1, 40, 3, 40, 586, 8, 40, 1, 40, 3, 40, 589, 8, 40, 3, 40, 591, 8, 40, 1, 40, 3, 40, 594, 8, 40, 1, 40, 1, 40, 3, 40, 598, 8, 40, 1, 40, 1, 40, 3, 40, 602, 8, 40, 1, 40, 3, 40, 605, 8, 40, 3, 40, 607, 8, 40, 3, 40, 609, 8, 40, 1, 41, 1, 41, 1, 41, 3, 41, 614, 8, 41, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 3, 42, 625, 8, 42, 1, 43, 1, 43, 1, 43, 1, 43, 3, 43, 631, 8, 43, 1, 44, 1, 44, 1, 44, 5, 44, 636, 8, 44, 10, 44, 12, 44, 639, 9, 44, 1, 45, 1, 45, 3, 45, 643, 8, 45, 1, 45, 1, 45, 3, 45, 647, 8, 45, 1, 45, 1, 45, 3, 45, 651, 8, 45, 1, 46, 1, 46, 1, 46, 1, 46, 3, 46, 657, 8, 46, 3, 46, 659, 8, 46, 1, 47, 1, 47, 1, 47, 5, 47, 664, 8, 47, 10, 47, 12, 47, 667, 9, 47, 1, 48, 1, 48, 1, 48, 1, 48, 1, 49, 3, 49, 674, 8, 49, 1, 49, 3, 49, 677, 8, 49, 1, 49, 3, 49, 680, 8, 49, 1, 50, 1, 50, 1, 50, 1, 50, 1, 51, 1, 51, 1, 51, 1, 51, 1, 52, 1, 52, 1, 52, 1, 53, 1, 53, 1, 53, 1, 53, 1, 53, 1, 53, 3, 53, 699, 8, 53, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 3, 54, 713, 8, 54, 1, 55, 1, 55, 1, 55, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 727, 8, 56, 10, 56, 12, 56, 730, 9, 56, 1, 56, 3, 56, 733, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 742, 8, 56, 10, 56, 12, 56, 745, 9, 56, 1, 56, 3, 56, 748, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 757, 8, 56, 10, 56, 12, 56, 760, 9, 56, 1, 56, 3, 56, 763, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 3, 56, 770, 8, 56, 1, 56, 1, 56, 3, 56, 774, 8, 56, 1, 57, 1, 57, 1, 57, 5, 57, 779, 8, 57, 10, 57, 12, 57, 782, 9, 57, 1, 57, 3, 57, 785, 8, 57, 1, 58, 1, 58, 1, 58, 3, 58, 790, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 4, 58, 797, 8, 58, 11, 58, 12, 58, 798, 1, 58, 1, 58, 3, 58, 803, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 827, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 844, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 850, 8, 58, 1, 58, 3, 58, 853, 8, 58, 1, 58, 3, 58, 856, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 866, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 872, 8, 58, 1, 58, 3, 58, 875, 8, 58, 1, 58, 3, 58, 878, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 886, 8, 58, 1, 58, 3, 58, 889, 8, 58, 1, 58, 1, 58, 3, 58, 893, 8, 58, 1, 58, 3, 58, 896, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 910, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 927, 8, 58, 1, 58, 1, 58, 1, 58, 3, 58, 932, 8, 58, 1, 58, 1, 58, 3, 58, 936, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 942, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 949, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 961, 8, 58, 1, 58, 1, 58, 3, 58, 965, 8, 58, 1, 58, 3, 58, 968, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 977, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 991, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 1030, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 1038, 8, 58, 5, 58, 1040, 8, 58, 10, 58, 12, 58, 1043, 9, 58, 1, 59, 1, 59, 1, 59, 5, 59, 1048, 8, 59, 10, 59, 12, 59, 1051, 9, 59, 1, 59, 3, 59, 1054, 8, 59, 1, 60, 1, 60, 3, 60, 1058, 8, 60, 1, 61, 1, 61, 1, 61, 1, 61, 5, 61, 1064, 8, 61, 10, 61, 12, 61, 1067, 9, 61, 1, 61, 3, 61, 1070, 8, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 5, 61, 1077, 8, 61, 10, 61, 12, 61, 1080, 9, 61, 1, 61, 3, 61, 1083, 8, 61, 3, 61, 1085, 8, 61, 1, 61, 1, 61, 1, 61, 1, 62, 1, 62, 1, 62, 5, 62, 1093, 8, 62, 10, 62, 12, 62, 1096, 9, 62, 1, 62, 1, 62, 1, 62, 1, 62, 1, 62, 1, 62, 5, 62, 1104, 8, 62, 10, 62, 12, 62, 1107, 9, 62, 1, 62, 1, 62, 1, 62, 1, 62, 1, 62, 1, 62, 3, 62, 1115, 8, 62, 1, 62, 1, 62, 1, 62, 1, 62, 1, 62, 3, 62, 1122, 8, 62, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 3, 63, 1135, 8, 63, 1, 64, 1, 64, 1, 64, 5, 64, 1140, 8, 64, 10, 64, 12, 64, 1143, 9, 64, 1, 64, 3, 64, 1146, 8, 64, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 1, 65, 3, 65, 1158, 8, 65, 1, 66, 1, 66, 1, 66, 1, 66, 3, 66, 1164, 8, 66, 1, 66, 3, 66, 1167, 8, 66, 1, 67, 1, 67, 1, 67, 5, 67, 1172, 8, 67, 10, 67, 12, 67, 1175, 9, 67, 1, 68, 1, 68, 1, 68, 1, 68, 1, 68, 1, 68, 1, 68, 1, 68, 1, 68, 3, 68, 1186, 8, 68, 1, 68, 1, 68, 1, 68, 1, 68, 3, 68, 1192, 8, 68, 5, 68, 1194, 8, 68, 10, 68, 12, 68, 1197, 9, 68, 1, 69, 1, 69, 1, 69, 3, 69, 1202, 8, 69, 1, 69, 1, 69, 1, 70, 1, 70, 1, 70, 3, 70, 1209, 8, 70, 1, 70, 1, 70, 1, 71, 1, 71, 1, 71, 5, 71, 1216, 8, 71, 10, 71, 12, 71, 1219, 9, 71, 1, 71, 3, 71, 1222, 8, 71, 1, 72, 1, 72, 1, 73, 1, 73, 1, 73, 1, 73, 1, 73, 1, 73, 3, 73, 1232, 8, 73, 3, 73, 1234, 8, 73, 1, 74, 3, 74, 1237, 8, 74, 1, 74, 1, 74, 1, 74, 1, 74, 1, 74, 1, 74, 3, 74, 1245, 8, 74, 1, 75, 1, 75, 1, 75, 3, 75, 1250, 8, 75, 1, 76, 1, 76, 1, 77, 1, 77, 1, 78, 1, 78, 1, 79, 1, 79, 3, 79, 1260, 8, 79, 1, 80, 1, 80, 1, 80, 3, 80, 1265, 8, 80, 1, 81, 1, 81, 1, 81, 1, 81, 1, 82, 1, 82, 1, 82, 1, 82, 1, 83, 1, 83, 3, 83, 1277, 8, 83, 1, 84, 1, 84, 5, 84, 1281, 8, 84, 10, 84, 12, 84, 1284, 9, 84, 1, 84, 1, 84, 1, 85, 1, 85, 1, 85, 1, 85, 1, 85, 3, 85, 1293, 8, 85, 1, 86, 1, 86, 5, 86, 1297, 8, 86, 10, 86, 12, 86, 1300, 9, 86, 1, 86, 1, 86, 1, 87, 1, 87, 1, 87, 1, 87, 1, 87, 3, 87, 1309, 8, 87, 1, 87, 0, 3, 78, 116, 136, 88, 0, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28, 30, 32, 34, 36, 38, 40, 42, 44, 46, 48, 50, 52, 54, 56, 58, 60, 62, 64, 66, 68, 70, 72, 74, 76, 78, 80, 82, 84, 86, 88, 90, 92, 94, 96, 98, 100, 102, 104, 106, 108, 110, 112, 114, 116, 118, 120, 122, 124, 126, 128, 130, 132, 134, 136, 138, 140, 142, 144, 146, 148, 150, 152, 154, 156, 158, 160, 162, 164, 166, 168, 170, 172, 174, 0, 16, 2, 0, 18, 18, 74, 74, 2, 0, 44, 44, 51, 51, 3, 0, 1, 1, 4, 4, 8, 8, 4, 0, 1, 1, 3, 4, 8, 8, 80, 80, 2, 0, 51, 51, 73, 73, 2, 0, 1, 1, 4, 4, 2, 0, 7, 7, 22, 23, 2, 0, 30, 30, 49, 49, 2, 0, 71, 71, 76, 76, 3, 0, 10, 10, 50, 50, 90, 90, 2, 0, 41, 41, 53, 53, 1, 0, 107, 108, 2, 0, 118, 118, 139, 139, 7, 0, 21, 21, 38, 38, 55, 56, 70, 70, 78, 78, 97, 97, 103, 103, 16, 0, 1, 13, 15, 20, 22, 28, 30, 30, 32, 37, 39, 42, 44, 51, 53, 54, 58, 58, 60, 69, 71, 77, 79, 83, 85, 92, 94, 96, 98, 99, 101, 102, 4, 0, 20, 20, 30, 30, 39, 39, 48, 48, 1479, 0, 179, 1, 0, 0, 0, 2, 186, 1, 0, 0, 0, 4, 188, 1, 0, 0, 0, 6, 190, 1, 0, 0, 0, 8, 197, 1, 0, 0, 0, 10, 220, 1, 0, 0, 0, 12, 222, 1, 0, 0, 0, 14, 229, 1, 0, 0, 0, 16, 236, 1, 0, 0, 0, 18, 249, 1, 0, 0, 0, 20, 261, 1, 0, 0, 0, 22, 270, 1, 0, 0, 0, 24, 278, 1, 0, 0, 0, 26, 300, 1, 0, 0, 0, 28, 315, 1, 0, 0, 0, 30, 324, 1, 0, 0, 0, 32, 329, 1, 0, 0, 0, 34, 333, 1, 0, 0, 0, 36, 335, 1, 0, 0, 0, 38, 344, 1, 0, 0, 0, 40, 348, 1, 0, 0, 0, 42, 362, 1, 0, 0, 0, 44, 366, 1, 0, 0, 0, 46, 381, 1, 0, 0, 0, 48, 384, 1, 0, 0, 0, 50, 433, 1, 0, 0, 0, 52, 436, 1, 0, 0, 0, 54, 442, 1, 0, 0, 0, 56, 446, 1, 0, 0, 0, 58, 452, 1, 0, 0, 0, 60, 470, 1, 0, 0, 0, 62, 473, 1, 0, 0, 0, 64, 476, 1, 0, 0, 0, 66, 486, 1, 0, 0, 0, 68, 489, 1, 0, 0, 0, 70, 493, 1, 0, 0, 0, 72, 526, 1, 0, 0, 0, 74, 528, 1, 0, 0, 0, 76, 531, 1, 0, 0, 0, 78, 546, 1, 0, 0, 0, 80, 608, 1, 0, 0, 0, 82, 613, 1, 0, 0, 0, 84, 624, 1, 0, 0, 0, 86, 626, 1, 0, 0, 0, 88, 632, 1, 0, 0, 0, 90, 640, 1, 0, 0, 0, 92, 658, 1, 0, 0, 0, 94, 660, 1, 0, 0, 0, 96, 668, 1, 0, 0, 0, 98, 673, 1, 0, 0, 0, 100, 681, 1, 0, 0, 0, 102, 685, 1, 0, 0, 0, 104, 689, 1, 0, 0, 0, 106, 698, 1, 0, 0, 0, 108, 712, 1, 0, 0, 0, 110, 714, 1, 0, 0, 0, 112, 773, 1, 0, 0, 0, 114, 775, 1, 0, 0, 0, 116, 935, 1, 0, 0, 0, 118, 1044, 1, 0, 0, 0, 120, 1057, 1, 0, 0, 0, 122, 1084, 1, 0, 0, 0, 124, 1121, 1, 0, 0, 0, 126, 1134, 1, 0, 0, 0, 128, 1136, 1, 0, 0, 0, 130, 1157, 1, 0, 0, 0, 132, 1166, 1, 0, 0, 0, 134, 1168, 1, 0, 0, 0, 136, 1185, 1, 0, 0, 0, 138, 1198, 1, 0, 0, 0, 140, 1208, 1, 0, 0, 0, 142, 1212, 1, 0, 0, 0, 144, 1223, 1, 0, 0, 0, 146, 1233, 1, 0, 0, 0, 148, 1236, 1, 0, 0, 0, 150, 1249, 1, 0, 0, 0, 152, 1251, 1, 0, 0, 0, 154, 1253, 1, 0, 0, 0, 156, 1255, 1, 0, 0, 0, 158, 1259, 1, 0, 0, 0, 160, 1264, 1, 0, 0, 0, 162, 1266, 1, 0, 0, 0, 164, 1270, 1, 0, 0, 0, 166, 1276, 1, 0, 0, 0, 168, 1278, 1, 0, 0, 0, 170, 1292, 1, 0, 0, 0, 172, 1294, 1, 0, 0, 0, 174, 1308, 1, 0, 0, 0, 176, 178, 3, 2, 1, 0, 177, 176, 1, 0, 0, 0, 178, 181, 1, 0, 0, 0, 179, 177, 1, 0, 0, 0, 179, 180, 1, 0, 0, 0, 180, 182, 1, 0, 0, 0, 181, 179, 1, 0, 0, 0, 182, 183, 5, 0, 0, 1, 183, 1, 1, 0, 0, 0, 184, 187, 3, 6, 3, 0, 185, 187, 3, 10, 5, 0, 186, 184, 1, 0, 0, 0, 186, 185, 1, 0, 0, 0, 187, 3, 1, 0, 0, 0, 188, 189, 3, 116, 58, 0, 189, 5, 1, 0, 0, 0, 190, 191, 5, 52, 0, 0, 191, 195, 3, 160, 80, 0, 192, 193, 5, 115, 0, 0, 193, 194, 5, 122, 0, 0, 194, 196, 3, 4, 2, 0, 195, 192, 1, 0, 0, 0, 195, 196, 1, 0, 0, 0, 196, 7, 1, 0, 0, 0, 197, 202, 3, 160, 80, 0, 198, 199, 5, 116, 0, 0, 199, 201, 3, 160, 80, 0, 200, 198, 1, 0, 0, 0, 201, 204, 1, 0, 0, 0, 202, 200, 1, 0, 0, 0, 202, 203, 1, 0, 0, 0, 203, 206, 1, 0, 0, 0, 204, 202, 1, 0, 0, 0, 205, 207, 5, 116, 0, 0, 206, 205, 1, 0, 0, 0, 206, 207, 1, 0, 0, 0, 207, 9, 1, 0, 0, 0, 208, 221, 3, 12, 6, 0, 209, 221, 3, 14, 7, 0, 210, 221, 3, 18, 9, 0, 211, 221, 3, 20, 10, 0, 212, 221, 3, 22, 11, 0, 213, 221, 3, 26, 13, 0, 214, 221, 3, 24, 12, 0, 215, 221, 3, 28, 14, 0, 216, 221, 3, 30, 15, 0, 217, 221, 3, 36, 18, 0, 218, 221, 3, 32, 16, 0, 219, 221, 3, 34, 17, 0, 220, 208, 1, 0, 0, 0, 220, 209, 1, 0, 0, 0, 220, 210, 1, 0, 0, 0, 220, 211, 1, 0, 0, 0, 220, 212, 1, 0, 0, 0, 220, 213, 1, 0, 0, 0, 220, 214, 1, 0, 0, 0, 220, 215, 1, 0, 0, 0, 220, 216, 1, 0, 0, 0, 220, 217, 1, 0, 0, 0, 220, 218, 1, 0, 0, 0, 220, 219, 1, 0, 0, 0, 221, 11, 1, 0, 0, 0, 222, 224, 5, 72, 0, 0, 223, 225, 3, 4, 2, 0, 224, 223, 1, 0, 0, 0, 224, 225, 1, 0, 0, 0, 225, 227, 1, 0, 0, 0, 226, 228, 5, 150, 0, 0, 227, 226, 1, 0, 0, 0, 227, 228, 1, 0, 0, 0, 228, 13, 1, 0, 0, 0, 229, 231, 5, 84, 0, 0, 230, 232, 3, 4, 2, 0, 231, 230, 1, 0, 0, 0, 231, 232, 1, 0, 0, 0, 232, 234, 1, 0, 0, 0, 233, 235, 5, 150, 0, 0, 234, 233, 1, 0, 0, 0, 234, 235, 1, 0, 0, 0, 235, 15, 1, 0, 0, 0, 236, 245, 5, 14, 0, 0, 237, 238, 5, 130, 0, 0, 238, 241, 3, 160, 80, 0, 239, 240, 5, 115, 0, 0, 240, 242, 3, 160, 80, 0, 241, 239, 1, 0, 0, 0, 241, 242, 1, 0, 0, 0, 242, 243, 1, 0, 0, 0, 243, 244, 5, 149, 0, 0, 244, 246, 1, 0, 0, 0, 245, 237, 1, 0, 0, 0, 245, 246, 1, 0, 0, 0, 246, 247, 1, 0, 0, 0, 247, 248, 3, 36, 18, 0, 248, 17, 1, 0, 0, 0, 249, 250, 5, 93, 0, 0, 250, 254, 3, 36, 18, 0, 251, 253, 3, 16, 8, 0, 252, 251, 1, 0, 0, 0, 253, 256, 1, 0, 0, 0, 254, 252, 1, 0, 0, 0, 254, 255, 1, 0, 0, 0, 255, 259, 1, 0, 0, 0, 256, 254, 1, 0, 0, 0, 257, 258, 5, 29, 0, 0, 258, 260, 3, 36, 18, 0, 259, 257, 1, 0, 0, 0, 259, 260, 1, 0, 0, 0, 260, 19, 1, 0, 0, 0, 261, 262, 5, 40, 0, 0, 262, 263, 5, 130, 0, 0, 263, 264, 3, 4, 2, 0, 264, 265, 5, 149, 0, 0, 265, 268, 3, 10, 5, 0, 266, 267, 5, 25, 0, 0, 267, 269, 3, 10, 5, 0, 268, 266, 1, 0, 0, 0, 268, 269, 1, 0, 0, 0, 269, 21, 1, 0, 0, 0, 270, 271, 5, 100, 0, 0, 271, 272, 5, 130, 0, 0, 272, 273, 3, 4, 2, 0, 273, 274, 5, 149, 0, 0, 274, 276, 3, 10, 5, 0, 275, 277, 5, 150, 0, 0, 276, 275, 1, 0, 0, 0, 276, 277, 1, 0, 0, 0, 277, 23, 1, 0, 0, 0, 278, 279, 5, 33, 0, 0, 279, 283, 5, 130, 0, 0, 280, 284, 3, 6, 3, 0, 281, 284, 3, 30, 15, 0, 282, 284, 3, 4, 2, 0, 283, 280, 1, 0, 0, 0, 283, 281, 1, 0, 0, 0, 283, 282, 1, 0, 0, 0, 283, 284, 1, 0, 0, 0, 284, 285, 1, 0, 0, 0, 285, 287, 5, 150, 0, 0, 286, 288, 3, 4, 2, 0, 287, 286, 1, 0, 0, 0, 287, 288, 1, 0, 0, 0, 288, 289, 1, 0, 0, 0, 289, 293, 5, 150, 0, 0, 290, 294, 3, 6, 3, 0, 291, 294, 3, 30, 15, 0, 292, 294, 3, 4, 2, 0, 293, 290, 1, 0, 0, 0, 293, 291, 1, 0, 0, 0, 293, 292, 1, 0, 0, 0, 293, 294, 1, 0, 0, 0, 294, 295, 1, 0, 0, 0, 295, 296, 5, 149, 0, 0, 296, 298, 3, 10, 5, 0, 297, 299, 5, 150, 0, 0, 298, 297, 1, 0, 0, 0, 298, 299, 1, 0, 0, 0, 299, 25, 1, 0, 0, 0, 300, 301, 5, 33, 0, 0, 301, 302, 5, 130, 0, 0, 302, 303, 5, 52, 0, 0, 303, 306, 3, 160, 80, 0, 304, 305, 5, 116, 0, 0, 305, 307, 3, 160, 80, 0, 306, 304, 1, 0, 0, 0, 306, 307, 1, 0, 0, 0, 307, 308, 1, 0, 0, 0, 308, 309, 5, 42, 0, 0, 309, 310, 3, 4, 2, 0, 310, 311, 5, 149, 0, 0, 311, 313, 3, 10, 5, 0, 312, 314, 5, 150, 0, 0, 313, 312, 1, 0, 0, 0, 313, 314, 1, 0, 0, 0, 314, 27, 1, 0, 0, 0, 315, 316, 5, 31, 0, 0, 316, 317, 3, 160, 80, 0, 317, 319, 5, 130, 0, 0, 318, 320, 3, 8, 4, 0, 319, 318, 1, 0, 0, 0, 319, 320, 1, 0, 0, 0, 320, 321, 1, 0, 0, 0, 321, 322, 5, 149, 0, 0, 322, 323, 3, 36, 18, 0, 323, 29, 1, 0, 0, 0, 324, 325, 3, 4, 2, 0, 325, 326, 5, 115, 0, 0, 326, 327, 5, 122, 0, 0, 327, 328, 3, 4, 2, 0, 328, 31, 1, 0, 0, 0, 329, 331, 3, 4, 2, 0, 330, 332, 5, 150, 0, 0, 331, 330, 1, 0, 0, 0, 331, 332, 1, 0, 0, 0, 332, 33, 1, 0, 0, 0, 333, 334, 5, 150, 0, 0, 334, 35, 1, 0, 0, 0, 335, 339, 5, 128, 0, 0, 336, 338, 3, 2, 1, 0, 337, 336, 1, 0, 0, 0, 338, 341, 1, 0, 0, 0, 339, 337, 1, 0, 0, 0, 339, 340, 1, 0, 0, 0, 340, 342, 1, 0, 0, 0, 341, 339, 1, 0, 0, 0, 342, 343, 5, 147, 0, 0, 343, 37, 1, 0, 0, 0, 344, 345, 3, 4, 2, 0, 345, 346, 5, 115, 0, 0, 346, 347, 3, 4, 2, 0, 347, 39, 1, 0, 0, 0, 348, 353, 3, 38, 19, 0, 349, 350, 5, 116, 0, 0, 350, 352, 3, 38, 19, 0, 351, 349, 1, 0, 0, 0, 352, 355, 1, 0, 0, 0, 353, 351, 1, 0, 0, 0, 353, 354, 1, 0, 0, 0, 354, 357, 1, 0, 0, 0, 355, 353, 1, 0, 0, 0, 356, 358, 5, 116, 0, 0, 357, 356, 1, 0, 0, 0, 357, 358, 1, 0, 0, 0, 358, 41, 1, 0, 0, 0, 359, 363, 3, 44, 22, 0, 360, 363, 3, 48, 24, 0, 361, 363, 3, 124, 62, 0, 362, 359, 1, 0, 0, 0, 362, 360, 1, 0, 0, 0, 362, 361, 1, 0, 0, 0, 363, 364, 1, 0, 0, 0, 364, 365, 5, 0, 0, 1, 365, 43, 1, 0, 0, 0, 366, 372, 3, 46, 23, 0, 367, 368, 5, 95, 0, 0, 368, 369, 5, 1, 0, 0, 369, 371, 3, 46, 23, 0, 370, 367, 1, 0, 0, 0, 371, 374, 1, 0, 0, 0, 372, 370, 1, 0, 0, 0, 372, 373, 1, 0, 0, 0, 373, 45, 1, 0, 0, 0, 374, 372, 1, 0, 0, 0, 375, 382, 3, 48, 24, 0, 376, 377, 5, 130, 0, 0, 377, 378, 3, 44, 22, 0, 378, 379, 5, 149, 0, 0, 379, 382, 1, 0, 0, 0, 380, 382, 3, 164, 82, 0, 381, 375, 1, 0, 0, 0, 381, 376, 1, 0, 0, 0, 381, 380, 1, 0, 0, 0, 382, 47, 1, 0, 0, 0, 383, 385, 3, 50, 25, 0, 384, 383, 1, 0, 0, 0, 384, 385, 1, 0, 0, 0, 385, 386, 1, 0, 0, 0, 386, 388, 5, 79, 0, 0, 387, 389, 5, 24, 0, 0, 388, 387, 1, 0, 0, 0, 388, 389, 1, 0, 0, 0, 389, 391, 1, 0, 0, 0, 390, 392, 3, 52, 26, 0, 391, 390, 1, 0, 0, 0, 391, 392, 1, 0, 0, 0, 392, 393, 1, 0, 0, 0, 393, 395, 3, 114, 57, 0, 394, 396, 3, 54, 27, 0, 395, 394, 1, 0, 0, 0, 395, 396, 1, 0, 0, 0, 396, 398, 1, 0, 0, 0, 397, 399, 3, 56, 28, 0, 398, 397, 1, 0, 0, 0, 398, 399, 1, 0, 0, 0, 399, 401, 1, 0, 0, 0, 400, 402, 3, 60, 30, 0, 401, 400, 1, 0, 0, 0, 401, 402, 1, 0, 0, 0, 402, 404, 1, 0, 0, 0, 403, 405, 3, 62, 31, 0, 404, 403, 1, 0, 0, 0, 404, 405, 1, 0, 0, 0, 405, 407, 1, 0, 0, 0, 406, 408, 3, 64, 32, 0, 407, 406, 1, 0, 0, 0, 407, 408, 1, 0, 0, 0, 408, 411, 1, 0, 0, 0, 409, 410, 5, 102, 0, 0, 410, 412, 7, 0, 0, 0, 411, 409, 1, 0, 0, 0, 411, 412, 1, 0, 0, 0, 412, 415, 1, 0, 0, 0, 413, 414, 5, 102, 0, 0, 414, 416, 5, 89, 0, 0, 415, 413, 1, 0, 0, 0, 415, 416, 1, 0, 0, 0, 416, 418, 1, 0, 0, 0, 417, 419, 3, 66, 33, 0, 418, 417, 1, 0, 0, 0, 418, 419, 1, 0, 0, 0, 419, 421, 1, 0, 0, 0, 420, 422, 3, 58, 29, 0, 421, 420, 1, 0, 0, 0, 421, 422, 1, 0, 0, 0, 422, 424, 1, 0, 0, 0, 423, 425, 3, 68, 34, 0, 424, 423, 1, 0, 0, 0, 424, 425, 1, 0, 0, 0, 425, 428, 1, 0, 0, 0, 426, 429, 3, 72, 36, 0, 427, 429, 3, 74, 37, 0, 428, 426, 1, 0, 0, 0, 428, 427, 1, 0, 0, 0, 428, 429, 1, 0, 0, 0, 429, 431, 1, 0, 0, 0, 430, 432, 3, 76, 38, 0, 431, 430, 1, 0, 0, 0, 431, 432, 1, 0, 0, 0, 432, 49, 1, 0, 0, 0, 433, 434, 5, 102, 0, 0, 434, 435, 3, 128, 64, 0, 435, 51, 1, 0, 0, 0, 436, 437, 5, 88, 0, 0, 437, 440, 5, 108, 0, 0, 438, 439, 5, 102, 0, 0, 439, 441, 5, 85, 0, 0, 440, 438, 1, 0, 0, 0, 440, 441, 1, 0, 0, 0, 441, 53, 1, 0, 0, 0, 442, 443, 5, 34, 0, 0, 443, 444, 3, 78, 39, 0, 444, 55, 1, 0, 0, 0, 445, 447, 7, 1, 0, 0, 446, 445, 1, 0, 0, 0, 446, 447, 1, 0, 0, 0, 447, 448, 1, 0, 0, 0, 448, 449, 5, 5, 0, 0, 449, 450, 5, 47, 0, 0, 450, 451, 3, 114, 57, 0, 451, 57, 1, 0, 0, 0, 452, 453, 5, 101, 0, 0, 453, 454, 3, 160, 80, 0, 454, 455, 5, 6, 0, 0, 455, 456, 5, 130, 0, 0, 456, 457, 3, 98, 49, 0, 457, 467, 5, 149, 0, 0, 458, 459, 5, 116, 0, 0, 459, 460, 3, 160, 80, 0, 460, 461, 5, 6, 0, 0, 461, 462, 5, 130, 0, 0, 462, 463, 3, 98, 49, 0, 463, 464, 5, 149, 0, 0, 464, 466, 1, 0, 0, 0, 465, 458, 1, 0, 0, 0, 466, 469, 1, 0, 0, 0, 467, 465, 1, 0, 0, 0, 467, 468, 1, 0, 0, 0, 468, 59, 1, 0, 0, 0, 469, 467, 1, 0, 0, 0, 470, 471, 5, 69, 0, 0, 471, 472, 3, 116, 58, 0, 472, 61, 1, 0, 0, 0, 473, 474, 5, 99, 0, 0, 474, 475, 3, 116, 58, 0, 475, 63, 1, 0, 0, 0, 476, 477, 5, 36, 0, 0, 477, 484, 5, 11, 0, 0, 478, 479, 7, 0, 0, 0, 479, 480, 5, 130, 0, 0, 480, 481, 3, 114, 57, 0, 481, 482, 5, 149, 0, 0, 482, 485, 1, 0, 0, 0, 483, 485, 3, 114, 57, 0, 484, 478, 1, 0, 0, 0, 484, 483, 1, 0, 0, 0, 485, 65, 1, 0, 0, 0, 486, 487, 5, 37, 0, 0, 487, 488, 3, 116, 58, 0, 488, 67, 1, 0, 0, 0, 489, 490, 5, 64, 0, 0, 490, 491, 5, 11, 0, 0, 491, 492, 3, 88, 44, 0, 492, 69, 1, 0, 0, 0, 493, 494, 5, 64, 0, 0, 494, 495, 5, 11, 0, 0, 495, 496, 3, 114, 57, 0, 496, 71, 1, 0, 0, 0, 497, 498, 5, 54, 0, 0, 498, 501, 3, 116, 58, 0, 499, 500, 5, 116, 0, 0, 500, 502, 3, 116, 58, 0, 501, 499, 1, 0, 0, 0, 501, 502, 1, 0, 0, 0, 502, 507, 1, 0, 0, 0, 503, 504, 5, 102, 0, 0, 504, 508, 5, 85, 0, 0, 505, 506, 5, 11, 0, 0, 506, 508, 3, 114, 57, 0, 507, 503, 1, 0, 0, 0, 507, 505, 1, 0, 0, 0, 507, 508, 1, 0, 0, 0, 508, 527, 1, 0, 0, 0, 509, 510, 5, 54, 0, 0, 510, 513, 3, 116, 58, 0, 511, 512, 5, 102, 0, 0, 512, 514, 5, 85, 0, 0, 513, 511, 1, 0, 0, 0, 513, 514, 1, 0, 0, 0, 514, 515, 1, 0, 0, 0, 515, 516, 5, 61, 0, 0, 516, 517, 3, 116, 58, 0, 517, 527, 1, 0, 0, 0, 518, 519, 5, 54, 0, 0, 519, 520, 3, 116, 58, 0, 520, 521, 5, 61, 0, 0, 521, 524, 3, 116, 58, 0, 522, 523, 5, 11, 0, 0, 523, 525, 3, 114, 57, 0, 524, 522, 1, 0, 0, 0, 524, 525, 1, 0, 0, 0, 525, 527, 1, 0, 0, 0, 526, 497, 1, 0, 0, 0, 526, 509, 1, 0, 0, 0, 526, 518, 1, 0, 0, 0, 527, 73, 1, 0, 0, 0, 528, 529, 5, 61, 0, 0, 529, 530, 3, 116, 58, 0, 530, 75, 1, 0, 0, 0, 531, 532, 5, 81, 0, 0, 532, 533, 3, 94, 47, 0, 533, 77, 1, 0, 0, 0, 534, 535, 6, 39, -1, 0, 535, 537, 3, 136, 68, 0, 536, 538, 5, 28, 0, 0, 537, 536, 1, 0, 0, 0, 537, 538, 1, 0, 0, 0, 538, 540, 1, 0, 0, 0, 539, 541, 3, 86, 43, 0, 540, 539, 1, 0, 0, 0, 540, 541, 1, 0, 0, 0, 541, 547, 1, 0, 0, 0, 542, 543, 5, 130, 0, 0, 543, 544, 3, 78, 39, 0, 544, 545, 5, 149, 0, 0, 545, 547, 1, 0, 0, 0, 546, 534, 1, 0, 0, 0, 546, 542, 1, 0, 0, 0, 547, 562, 1, 0, 0, 0, 548, 549, 10, 3, 0, 0, 549, 550, 3, 82, 41, 0, 550, 551, 3, 78, 39, 4, 551, 561, 1, 0, 0, 0, 552, 554, 10, 4, 0, 0, 553, 555, 3, 80, 40, 0, 554, 553, 1, 0, 0, 0, 554, 555, 1, 0, 0, 0, 555, 556, 1, 0, 0, 0, 556, 557, 5, 47, 0, 0, 557, 558, 3, 78, 39, 0, 558, 559, 3, 84, 42, 0, 559, 561, 1, 0, 0, 0, 560, 548, 1, 0, 0, 0, 560, 552, 1, 0, 0, 0, 561, 564, 1, 0, 0, 0, 562, 560, 1, 0, 0, 0, 562, 563, 1, 0, 0, 0, 563, 79, 1, 0, 0, 0, 564, 562, 1, 0, 0, 0, 565, 567, 7, 2, 0, 0, 566, 565, 1, 0, 0, 0, 566, 567, 1, 0, 0, 0, 567, 568, 1, 0, 0, 0, 568, 575, 5, 44, 0, 0, 569, 571, 5, 44, 0, 0, 570, 572, 7, 2, 0, 0, 571, 570, 1, 0, 0, 0, 571, 572, 1, 0, 0, 0, 572, 575, 1, 0, 0, 0, 573, 575, 7, 2, 0, 0, 574, 566, 1, 0, 0, 0, 574, 569, 1, 0, 0, 0, 574, 573, 1, 0, 0, 0, 575, 609, 1, 0, 0, 0, 576, 578, 7, 3, 0, 0, 577, 576, 1, 0, 0, 0, 577, 578, 1, 0, 0, 0, 578, 579, 1, 0, 0, 0, 579, 581, 7, 4, 0, 0, 580, 582, 5, 65, 0, 0, 581, 580, 1, 0, 0, 0, 581, 582, 1, 0, 0, 0, 582, 591, 1, 0, 0, 0, 583, 585, 7, 4, 0, 0, 584, 586, 5, 65, 0, 0, 585, 584, 1, 0, 0, 0, 585, 586, 1, 0, 0, 0, 586, 588, 1, 0, 0, 0, 587, 589, 7, 3, 0, 0, 588, 587, 1, 0, 0, 0, 588, 589, 1, 0, 0, 0, 589, 591, 1, 0, 0, 0, 590, 577, 1, 0, 0, 0, 590, 583, 1, 0, 0, 0, 591, 609, 1, 0, 0, 0, 592, 594, 7, 5, 0, 0, 593, 592, 1, 0, 0, 0, 593, 594, 1, 0, 0, 0, 594, 595, 1, 0, 0, 0, 595, 597, 5, 35, 0, 0, 596, 598, 5, 65, 0, 0, 597, 596, 1, 0, 0, 0, 597, 598, 1, 0, 0, 0, 598, 607, 1, 0, 0, 0, 599, 601, 5, 35, 0, 0, 600, 602, 5, 65, 0, 0, 601, 600, 1, 0, 0, 0, 601, 602, 1, 0, 0, 0, 602, 604, 1, 0, 0, 0, 603, 605, 7, 5, 0, 0, 604, 603, 1, 0, 0, 0, 604, 605, 1, 0, 0, 0, 605, 607, 1, 0, 0, 0, 606, 593, 1, 0, 0, 0, 606, 599, 1, 0, 0, 0, 607, 609, 1, 0, 0, 0, 608, 574, 1, 0, 0, 0, 608, 590, 1, 0, 0, 0, 608, 606, 1, 0, 0, 0, 609, 81, 1, 0, 0, 0, 610, 611, 5, 17, 0, 0, 611, 614, 5, 47, 0, 0, 612, 614, 5, 116, 0, 0, 613, 610, 1, 0, 0, 0, 613, 612, 1, 0, 0, 0, 614, 83, 1, 0, 0, 0, 615, 616, 5, 62, 0, 0, 616, 625, 3, 114, 57, 0, 617, 618, 5, 96, 0, 0, 618, 619, 5, 130, 0, 0, 619, 620, 3, 114, 57, 0, 620, 621, 5, 149, 0, 0, 621, 625, 1, 0, 0, 0, 622, 623, 5, 96, 0, 0, 623, 625, 3, 114, 57, 0, 624, 615, 1, 0, 0, 0, 624, 617, 1, 0, 0, 0, 624, 622, 1, 0, 0, 0, 625, 85, 1, 0, 0, 0, 626, 627, 5, 77, 0, 0, 627, 630, 3, 92, 46, 0, 628, 629, 5, 61, 0, 0, 629, 631, 3, 92, 46, 0, 630, 628, 1, 0, 0, 0, 630, 631, 1, 0, 0, 0, 631, 87, 1, 0, 0, 0, 632, 637, 3, 90, 45, 0, 633, 634, 5, 116, 0, 0, 634, 636, 3, 90, 45, 0, 635, 633, 1, 0, 0, 0, 636, 639, 1, 0, 0, 0, 637, 635, 1, 0, 0, 0, 637, 638, 1, 0, 0, 0, 638, 89, 1, 0, 0, 0, 639, 637, 1, 0, 0, 0, 640, 642, 3, 116, 58, 0, 641, 643, 7, 6, 0, 0, 642, 641, 1, 0, 0, 0, 642, 643, 1, 0, 0, 0, 643, 646, 1, 0, 0, 0, 644, 645, 5, 60, 0, 0, 645, 647, 7, 7, 0, 0, 646, 644, 1, 0, 0, 0, 646, 647, 1, 0, 0, 0, 647, 650, 1, 0, 0, 0, 648, 649, 5, 16, 0, 0, 649, 651, 5, 110, 0, 0, 650, 648, 1, 0, 0, 0, 650, 651, 1, 0, 0, 0, 651, 91, 1, 0, 0, 0, 652, 659, 3, 164, 82, 0, 653, 656, 3, 148, 74, 0, 654, 655, 5, 151, 0, 0, 655, 657, 3, 148, 74, 0, 656, 654, 1, 0, 0, 0, 656, 657, 1, 0, 0, 0, 657, 659, 1, 0, 0, 0, 658, 652, 1, 0, 0, 0, 658, 653, 1, 0, 0, 0, 659, 93, 1, 0, 0, 0, 660, 665, 3, 96, 48, 0, 661, 662, 5, 116, 0, 0, 662, 664, 3, 96, 48, 0, 663, 661, 1, 0, 0, 0, 664, 667, 1, 0, 0, 0, 665, 663, 1, 0, 0, 0, 665, 666, 1, 0, 0, 0, 666, 95, 1, 0, 0, 0, 667, 665, 1, 0, 0, 0, 668, 669, 3, 160, 80, 0, 669, 670, 5, 122, 0, 0, 670, 671, 3, 150, 75, 0, 671, 97, 1, 0, 0, 0, 672, 674, 3, 100, 50, 0, 673, 672, 1, 0, 0, 0, 673, 674, 1, 0, 0, 0, 674, 676, 1, 0, 0, 0, 675, 677, 3, 102, 51, 0, 676, 675, 1, 0, 0, 0, 676, 677, 1, 0, 0, 0, 677, 679, 1, 0, 0, 0, 678, 680, 3, 104, 52, 0, 679, 678, 1, 0, 0, 0, 679, 680, 1, 0, 0, 0, 680, 99, 1, 0, 0, 0, 681, 682, 5, 67, 0, 0, 682, 683, 5, 11, 0, 0, 683, 684, 3, 114, 57, 0, 684, 101, 1, 0, 0, 0, 685, 686, 5, 64, 0, 0, 686, 687, 5, 11, 0, 0, 687, 688, 3, 88, 44, 0, 688, 103, 1, 0, 0, 0, 689, 690, 7, 8, 0, 0, 690, 691, 3, 106, 53, 0, 691, 105, 1, 0, 0, 0, 692, 699, 3, 108, 54, 0, 693, 694, 5, 9, 0, 0, 694, 695, 3, 108, 54, 0, 695, 696, 5, 2, 0, 0, 696, 697, 3, 108, 54, 0, 697, 699, 1, 0, 0, 0, 698, 692, 1, 0, 0, 0, 698, 693, 1, 0, 0, 0, 699, 107, 1, 0, 0, 0, 700, 701, 5, 19, 0, 0, 701, 713, 5, 75, 0, 0, 702, 703, 5, 94, 0, 0, 703, 713, 5, 68, 0, 0, 704, 705, 5, 94, 0, 0, 705, 713, 5, 32, 0, 0, 706, 707, 3, 148, 74, 0, 707, 708, 5, 68, 0, 0, 708, 713, 1, 0, 0, 0, 709, 710, 3, 148, 74, 0, 710, 711, 5, 32, 0, 0, 711, 713, 1, 0, 0, 0, 712, 700, 1, 0, 0, 0, 712, 702, 1, 0, 0, 0, 712, 704, 1, 0, 0, 0, 712, 706, 1, 0, 0, 0, 712, 709, 1, 0, 0, 0, 713, 109, 1, 0, 0, 0, 714, 715, 3, 116, 58, 0, 715, 716, 5, 0, 0, 1, 716, 111, 1, 0, 0, 0, 717, 774, 3, 160, 80, 0, 718, 719, 3, 160, 80, 0, 719, 720, 5, 130, 0, 0, 720, 721, 3, 160, 80, 0, 721, 728, 3, 112, 56, 0, 722, 723, 5, 116, 0, 0, 723, 724, 3, 160, 80, 0, 724, 725, 3, 112, 56, 0, 725, 727, 1, 0, 0, 0, 726, 722, 1, 0, 0, 0, 727, 730, 1, 0, 0, 0, 728, 726, 1, 0, 0, 0, 728, 729, 1, 0, 0, 0, 729, 732, 1, 0, 0, 0, 730, 728, 1, 0, 0, 0, 731, 733, 5, 116, 0, 0, 732, 731, 1, 0, 0, 0, 732, 733, 1, 0, 0, 0, 733, 734, 1, 0, 0, 0, 734, 735, 5, 149, 0, 0, 735, 774, 1, 0, 0, 0, 736, 737, 3, 160, 80, 0, 737, 738, 5, 130, 0, 0, 738, 743, 3, 162, 81, 0, 739, 740, 5, 116, 0, 0, 740, 742, 3, 162, 81, 0, 741, 739, 1, 0, 0, 0, 742, 745, 1, 0, 0, 0, 743, 741, 1, 0, 0, 0, 743, 744, 1, 0, 0, 0, 744, 747, 1, 0, 0, 0, 745, 743, 1, 0, 0, 0, 746, 748, 5, 116, 0, 0, 747, 746, 1, 0, 0, 0, 747, 748, 1, 0, 0, 0, 748, 749, 1, 0, 0, 0, 749, 750, 5, 149, 0, 0, 750, 774, 1, 0, 0, 0, 751, 752, 3, 160, 80, 0, 752, 753, 5, 130, 0, 0, 753, 758, 3, 112, 56, 0, 754, 755, 5, 116, 0, 0, 755, 757, 3, 112, 56, 0, 756, 754, 1, 0, 0, 0, 757, 760, 1, 0, 0, 0, 758, 756, 1, 0, 0, 0, 758, 759, 1, 0, 0, 0, 759, 762, 1, 0, 0, 0, 760, 758, 1, 0, 0, 0, 761, 763, 5, 116, 0, 0, 762, 761, 1, 0, 0, 0, 762, 763, 1, 0, 0, 0, 763, 764, 1, 0, 0, 0, 764, 765, 5, 149, 0, 0, 765, 774, 1, 0, 0, 0, 766, 767, 3, 160, 80, 0, 767, 769, 5, 130, 0, 0, 768, 770, 3, 114, 57, 0, 769, 768, 1, 0, 0, 0, 769, 770, 1, 0, 0, 0, 770, 771, 1, 0, 0, 0, 771, 772, 5, 149, 0, 0, 772, 774, 1, 0, 0, 0, 773, 717, 1, 0, 0, 0, 773, 718, 1, 0, 0, 0, 773, 736, 1, 0, 0, 0, 773, 751, 1, 0, 0, 0, 773, 766, 1, 0, 0, 0, 774, 113, 1, 0, 0, 0, 775, 780, 3, 116, 58, 0, 776, 777, 5, 116, 0, 0, 777, 779, 3, 116, 58, 0, 778, 776, 1, 0, 0, 0, 779, 782, 1, 0, 0, 0, 780, 778, 1, 0, 0, 0, 780, 781, 1, 0, 0, 0, 781, 784, 1, 0, 0, 0, 782, 780, 1, 0, 0, 0, 783, 785, 5, 116, 0, 0, 784, 783, 1, 0, 0, 0, 784, 785, 1, 0, 0, 0, 785, 115, 1, 0, 0, 0, 786, 787, 6, 58, -1, 0, 787, 789, 5, 12, 0, 0, 788, 790, 3, 116, 58, 0, 789, 788, 1, 0, 0, 0, 789, 790, 1, 0, 0, 0, 790, 796, 1, 0, 0, 0, 791, 792, 5, 98, 0, 0, 792, 793, 3, 116, 58, 0, 793, 794, 5, 83, 0, 0, 794, 795, 3, 116, 58, 0, 795, 797, 1, 0, 0, 0, 796, 791, 1, 0, 0, 0, 797, 798, 1, 0, 0, 0, 798, 796, 1, 0, 0, 0, 798, 799, 1, 0, 0, 0, 799, 802, 1, 0, 0, 0, 800, 801, 5, 25, 0, 0, 801, 803, 3, 116, 58, 0, 802, 800, 1, 0, 0, 0, 802, 803, 1, 0, 0, 0, 803, 804, 1, 0, 0, 0, 804, 805, 5, 26, 0, 0, 805, 936, 1, 0, 0, 0, 806, 807, 5, 13, 0, 0, 807, 808, 5, 130, 0, 0, 808, 809, 3, 116, 58, 0, 809, 810, 5, 6, 0, 0, 810, 811, 3, 112, 56, 0, 811, 812, 5, 149, 0, 0, 812, 936, 1, 0, 0, 0, 813, 814, 5, 20, 0, 0, 814, 936, 5, 110, 0, 0, 815, 816, 5, 45, 0, 0, 816, 817, 3, 116, 58, 0, 817, 818, 3, 152, 76, 0, 818, 936, 1, 0, 0, 0, 819, 820, 5, 82, 0, 0, 820, 821, 5, 130, 0, 0, 821, 822, 3, 116, 58, 0, 822, 823, 5, 34, 0, 0, 823, 826, 3, 116, 58, 0, 824, 825, 5, 33, 0, 0, 825, 827, 3, 116, 58, 0, 826, 824, 1, 0, 0, 0, 826, 827, 1, 0, 0, 0, 827, 828, 1, 0, 0, 0, 828, 829, 5, 149, 0, 0, 829, 936, 1, 0, 0, 0, 830, 831, 5, 86, 0, 0, 831, 936, 5, 110, 0, 0, 832, 833, 5, 91, 0, 0, 833, 834, 5, 130, 0, 0, 834, 835, 7, 9, 0, 0, 835, 836, 3, 166, 83, 0, 836, 837, 5, 34, 0, 0, 837, 838, 3, 116, 58, 0, 838, 839, 5, 149, 0, 0, 839, 936, 1, 0, 0, 0, 840, 841, 3, 160, 80, 0, 841, 843, 5, 130, 0, 0, 842, 844, 3, 114, 57, 0, 843, 842, 1, 0, 0, 0, 843, 844, 1, 0, 0, 0, 844, 845, 1, 0, 0, 0, 845, 846, 5, 149, 0, 0, 846, 855, 1, 0, 0, 0, 847, 849, 5, 130, 0, 0, 848, 850, 5, 24, 0, 0, 849, 848, 1, 0, 0, 0, 849, 850, 1, 0, 0, 0, 850, 852, 1, 0, 0, 0, 851, 853, 3, 118, 59, 0, 852, 851, 1, 0, 0, 0, 852, 853, 1, 0, 0, 0, 853, 854, 1, 0, 0, 0, 854, 856, 5, 149, 0, 0, 855, 847, 1, 0, 0, 0, 855, 856, 1, 0, 0, 0, 856, 857, 1, 0, 0, 0, 857, 858, 5, 66, 0, 0, 858, 859, 5, 130, 0, 0, 859, 860, 3, 98, 49, 0, 860, 861, 5, 149, 0, 0, 861, 936, 1, 0, 0, 0, 862, 863, 3, 160, 80, 0, 863, 865, 5, 130, 0, 0, 864, 866, 3, 114, 57, 0, 865, 864, 1, 0, 0, 0, 865, 866, 1, 0, 0, 0, 866, 867, 1, 0, 0, 0, 867, 868, 5, 149, 0, 0, 868, 877, 1, 0, 0, 0, 869, 871, 5, 130, 0, 0, 870, 872, 5, 24, 0, 0, 871, 870, 1, 0, 0, 0, 871, 872, 1, 0, 0, 0, 872, 874, 1, 0, 0, 0, 873, 875, 3, 118, 59, 0, 874, 873, 1, 0, 0, 0, 874, 875, 1, 0, 0, 0, 875, 876, 1, 0, 0, 0, 876, 878, 5, 149, 0, 0, 877, 869, 1, 0, 0, 0, 877, 878, 1, 0, 0, 0, 878, 879, 1, 0, 0, 0, 879, 880, 5, 66, 0, 0, 880, 881, 3, 160, 80, 0, 881, 936, 1, 0, 0, 0, 882, 888, 3, 160, 80, 0, 883, 885, 5, 130, 0, 0, 884, 886, 3, 114, 57, 0, 885, 884, 1, 0, 0, 0, 885, 886, 1, 0, 0, 0, 886, 887, 1, 0, 0, 0, 887, 889, 5, 149, 0, 0, 888, 883, 1, 0, 0, 0, 888, 889, 1, 0, 0, 0, 889, 890, 1, 0, 0, 0, 890, 892, 5, 130, 0, 0, 891, 893, 5, 24, 0, 0, 892, 891, 1, 0, 0, 0, 892, 893, 1, 0, 0, 0, 893, 895, 1, 0, 0, 0, 894, 896, 3, 118, 59, 0, 895, 894, 1, 0, 0, 0, 895, 896, 1, 0, 0, 0, 896, 897, 1, 0, 0, 0, 897, 898, 5, 149, 0, 0, 898, 936, 1, 0, 0, 0, 899, 936, 3, 124, 62, 0, 900, 936, 3, 168, 84, 0, 901, 936, 3, 150, 75, 0, 902, 903, 5, 118, 0, 0, 903, 936, 3, 116, 58, 19, 904, 905, 5, 58, 0, 0, 905, 936, 3, 116, 58, 13, 906, 907, 3, 140, 70, 0, 907, 908, 5, 120, 0, 0, 908, 910, 1, 0, 0, 0, 909, 906, 1, 0, 0, 0, 909, 910, 1, 0, 0, 0, 910, 911, 1, 0, 0, 0, 911, 936, 5, 112, 0, 0, 912, 913, 5, 130, 0, 0, 913, 914, 3, 44, 22, 0, 914, 915, 5, 149, 0, 0, 915, 936, 1, 0, 0, 0, 916, 917, 5, 130, 0, 0, 917, 918, 3, 116, 58, 0, 918, 919, 5, 149, 0, 0, 919, 936, 1, 0, 0, 0, 920, 921, 5, 130, 0, 0, 921, 922, 3, 114, 57, 0, 922, 923, 5, 149, 0, 0, 923, 936, 1, 0, 0, 0, 924, 926, 5, 129, 0, 0, 925, 927, 3, 114, 57, 0, 926, 925, 1, 0, 0, 0, 926, 927, 1, 0, 0, 0, 927, 928, 1, 0, 0, 0, 928, 936, 5, 148, 0, 0, 929, 931, 5, 128, 0, 0, 930, 932, 3, 40, 20, 0, 931, 930, 1, 0, 0, 0, 931, 932, 1, 0, 0, 0, 932, 933, 1, 0, 0, 0, 933, 936, 5, 147, 0, 0, 934, 936, 3, 132, 66, 0, 935, 786, 1, 0, 0, 0, 935, 806, 1, 0, 0, 0, 935, 813, 1, 0, 0, 0, 935, 815, 1, 0, 0, 0, 935, 819, 1, 0, 0, 0, 935, 830, 1, 0, 0, 0, 935, 832, 1, 0, 0, 0, 935, 840, 1, 0, 0, 0, 935, 862, 1, 0, 0, 0, 935, 882, 1, 0, 0, 0, 935, 899, 1, 0, 0, 0, 935, 900, 1, 0, 0, 0, 935, 901, 1, 0, 0, 0, 935, 902, 1, 0, 0, 0, 935, 904, 1, 0, 0, 0, 935, 909, 1, 0, 0, 0, 935, 912, 1, 0, 0, 0, 935, 916, 1, 0, 0, 0, 935, 920, 1, 0, 0, 0, 935, 924, 1, 0, 0, 0, 935, 929, 1, 0, 0, 0, 935, 934, 1, 0, 0, 0, 936, 1041, 1, 0, 0, 0, 937, 941, 10, 18, 0, 0, 938, 942, 5, 112, 0, 0, 939, 942, 5, 151, 0, 0, 940, 942, 5, 138, 0, 0, 941, 938, 1, 0, 0, 0, 941, 939, 1, 0, 0, 0, 941, 940, 1, 0, 0, 0, 942, 943, 1, 0, 0, 0, 943, 1040, 3, 116, 58, 19, 944, 948, 10, 17, 0, 0, 945, 949, 5, 139, 0, 0, 946, 949, 5, 118, 0, 0, 947, 949, 5, 117, 0, 0, 948, 945, 1, 0, 0, 0, 948, 946, 1, 0, 0, 0, 948, 947, 1, 0, 0, 0, 949, 950, 1, 0, 0, 0, 950, 1040, 3, 116, 58, 18, 951, 976, 10, 16, 0, 0, 952, 977, 5, 121, 0, 0, 953, 977, 5, 122, 0, 0, 954, 977, 5, 133, 0, 0, 955, 977, 5, 131, 0, 0, 956, 977, 5, 132, 0, 0, 957, 977, 5, 123, 0, 0, 958, 977, 5, 124, 0, 0, 959, 961, 5, 58, 0, 0, 960, 959, 1, 0, 0, 0, 960, 961, 1, 0, 0, 0, 961, 962, 1, 0, 0, 0, 962, 964, 5, 42, 0, 0, 963, 965, 5, 15, 0, 0, 964, 963, 1, 0, 0, 0, 964, 965, 1, 0, 0, 0, 965, 977, 1, 0, 0, 0, 966, 968, 5, 58, 0, 0, 967, 966, 1, 0, 0, 0, 967, 968, 1, 0, 0, 0, 968, 969, 1, 0, 0, 0, 969, 977, 7, 10, 0, 0, 970, 977, 5, 145, 0, 0, 971, 977, 5, 146, 0, 0, 972, 977, 5, 135, 0, 0, 973, 977, 5, 126, 0, 0, 974, 977, 5, 127, 0, 0, 975, 977, 5, 134, 0, 0, 976, 952, 1, 0, 0, 0, 976, 953, 1, 0, 0, 0, 976, 954, 1, 0, 0, 0, 976, 955, 1, 0, 0, 0, 976, 956, 1, 0, 0, 0, 976, 957, 1, 0, 0, 0, 976, 958, 1, 0, 0, 0, 976, 960, 1, 0, 0, 0, 976, 967, 1, 0, 0, 0, 976, 970, 1, 0, 0, 0, 976, 971, 1, 0, 0, 0, 976, 972, 1, 0, 0, 0, 976, 973, 1, 0, 0, 0, 976, 974, 1, 0, 0, 0, 976, 975, 1, 0, 0, 0, 977, 978, 1, 0, 0, 0, 978, 1040, 3, 116, 58, 17, 979, 980, 10, 14, 0, 0, 980, 981, 5, 137, 0, 0, 981, 1040, 3, 116, 58, 15, 982, 983, 10, 12, 0, 0, 983, 984, 5, 2, 0, 0, 984, 1040, 3, 116, 58, 13, 985, 986, 10, 11, 0, 0, 986, 987, 5, 63, 0, 0, 987, 1040, 3, 116, 58, 12, 988, 990, 10, 10, 0, 0, 989, 991, 5, 58, 0, 0, 990, 989, 1, 0, 0, 0, 990, 991, 1, 0, 0, 0, 991, 992, 1, 0, 0, 0, 992, 993, 5, 9, 0, 0, 993, 994, 3, 116, 58, 0, 994, 995, 5, 2, 0, 0, 995, 996, 3, 116, 58, 11, 996, 1040, 1, 0, 0, 0, 997, 998, 10, 9, 0, 0, 998, 999, 5, 140, 0, 0, 999, 1000, 3, 116, 58, 0, 1000, 1001, 5, 115, 0, 0, 1001, 1002, 3, 116, 58, 9, 1002, 1040, 1, 0, 0, 0, 1003, 1004, 10, 25, 0, 0, 1004, 1005, 5, 129, 0, 0, 1005, 1006, 3, 116, 58, 0, 1006, 1007, 5, 148, 0, 0, 1007, 1040, 1, 0, 0, 0, 1008, 1009, 10, 24, 0, 0, 1009, 1010, 5, 120, 0, 0, 1010, 1040, 5, 108, 0, 0, 1011, 1012, 10, 23, 0, 0, 1012, 1013, 5, 120, 0, 0, 1013, 1040, 3, 160, 80, 0, 1014, 1015, 10, 22, 0, 0, 1015, 1016, 5, 136, 0, 0, 1016, 1017, 5, 129, 0, 0, 1017, 1018, 3, 116, 58, 0, 1018, 1019, 5, 148, 0, 0, 1019, 1040, 1, 0, 0, 0, 1020, 1021, 10, 21, 0, 0, 1021, 1022, 5, 136, 0, 0, 1022, 1040, 5, 108, 0, 0, 1023, 1024, 10, 20, 0, 0, 1024, 1025, 5, 136, 0, 0, 1025, 1040, 3, 160, 80, 0, 1026, 1027, 10, 15, 0, 0, 1027, 1029, 5, 46, 0, 0, 1028, 1030, 5, 58, 0, 0, 1029, 1028, 1, 0, 0, 0, 1029, 1030, 1, 0, 0, 0, 1030, 1031, 1, 0, 0, 0, 1031, 1040, 5, 59, 0, 0, 1032, 1037, 10, 8, 0, 0, 1033, 1034, 5, 6, 0, 0, 1034, 1038, 3, 160, 80, 0, 1035, 1036, 5, 6, 0, 0, 1036, 1038, 5, 110, 0, 0, 1037, 1033, 1, 0, 0, 0, 1037, 1035, 1, 0, 0, 0, 1038, 1040, 1, 0, 0, 0, 1039, 937, 1, 0, 0, 0, 1039, 944, 1, 0, 0, 0, 1039, 951, 1, 0, 0, 0, 1039, 979, 1, 0, 0, 0, 1039, 982, 1, 0, 0, 0, 1039, 985, 1, 0, 0, 0, 1039, 988, 1, 0, 0, 0, 1039, 997, 1, 0, 0, 0, 1039, 1003, 1, 0, 0, 0, 1039, 1008, 1, 0, 0, 0, 1039, 1011, 1, 0, 0, 0, 1039, 1014, 1, 0, 0, 0, 1039, 1020, 1, 0, 0, 0, 1039, 1023, 1, 0, 0, 0, 1039, 1026, 1, 0, 0, 0, 1039, 1032, 1, 0, 0, 0, 1040, 1043, 1, 0, 0, 0, 1041, 1039, 1, 0, 0, 0, 1041, 1042, 1, 0, 0, 0, 1042, 117, 1, 0, 0, 0, 1043, 1041, 1, 0, 0, 0, 1044, 1049, 3, 120, 60, 0, 1045, 1046, 5, 116, 0, 0, 1046, 1048, 3, 120, 60, 0, 1047, 1045, 1, 0, 0, 0, 1048, 1051, 1, 0, 0, 0, 1049, 1047, 1, 0, 0, 0, 1049, 1050, 1, 0, 0, 0, 1050, 1053, 1, 0, 0, 0, 1051, 1049, 1, 0, 0, 0, 1052, 1054, 5, 116, 0, 0, 1053, 1052, 1, 0, 0, 0, 1053, 1054, 1, 0, 0, 0, 1054, 119, 1, 0, 0, 0, 1055, 1058, 3, 122, 61, 0, 1056, 1058, 3, 116, 58, 0, 1057, 1055, 1, 0, 0, 0, 1057, 1056, 1, 0, 0, 0, 1058, 121, 1, 0, 0, 0, 1059, 1060, 5, 130, 0, 0, 1060, 1065, 3, 160, 80, 0, 1061, 1062, 5, 116, 0, 0, 1062, 1064, 3, 160, 80, 0, 1063, 1061, 1, 0, 0, 0, 1064, 1067, 1, 0, 0, 0, 1065, 1063, 1, 0, 0, 0, 1065, 1066, 1, 0, 0, 0, 1066, 1069, 1, 0, 0, 0, 1067, 1065, 1, 0, 0, 0, 1068, 1070, 5, 116, 0, 0, 1069, 1068, 1, 0, 0, 0, 1069, 1070, 1, 0, 0, 0, 1070, 1071, 1, 0, 0, 0, 1071, 1072, 5, 149, 0, 0, 1072, 1085, 1, 0, 0, 0, 1073, 1078, 3, 160, 80, 0, 1074, 1075, 5, 116, 0, 0, 1075, 1077, 3, 160, 80, 0, 1076, 1074, 1, 0, 0, 0, 1077, 1080, 1, 0, 0, 0, 1078, 1076, 1, 0, 0, 0, 1078, 1079, 1, 0, 0, 0, 1079, 1082, 1, 0, 0, 0, 1080, 1078, 1, 0, 0, 0, 1081, 1083, 5, 116, 0, 0, 1082, 1081, 1, 0, 0, 0, 1082, 1083, 1, 0, 0, 0, 1083, 1085, 1, 0, 0, 0, 1084, 1059, 1, 0, 0, 0, 1084, 1073, 1, 0, 0, 0, 1085, 1086, 1, 0, 0, 0, 1086, 1087, 5, 111, 0, 0, 1087, 1088, 3, 116, 58, 0, 1088, 123, 1, 0, 0, 0, 1089, 1090, 5, 132, 0, 0, 1090, 1094, 3, 160, 80, 0, 1091, 1093, 3, 126, 63, 0, 1092, 1091, 1, 0, 0, 0, 1093, 1096, 1, 0, 0, 0, 1094, 1092, 1, 0, 0, 0, 1094, 1095, 1, 0, 0, 0, 1095, 1097, 1, 0, 0, 0, 1096, 1094, 1, 0, 0, 0, 1097, 1098, 5, 151, 0, 0, 1098, 1099, 5, 124, 0, 0, 1099, 1122, 1, 0, 0, 0, 1100, 1101, 5, 132, 0, 0, 1101, 1105, 3, 160, 80, 0, 1102, 1104, 3, 126, 63, 0, 1103, 1102, 1, 0, 0, 0, 1104, 1107, 1, 0, 0, 0, 1105, 1103, 1, 0, 0, 0, 1105, 1106, 1, 0, 0, 0, 1106, 1108, 1, 0, 0, 0, 1107, 1105, 1, 0, 0, 0, 1108, 1114, 5, 124, 0, 0, 1109, 1115, 3, 124, 62, 0, 1110, 1111, 5, 128, 0, 0, 1111, 1112, 3, 116, 58, 0, 1112, 1113, 5, 147, 0, 0, 1113, 1115, 1, 0, 0, 0, 1114, 1109, 1, 0, 0, 0, 1114, 1110, 1, 0, 0, 0, 1114, 1115, 1, 0, 0, 0, 1115, 1116, 1, 0, 0, 0, 1116, 1117, 5, 132, 0, 0, 1117, 1118, 5, 151, 0, 0, 1118, 1119, 3, 160, 80, 0, 1119, 1120, 5, 124, 0, 0, 1120, 1122, 1, 0, 0, 0, 1121, 1089, 1, 0, 0, 0, 1121, 1100, 1, 0, 0, 0, 1122, 125, 1, 0, 0, 0, 1123, 1124, 3, 160, 80, 0, 1124, 1125, 5, 122, 0, 0, 1125, 1126, 3, 166, 83, 0, 1126, 1135, 1, 0, 0, 0, 1127, 1128, 3, 160, 80, 0, 1128, 1129, 5, 122, 0, 0, 1129, 1130, 5, 128, 0, 0, 1130, 1131, 3, 116, 58, 0, 1131, 1132, 5, 147, 0, 0, 1132, 1135, 1, 0, 0, 0, 1133, 1135, 3, 160, 80, 0, 1134, 1123, 1, 0, 0, 0, 1134, 1127, 1, 0, 0, 0, 1134, 1133, 1, 0, 0, 0, 1135, 127, 1, 0, 0, 0, 1136, 1141, 3, 130, 65, 0, 1137, 1138, 5, 116, 0, 0, 1138, 1140, 3, 130, 65, 0, 1139, 1137, 1, 0, 0, 0, 1140, 1143, 1, 0, 0, 0, 1141, 1139, 1, 0, 0, 0, 1141, 1142, 1, 0, 0, 0, 1142, 1145, 1, 0, 0, 0, 1143, 1141, 1, 0, 0, 0, 1144, 1146, 5, 116, 0, 0, 1145, 1144, 1, 0, 0, 0, 1145, 1146, 1, 0, 0, 0, 1146, 129, 1, 0, 0, 0, 1147, 1148, 3, 160, 80, 0, 1148, 1149, 5, 6, 0, 0, 1149, 1150, 5, 130, 0, 0, 1150, 1151, 3, 44, 22, 0, 1151, 1152, 5, 149, 0, 0, 1152, 1158, 1, 0, 0, 0, 1153, 1154, 3, 116, 58, 0, 1154, 1155, 5, 6, 0, 0, 1155, 1156, 3, 160, 80, 0, 1156, 1158, 1, 0, 0, 0, 1157, 1147, 1, 0, 0, 0, 1157, 1153, 1, 0, 0, 0, 1158, 131, 1, 0, 0, 0, 1159, 1167, 3, 164, 82, 0, 1160, 1161, 3, 140, 70, 0, 1161, 1162, 5, 120, 0, 0, 1162, 1164, 1, 0, 0, 0, 1163, 1160, 1, 0, 0, 0, 1163, 1164, 1, 0, 0, 0, 1164, 1165, 1, 0, 0, 0, 1165, 1167, 3, 134, 67, 0, 1166, 1159, 1, 0, 0, 0, 1166, 1163, 1, 0, 0, 0, 1167, 133, 1, 0, 0, 0, 1168, 1173, 3, 160, 80, 0, 1169, 1170, 5, 120, 0, 0, 1170, 1172, 3, 160, 80, 0, 1171, 1169, 1, 0, 0, 0, 1172, 1175, 1, 0, 0, 0, 1173, 1171, 1, 0, 0, 0, 1173, 1174, 1, 0, 0, 0, 1174, 135, 1, 0, 0, 0, 1175, 1173, 1, 0, 0, 0, 1176, 1177, 6, 68, -1, 0, 1177, 1186, 3, 140, 70, 0, 1178, 1186, 3, 138, 69, 0, 1179, 1180, 5, 130, 0, 0, 1180, 1181, 3, 44, 22, 0, 1181, 1182, 5, 149, 0, 0, 1182, 1186, 1, 0, 0, 0, 1183, 1186, 3, 124, 62, 0, 1184, 1186, 3, 164, 82, 0, 1185, 1176, 1, 0, 0, 0, 1185, 1178, 1, 0, 0, 0, 1185, 1179, 1, 0, 0, 0, 1185, 1183, 1, 0, 0, 0, 1185, 1184, 1, 0, 0, 0, 1186, 1195, 1, 0, 0, 0, 1187, 1191, 10, 3, 0, 0, 1188, 1192, 3, 158, 79, 0, 1189, 1190, 5, 6, 0, 0, 1190, 1192, 3, 160, 80, 0, 1191, 1188, 1, 0, 0, 0, 1191, 1189, 1, 0, 0, 0, 1192, 1194, 1, 0, 0, 0, 1193, 1187, 1, 0, 0, 0, 1194, 1197, 1, 0, 0, 0, 1195, 1193, 1, 0, 0, 0, 1195, 1196, 1, 0, 0, 0, 1196, 137, 1, 0, 0, 0, 1197, 1195, 1, 0, 0, 0, 1198, 1199, 3, 160, 80, 0, 1199, 1201, 5, 130, 0, 0, 1200, 1202, 3, 142, 71, 0, 1201, 1200, 1, 0, 0, 0, 1201, 1202, 1, 0, 0, 0, 1202, 1203, 1, 0, 0, 0, 1203, 1204, 5, 149, 0, 0, 1204, 139, 1, 0, 0, 0, 1205, 1206, 3, 144, 72, 0, 1206, 1207, 5, 120, 0, 0, 1207, 1209, 1, 0, 0, 0, 1208, 1205, 1, 0, 0, 0, 1208, 1209, 1, 0, 0, 0, 1209, 1210, 1, 0, 0, 0, 1210, 1211, 3, 160, 80, 0, 1211, 141, 1, 0, 0, 0, 1212, 1217, 3, 116, 58, 0, 1213, 1214, 5, 116, 0, 0, 1214, 1216, 3, 116, 58, 0, 1215, 1213, 1, 0, 0, 0, 1216, 1219, 1, 0, 0, 0, 1217, 1215, 1, 0, 0, 0, 1217, 1218, 1, 0, 0, 0, 1218, 1221, 1, 0, 0, 0, 1219, 1217, 1, 0, 0, 0, 1220, 1222, 5, 116, 0, 0, 1221, 1220, 1, 0, 0, 0, 1221, 1222, 1, 0, 0, 0, 1222, 143, 1, 0, 0, 0, 1223, 1224, 3, 160, 80, 0, 1224, 145, 1, 0, 0, 0, 1225, 1234, 5, 106, 0, 0, 1226, 1227, 5, 120, 0, 0, 1227, 1234, 7, 11, 0, 0, 1228, 1229, 5, 108, 0, 0, 1229, 1231, 5, 120, 0, 0, 1230, 1232, 7, 11, 0, 0, 1231, 1230, 1, 0, 0, 0, 1231, 1232, 1, 0, 0, 0, 1232, 1234, 1, 0, 0, 0, 1233, 1225, 1, 0, 0, 0, 1233, 1226, 1, 0, 0, 0, 1233, 1228, 1, 0, 0, 0, 1234, 147, 1, 0, 0, 0, 1235, 1237, 7, 12, 0, 0, 1236, 1235, 1, 0, 0, 0, 1236, 1237, 1, 0, 0, 0, 1237, 1244, 1, 0, 0, 0, 1238, 1245, 3, 146, 73, 0, 1239, 1245, 5, 107, 0, 0, 1240, 1245, 5, 108, 0, 0, 1241, 1245, 5, 109, 0, 0, 1242, 1245, 5, 43, 0, 0, 1243, 1245, 5, 57, 0, 0, 1244, 1238, 1, 0, 0, 0, 1244, 1239, 1, 0, 0, 0, 1244, 1240, 1, 0, 0, 0, 1244, 1241, 1, 0, 0, 0, 1244, 1242, 1, 0, 0, 0, 1244, 1243, 1, 0, 0, 0, 1245, 149, 1, 0, 0, 0, 1246, 1250, 3, 148, 74, 0, 1247, 1250, 5, 110, 0, 0, 1248, 1250, 5, 59, 0, 0, 1249, 1246, 1, 0, 0, 0, 1249, 1247, 1, 0, 0, 0, 1249, 1248, 1, 0, 0, 0, 1250, 151, 1, 0, 0, 0, 1251, 1252, 7, 13, 0, 0, 1252, 153, 1, 0, 0, 0, 1253, 1254, 7, 14, 0, 0, 1254, 155, 1, 0, 0, 0, 1255, 1256, 7, 15, 0, 0, 1256, 157, 1, 0, 0, 0, 1257, 1260, 5, 105, 0, 0, 1258, 1260, 3, 156, 78, 0, 1259, 1257, 1, 0, 0, 0, 1259, 1258, 1, 0, 0, 0, 1260, 159, 1, 0, 0, 0, 1261, 1265, 5, 105, 0, 0, 1262, 1265, 3, 152, 76, 0, 1263, 1265, 3, 154, 77, 0, 1264, 1261, 1, 0, 0, 0, 1264, 1262, 1, 0, 0, 0, 1264, 1263, 1, 0, 0, 0, 1265, 161, 1, 0, 0, 0, 1266, 1267, 3, 166, 83, 0, 1267, 1268, 5, 122, 0, 0, 1268, 1269, 3, 148, 74, 0, 1269, 163, 1, 0, 0, 0, 1270, 1271, 5, 128, 0, 0, 1271, 1272, 3, 134, 67, 0, 1272, 1273, 5, 147, 0, 0, 1273, 165, 1, 0, 0, 0, 1274, 1277, 5, 110, 0, 0, 1275, 1277, 3, 168, 84, 0, 1276, 1274, 1, 0, 0, 0, 1276, 1275, 1, 0, 0, 0, 1277, 167, 1, 0, 0, 0, 1278, 1282, 5, 142, 0, 0, 1279, 1281, 3, 170, 85, 0, 1280, 1279, 1, 0, 0, 0, 1281, 1284, 1, 0, 0, 0, 1282, 1280, 1, 0, 0, 0, 1282, 1283, 1, 0, 0, 0, 1283, 1285, 1, 0, 0, 0, 1284, 1282, 1, 0, 0, 0, 1285, 1286, 5, 144, 0, 0, 1286, 169, 1, 0, 0, 0, 1287, 1288, 5, 157, 0, 0, 1288, 1289, 3, 116, 58, 0, 1289, 1290, 5, 147, 0, 0, 1290, 1293, 1, 0, 0, 0, 1291, 1293, 5, 156, 0, 0, 1292, 1287, 1, 0, 0, 0, 1292, 1291, 1, 0, 0, 0, 1293, 171, 1, 0, 0, 0, 1294, 1298, 5, 143, 0, 0, 1295, 1297, 3, 174, 87, 0, 1296, 1295, 1, 0, 0, 0, 1297, 1300, 1, 0, 0, 0, 1298, 1296, 1, 0, 0, 0, 1298, 1299, 1, 0, 0, 0, 1299, 1301, 1, 0, 0, 0, 1300, 1298, 1, 0, 0, 0, 1301, 1302, 5, 0, 0, 1, 1302, 173, 1, 0, 0, 0, 1303, 1304, 5, 159, 0, 0, 1304, 1305, 3, 116, 58, 0, 1305, 1306, 5, 147, 0, 0, 1306, 1309, 1, 0, 0, 0, 1307, 1309, 5, 158, 0, 0, 1308, 1303, 1, 0, 0, 0, 1308, 1307, 1, 0, 0, 0, 1309, 175, 1, 0, 0, 0, 168, 179, 186, 195, 202, 206, 220, 224, 227, 231, 234, 241, 245, 254, 259, 268, 276, 283, 287, 293, 298, 306, 313, 319, 331, 339, 353, 357, 362, 372, 381, 384, 388, 391, 395, 398, 401, 404, 407, 411, 415, 418, 421, 424, 428, 431, 440, 446, 467, 484, 501, 507, 513, 524, 526, 537, 540, 546, 554, 560, 562, 566, 571, 574, 577, 581, 585, 588, 590, 593, 597, 601, 604, 606, 608, 613, 624, 630, 637, 642, 646, 650, 656, 658, 665, 673, 676, 679, 698, 712, 728, 732, 743, 747, 758, 762, 769, 773, 780, 784, 789, 798, 802, 826, 843, 849, 852, 855, 865, 871, 874, 877, 885, 888, 892, 895, 909, 926, 931, 935, 941, 948, 960, 964, 967, 976, 990, 1029, 1037, 1039, 1041, 1049, 1053, 1057, 1065, 1069, 1078, 1082, 1084, 1094, 1105, 1114, 1121, 1134, 1141, 1145, 1157, 1163, 1166, 1173, 1185, 1191, 1195, 1201, 1208, 1217, 1221, 1231, 1233, 1236, 1244, 1249, 1259, 1264, 1276, 1282, 1292, 1298, 1308] \ No newline at end of file +[4, 1, 159, 1303, 2, 0, 7, 0, 2, 1, 7, 1, 2, 2, 7, 2, 2, 3, 7, 3, 2, 4, 7, 4, 2, 5, 7, 5, 2, 6, 7, 6, 2, 7, 7, 7, 2, 8, 7, 8, 2, 9, 7, 9, 2, 10, 7, 10, 2, 11, 7, 11, 2, 12, 7, 12, 2, 13, 7, 13, 2, 14, 7, 14, 2, 15, 7, 15, 2, 16, 7, 16, 2, 17, 7, 17, 2, 18, 7, 18, 2, 19, 7, 19, 2, 20, 7, 20, 2, 21, 7, 21, 2, 22, 7, 22, 2, 23, 7, 23, 2, 24, 7, 24, 2, 25, 7, 25, 2, 26, 7, 26, 2, 27, 7, 27, 2, 28, 7, 28, 2, 29, 7, 29, 2, 30, 7, 30, 2, 31, 7, 31, 2, 32, 7, 32, 2, 33, 7, 33, 2, 34, 7, 34, 2, 35, 7, 35, 2, 36, 7, 36, 2, 37, 7, 37, 2, 38, 7, 38, 2, 39, 7, 39, 2, 40, 7, 40, 2, 41, 7, 41, 2, 42, 7, 42, 2, 43, 7, 43, 2, 44, 7, 44, 2, 45, 7, 45, 2, 46, 7, 46, 2, 47, 7, 47, 2, 48, 7, 48, 2, 49, 7, 49, 2, 50, 7, 50, 2, 51, 7, 51, 2, 52, 7, 52, 2, 53, 7, 53, 2, 54, 7, 54, 2, 55, 7, 55, 2, 56, 7, 56, 2, 57, 7, 57, 2, 58, 7, 58, 2, 59, 7, 59, 2, 60, 7, 60, 2, 61, 7, 61, 2, 62, 7, 62, 2, 63, 7, 63, 2, 64, 7, 64, 2, 65, 7, 65, 2, 66, 7, 66, 2, 67, 7, 67, 2, 68, 7, 68, 2, 69, 7, 69, 2, 70, 7, 70, 2, 71, 7, 71, 2, 72, 7, 72, 2, 73, 7, 73, 2, 74, 7, 74, 2, 75, 7, 75, 2, 76, 7, 76, 2, 77, 7, 77, 2, 78, 7, 78, 2, 79, 7, 79, 2, 80, 7, 80, 2, 81, 7, 81, 2, 82, 7, 82, 2, 83, 7, 83, 2, 84, 7, 84, 2, 85, 7, 85, 1, 0, 5, 0, 174, 8, 0, 10, 0, 12, 0, 177, 9, 0, 1, 0, 1, 0, 1, 1, 1, 1, 3, 1, 183, 8, 1, 1, 2, 1, 2, 1, 3, 1, 3, 1, 3, 1, 3, 1, 3, 3, 3, 192, 8, 3, 1, 4, 1, 4, 1, 4, 5, 4, 197, 8, 4, 10, 4, 12, 4, 200, 9, 4, 1, 4, 3, 4, 203, 8, 4, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 1, 5, 3, 5, 217, 8, 5, 1, 6, 1, 6, 3, 6, 221, 8, 6, 1, 6, 3, 6, 224, 8, 6, 1, 7, 1, 7, 3, 7, 228, 8, 7, 1, 7, 3, 7, 231, 8, 7, 1, 8, 1, 8, 1, 8, 1, 8, 1, 8, 3, 8, 238, 8, 8, 1, 8, 1, 8, 3, 8, 242, 8, 8, 1, 8, 1, 8, 1, 9, 1, 9, 1, 9, 5, 9, 249, 8, 9, 10, 9, 12, 9, 252, 9, 9, 1, 9, 1, 9, 3, 9, 256, 8, 9, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 1, 10, 3, 10, 265, 8, 10, 1, 11, 1, 11, 1, 11, 1, 11, 1, 11, 1, 11, 3, 11, 273, 8, 11, 1, 12, 1, 12, 1, 12, 1, 12, 1, 12, 3, 12, 280, 8, 12, 1, 12, 1, 12, 3, 12, 284, 8, 12, 1, 12, 1, 12, 1, 12, 1, 12, 3, 12, 290, 8, 12, 1, 12, 1, 12, 1, 12, 3, 12, 295, 8, 12, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 3, 13, 303, 8, 13, 1, 13, 1, 13, 1, 13, 1, 13, 1, 13, 3, 13, 310, 8, 13, 1, 14, 1, 14, 1, 14, 1, 14, 3, 14, 316, 8, 14, 1, 14, 1, 14, 1, 14, 1, 15, 1, 15, 1, 15, 1, 15, 1, 15, 1, 16, 1, 16, 3, 16, 328, 8, 16, 1, 17, 1, 17, 1, 18, 1, 18, 5, 18, 334, 8, 18, 10, 18, 12, 18, 337, 9, 18, 1, 18, 1, 18, 1, 19, 1, 19, 1, 19, 1, 19, 1, 20, 1, 20, 1, 20, 5, 20, 348, 8, 20, 10, 20, 12, 20, 351, 9, 20, 1, 20, 3, 20, 354, 8, 20, 1, 21, 1, 21, 1, 21, 3, 21, 359, 8, 21, 1, 21, 1, 21, 1, 22, 1, 22, 1, 22, 1, 22, 5, 22, 367, 8, 22, 10, 22, 12, 22, 370, 9, 22, 1, 23, 1, 23, 1, 23, 1, 23, 1, 23, 1, 23, 3, 23, 378, 8, 23, 1, 24, 3, 24, 381, 8, 24, 1, 24, 1, 24, 3, 24, 385, 8, 24, 1, 24, 3, 24, 388, 8, 24, 1, 24, 1, 24, 3, 24, 392, 8, 24, 1, 24, 3, 24, 395, 8, 24, 1, 24, 3, 24, 398, 8, 24, 1, 24, 3, 24, 401, 8, 24, 1, 24, 3, 24, 404, 8, 24, 1, 24, 1, 24, 3, 24, 408, 8, 24, 1, 24, 1, 24, 3, 24, 412, 8, 24, 1, 24, 3, 24, 415, 8, 24, 1, 24, 3, 24, 418, 8, 24, 1, 24, 3, 24, 421, 8, 24, 1, 24, 1, 24, 3, 24, 425, 8, 24, 1, 24, 3, 24, 428, 8, 24, 1, 25, 1, 25, 1, 25, 1, 26, 1, 26, 1, 26, 1, 26, 3, 26, 437, 8, 26, 1, 27, 1, 27, 1, 27, 1, 28, 3, 28, 443, 8, 28, 1, 28, 1, 28, 1, 28, 1, 28, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 1, 29, 5, 29, 462, 8, 29, 10, 29, 12, 29, 465, 9, 29, 1, 30, 1, 30, 1, 30, 1, 31, 1, 31, 1, 31, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 1, 32, 3, 32, 481, 8, 32, 1, 33, 1, 33, 1, 33, 1, 34, 1, 34, 1, 34, 1, 34, 1, 35, 1, 35, 1, 35, 1, 35, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 498, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 504, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 510, 8, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 1, 36, 3, 36, 521, 8, 36, 3, 36, 523, 8, 36, 1, 37, 1, 37, 1, 37, 1, 38, 1, 38, 1, 38, 1, 39, 1, 39, 1, 39, 3, 39, 534, 8, 39, 1, 39, 3, 39, 537, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 3, 39, 543, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 1, 39, 1, 39, 3, 39, 551, 8, 39, 1, 39, 1, 39, 1, 39, 1, 39, 5, 39, 557, 8, 39, 10, 39, 12, 39, 560, 9, 39, 1, 40, 3, 40, 563, 8, 40, 1, 40, 1, 40, 1, 40, 3, 40, 568, 8, 40, 1, 40, 3, 40, 571, 8, 40, 1, 40, 3, 40, 574, 8, 40, 1, 40, 1, 40, 3, 40, 578, 8, 40, 1, 40, 1, 40, 3, 40, 582, 8, 40, 1, 40, 3, 40, 585, 8, 40, 3, 40, 587, 8, 40, 1, 40, 3, 40, 590, 8, 40, 1, 40, 1, 40, 3, 40, 594, 8, 40, 1, 40, 1, 40, 3, 40, 598, 8, 40, 1, 40, 3, 40, 601, 8, 40, 3, 40, 603, 8, 40, 3, 40, 605, 8, 40, 1, 41, 1, 41, 1, 41, 3, 41, 610, 8, 41, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 1, 42, 3, 42, 621, 8, 42, 1, 43, 1, 43, 1, 43, 1, 43, 3, 43, 627, 8, 43, 1, 44, 1, 44, 1, 44, 5, 44, 632, 8, 44, 10, 44, 12, 44, 635, 9, 44, 1, 45, 1, 45, 3, 45, 639, 8, 45, 1, 45, 1, 45, 3, 45, 643, 8, 45, 1, 45, 1, 45, 3, 45, 647, 8, 45, 1, 46, 1, 46, 1, 46, 1, 46, 3, 46, 653, 8, 46, 3, 46, 655, 8, 46, 1, 47, 1, 47, 1, 47, 5, 47, 660, 8, 47, 10, 47, 12, 47, 663, 9, 47, 1, 48, 1, 48, 1, 48, 1, 48, 1, 49, 3, 49, 670, 8, 49, 1, 49, 3, 49, 673, 8, 49, 1, 49, 3, 49, 676, 8, 49, 1, 50, 1, 50, 1, 50, 1, 50, 1, 51, 1, 51, 1, 51, 1, 51, 1, 52, 1, 52, 1, 52, 1, 53, 1, 53, 1, 53, 1, 53, 1, 53, 1, 53, 3, 53, 695, 8, 53, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 1, 54, 3, 54, 709, 8, 54, 1, 55, 1, 55, 1, 55, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 723, 8, 56, 10, 56, 12, 56, 726, 9, 56, 1, 56, 3, 56, 729, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 738, 8, 56, 10, 56, 12, 56, 741, 9, 56, 1, 56, 3, 56, 744, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 5, 56, 753, 8, 56, 10, 56, 12, 56, 756, 9, 56, 1, 56, 3, 56, 759, 8, 56, 1, 56, 1, 56, 1, 56, 1, 56, 1, 56, 3, 56, 766, 8, 56, 1, 56, 1, 56, 3, 56, 770, 8, 56, 1, 57, 1, 57, 1, 57, 5, 57, 775, 8, 57, 10, 57, 12, 57, 778, 9, 57, 1, 57, 3, 57, 781, 8, 57, 1, 58, 1, 58, 1, 58, 3, 58, 786, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 4, 58, 793, 8, 58, 11, 58, 12, 58, 794, 1, 58, 1, 58, 3, 58, 799, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 823, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 840, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 846, 8, 58, 1, 58, 3, 58, 849, 8, 58, 1, 58, 3, 58, 852, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 862, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 868, 8, 58, 1, 58, 3, 58, 871, 8, 58, 1, 58, 3, 58, 874, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 882, 8, 58, 1, 58, 3, 58, 885, 8, 58, 1, 58, 1, 58, 3, 58, 889, 8, 58, 1, 58, 3, 58, 892, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 906, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 923, 8, 58, 1, 58, 1, 58, 1, 58, 3, 58, 928, 8, 58, 1, 58, 1, 58, 1, 58, 3, 58, 933, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 939, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 946, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 958, 8, 58, 1, 58, 1, 58, 3, 58, 962, 8, 58, 1, 58, 3, 58, 965, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 974, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 988, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 1004, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 1033, 8, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 1, 58, 3, 58, 1041, 8, 58, 5, 58, 1043, 8, 58, 10, 58, 12, 58, 1046, 9, 58, 1, 59, 1, 59, 1, 59, 1, 59, 5, 59, 1052, 8, 59, 10, 59, 12, 59, 1055, 9, 59, 1, 59, 3, 59, 1058, 8, 59, 1, 59, 1, 59, 1, 59, 1, 59, 1, 59, 5, 59, 1065, 8, 59, 10, 59, 12, 59, 1068, 9, 59, 1, 59, 3, 59, 1071, 8, 59, 1, 59, 1, 59, 3, 59, 1075, 8, 59, 1, 59, 1, 59, 1, 59, 3, 59, 1080, 8, 59, 1, 60, 1, 60, 1, 60, 5, 60, 1085, 8, 60, 10, 60, 12, 60, 1088, 9, 60, 1, 60, 1, 60, 1, 60, 1, 60, 1, 60, 1, 60, 5, 60, 1096, 8, 60, 10, 60, 12, 60, 1099, 9, 60, 1, 60, 1, 60, 1, 60, 1, 60, 1, 60, 1, 60, 3, 60, 1107, 8, 60, 1, 60, 1, 60, 1, 60, 1, 60, 1, 60, 3, 60, 1114, 8, 60, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 1, 61, 3, 61, 1127, 8, 61, 1, 62, 1, 62, 1, 62, 5, 62, 1132, 8, 62, 10, 62, 12, 62, 1135, 9, 62, 1, 62, 3, 62, 1138, 8, 62, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 1, 63, 3, 63, 1150, 8, 63, 1, 64, 1, 64, 1, 64, 1, 64, 3, 64, 1156, 8, 64, 1, 64, 3, 64, 1159, 8, 64, 1, 65, 1, 65, 1, 65, 5, 65, 1164, 8, 65, 10, 65, 12, 65, 1167, 9, 65, 1, 66, 1, 66, 1, 66, 1, 66, 1, 66, 1, 66, 1, 66, 1, 66, 1, 66, 3, 66, 1178, 8, 66, 1, 66, 1, 66, 1, 66, 1, 66, 3, 66, 1184, 8, 66, 5, 66, 1186, 8, 66, 10, 66, 12, 66, 1189, 9, 66, 1, 67, 1, 67, 1, 67, 3, 67, 1194, 8, 67, 1, 67, 1, 67, 1, 68, 1, 68, 1, 68, 3, 68, 1201, 8, 68, 1, 68, 1, 68, 1, 69, 1, 69, 1, 69, 5, 69, 1208, 8, 69, 10, 69, 12, 69, 1211, 9, 69, 1, 69, 3, 69, 1214, 8, 69, 1, 70, 1, 70, 1, 71, 1, 71, 1, 71, 1, 71, 1, 71, 1, 71, 3, 71, 1224, 8, 71, 3, 71, 1226, 8, 71, 1, 72, 3, 72, 1229, 8, 72, 1, 72, 1, 72, 1, 72, 1, 72, 1, 72, 1, 72, 3, 72, 1237, 8, 72, 1, 73, 1, 73, 1, 73, 3, 73, 1242, 8, 73, 1, 74, 1, 74, 1, 75, 1, 75, 1, 76, 1, 76, 1, 77, 1, 77, 3, 77, 1252, 8, 77, 1, 78, 1, 78, 1, 78, 3, 78, 1257, 8, 78, 1, 79, 1, 79, 1, 79, 1, 79, 1, 80, 1, 80, 1, 80, 1, 80, 1, 81, 1, 81, 3, 81, 1269, 8, 81, 1, 82, 1, 82, 5, 82, 1273, 8, 82, 10, 82, 12, 82, 1276, 9, 82, 1, 82, 1, 82, 1, 83, 1, 83, 1, 83, 1, 83, 1, 83, 3, 83, 1285, 8, 83, 1, 84, 1, 84, 5, 84, 1289, 8, 84, 10, 84, 12, 84, 1292, 9, 84, 1, 84, 1, 84, 1, 85, 1, 85, 1, 85, 1, 85, 1, 85, 3, 85, 1301, 8, 85, 1, 85, 0, 3, 78, 116, 132, 86, 0, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28, 30, 32, 34, 36, 38, 40, 42, 44, 46, 48, 50, 52, 54, 56, 58, 60, 62, 64, 66, 68, 70, 72, 74, 76, 78, 80, 82, 84, 86, 88, 90, 92, 94, 96, 98, 100, 102, 104, 106, 108, 110, 112, 114, 116, 118, 120, 122, 124, 126, 128, 130, 132, 134, 136, 138, 140, 142, 144, 146, 148, 150, 152, 154, 156, 158, 160, 162, 164, 166, 168, 170, 0, 16, 2, 0, 18, 18, 74, 74, 2, 0, 44, 44, 51, 51, 3, 0, 1, 1, 4, 4, 8, 8, 4, 0, 1, 1, 3, 4, 8, 8, 80, 80, 2, 0, 51, 51, 73, 73, 2, 0, 1, 1, 4, 4, 2, 0, 7, 7, 22, 23, 2, 0, 30, 30, 49, 49, 2, 0, 71, 71, 76, 76, 3, 0, 10, 10, 50, 50, 90, 90, 2, 0, 41, 41, 53, 53, 1, 0, 107, 108, 2, 0, 118, 118, 139, 139, 7, 0, 21, 21, 38, 38, 55, 56, 70, 70, 78, 78, 97, 97, 103, 103, 16, 0, 1, 13, 15, 20, 22, 28, 30, 30, 32, 37, 39, 42, 44, 51, 53, 54, 58, 58, 60, 69, 71, 77, 79, 83, 85, 92, 94, 96, 98, 99, 101, 102, 4, 0, 20, 20, 30, 30, 39, 39, 48, 48, 1475, 0, 175, 1, 0, 0, 0, 2, 182, 1, 0, 0, 0, 4, 184, 1, 0, 0, 0, 6, 186, 1, 0, 0, 0, 8, 193, 1, 0, 0, 0, 10, 216, 1, 0, 0, 0, 12, 218, 1, 0, 0, 0, 14, 225, 1, 0, 0, 0, 16, 232, 1, 0, 0, 0, 18, 245, 1, 0, 0, 0, 20, 257, 1, 0, 0, 0, 22, 266, 1, 0, 0, 0, 24, 274, 1, 0, 0, 0, 26, 296, 1, 0, 0, 0, 28, 311, 1, 0, 0, 0, 30, 320, 1, 0, 0, 0, 32, 325, 1, 0, 0, 0, 34, 329, 1, 0, 0, 0, 36, 331, 1, 0, 0, 0, 38, 340, 1, 0, 0, 0, 40, 344, 1, 0, 0, 0, 42, 358, 1, 0, 0, 0, 44, 362, 1, 0, 0, 0, 46, 377, 1, 0, 0, 0, 48, 380, 1, 0, 0, 0, 50, 429, 1, 0, 0, 0, 52, 432, 1, 0, 0, 0, 54, 438, 1, 0, 0, 0, 56, 442, 1, 0, 0, 0, 58, 448, 1, 0, 0, 0, 60, 466, 1, 0, 0, 0, 62, 469, 1, 0, 0, 0, 64, 472, 1, 0, 0, 0, 66, 482, 1, 0, 0, 0, 68, 485, 1, 0, 0, 0, 70, 489, 1, 0, 0, 0, 72, 522, 1, 0, 0, 0, 74, 524, 1, 0, 0, 0, 76, 527, 1, 0, 0, 0, 78, 542, 1, 0, 0, 0, 80, 604, 1, 0, 0, 0, 82, 609, 1, 0, 0, 0, 84, 620, 1, 0, 0, 0, 86, 622, 1, 0, 0, 0, 88, 628, 1, 0, 0, 0, 90, 636, 1, 0, 0, 0, 92, 654, 1, 0, 0, 0, 94, 656, 1, 0, 0, 0, 96, 664, 1, 0, 0, 0, 98, 669, 1, 0, 0, 0, 100, 677, 1, 0, 0, 0, 102, 681, 1, 0, 0, 0, 104, 685, 1, 0, 0, 0, 106, 694, 1, 0, 0, 0, 108, 708, 1, 0, 0, 0, 110, 710, 1, 0, 0, 0, 112, 769, 1, 0, 0, 0, 114, 771, 1, 0, 0, 0, 116, 932, 1, 0, 0, 0, 118, 1074, 1, 0, 0, 0, 120, 1113, 1, 0, 0, 0, 122, 1126, 1, 0, 0, 0, 124, 1128, 1, 0, 0, 0, 126, 1149, 1, 0, 0, 0, 128, 1158, 1, 0, 0, 0, 130, 1160, 1, 0, 0, 0, 132, 1177, 1, 0, 0, 0, 134, 1190, 1, 0, 0, 0, 136, 1200, 1, 0, 0, 0, 138, 1204, 1, 0, 0, 0, 140, 1215, 1, 0, 0, 0, 142, 1225, 1, 0, 0, 0, 144, 1228, 1, 0, 0, 0, 146, 1241, 1, 0, 0, 0, 148, 1243, 1, 0, 0, 0, 150, 1245, 1, 0, 0, 0, 152, 1247, 1, 0, 0, 0, 154, 1251, 1, 0, 0, 0, 156, 1256, 1, 0, 0, 0, 158, 1258, 1, 0, 0, 0, 160, 1262, 1, 0, 0, 0, 162, 1268, 1, 0, 0, 0, 164, 1270, 1, 0, 0, 0, 166, 1284, 1, 0, 0, 0, 168, 1286, 1, 0, 0, 0, 170, 1300, 1, 0, 0, 0, 172, 174, 3, 2, 1, 0, 173, 172, 1, 0, 0, 0, 174, 177, 1, 0, 0, 0, 175, 173, 1, 0, 0, 0, 175, 176, 1, 0, 0, 0, 176, 178, 1, 0, 0, 0, 177, 175, 1, 0, 0, 0, 178, 179, 5, 0, 0, 1, 179, 1, 1, 0, 0, 0, 180, 183, 3, 6, 3, 0, 181, 183, 3, 10, 5, 0, 182, 180, 1, 0, 0, 0, 182, 181, 1, 0, 0, 0, 183, 3, 1, 0, 0, 0, 184, 185, 3, 116, 58, 0, 185, 5, 1, 0, 0, 0, 186, 187, 5, 52, 0, 0, 187, 191, 3, 156, 78, 0, 188, 189, 5, 115, 0, 0, 189, 190, 5, 122, 0, 0, 190, 192, 3, 4, 2, 0, 191, 188, 1, 0, 0, 0, 191, 192, 1, 0, 0, 0, 192, 7, 1, 0, 0, 0, 193, 198, 3, 156, 78, 0, 194, 195, 5, 116, 0, 0, 195, 197, 3, 156, 78, 0, 196, 194, 1, 0, 0, 0, 197, 200, 1, 0, 0, 0, 198, 196, 1, 0, 0, 0, 198, 199, 1, 0, 0, 0, 199, 202, 1, 0, 0, 0, 200, 198, 1, 0, 0, 0, 201, 203, 5, 116, 0, 0, 202, 201, 1, 0, 0, 0, 202, 203, 1, 0, 0, 0, 203, 9, 1, 0, 0, 0, 204, 217, 3, 12, 6, 0, 205, 217, 3, 14, 7, 0, 206, 217, 3, 18, 9, 0, 207, 217, 3, 20, 10, 0, 208, 217, 3, 22, 11, 0, 209, 217, 3, 26, 13, 0, 210, 217, 3, 24, 12, 0, 211, 217, 3, 28, 14, 0, 212, 217, 3, 30, 15, 0, 213, 217, 3, 36, 18, 0, 214, 217, 3, 32, 16, 0, 215, 217, 3, 34, 17, 0, 216, 204, 1, 0, 0, 0, 216, 205, 1, 0, 0, 0, 216, 206, 1, 0, 0, 0, 216, 207, 1, 0, 0, 0, 216, 208, 1, 0, 0, 0, 216, 209, 1, 0, 0, 0, 216, 210, 1, 0, 0, 0, 216, 211, 1, 0, 0, 0, 216, 212, 1, 0, 0, 0, 216, 213, 1, 0, 0, 0, 216, 214, 1, 0, 0, 0, 216, 215, 1, 0, 0, 0, 217, 11, 1, 0, 0, 0, 218, 220, 5, 72, 0, 0, 219, 221, 3, 4, 2, 0, 220, 219, 1, 0, 0, 0, 220, 221, 1, 0, 0, 0, 221, 223, 1, 0, 0, 0, 222, 224, 5, 150, 0, 0, 223, 222, 1, 0, 0, 0, 223, 224, 1, 0, 0, 0, 224, 13, 1, 0, 0, 0, 225, 227, 5, 84, 0, 0, 226, 228, 3, 4, 2, 0, 227, 226, 1, 0, 0, 0, 227, 228, 1, 0, 0, 0, 228, 230, 1, 0, 0, 0, 229, 231, 5, 150, 0, 0, 230, 229, 1, 0, 0, 0, 230, 231, 1, 0, 0, 0, 231, 15, 1, 0, 0, 0, 232, 241, 5, 14, 0, 0, 233, 234, 5, 130, 0, 0, 234, 237, 3, 156, 78, 0, 235, 236, 5, 115, 0, 0, 236, 238, 3, 156, 78, 0, 237, 235, 1, 0, 0, 0, 237, 238, 1, 0, 0, 0, 238, 239, 1, 0, 0, 0, 239, 240, 5, 149, 0, 0, 240, 242, 1, 0, 0, 0, 241, 233, 1, 0, 0, 0, 241, 242, 1, 0, 0, 0, 242, 243, 1, 0, 0, 0, 243, 244, 3, 36, 18, 0, 244, 17, 1, 0, 0, 0, 245, 246, 5, 93, 0, 0, 246, 250, 3, 36, 18, 0, 247, 249, 3, 16, 8, 0, 248, 247, 1, 0, 0, 0, 249, 252, 1, 0, 0, 0, 250, 248, 1, 0, 0, 0, 250, 251, 1, 0, 0, 0, 251, 255, 1, 0, 0, 0, 252, 250, 1, 0, 0, 0, 253, 254, 5, 29, 0, 0, 254, 256, 3, 36, 18, 0, 255, 253, 1, 0, 0, 0, 255, 256, 1, 0, 0, 0, 256, 19, 1, 0, 0, 0, 257, 258, 5, 40, 0, 0, 258, 259, 5, 130, 0, 0, 259, 260, 3, 4, 2, 0, 260, 261, 5, 149, 0, 0, 261, 264, 3, 10, 5, 0, 262, 263, 5, 25, 0, 0, 263, 265, 3, 10, 5, 0, 264, 262, 1, 0, 0, 0, 264, 265, 1, 0, 0, 0, 265, 21, 1, 0, 0, 0, 266, 267, 5, 100, 0, 0, 267, 268, 5, 130, 0, 0, 268, 269, 3, 4, 2, 0, 269, 270, 5, 149, 0, 0, 270, 272, 3, 10, 5, 0, 271, 273, 5, 150, 0, 0, 272, 271, 1, 0, 0, 0, 272, 273, 1, 0, 0, 0, 273, 23, 1, 0, 0, 0, 274, 275, 5, 33, 0, 0, 275, 279, 5, 130, 0, 0, 276, 280, 3, 6, 3, 0, 277, 280, 3, 30, 15, 0, 278, 280, 3, 4, 2, 0, 279, 276, 1, 0, 0, 0, 279, 277, 1, 0, 0, 0, 279, 278, 1, 0, 0, 0, 279, 280, 1, 0, 0, 0, 280, 281, 1, 0, 0, 0, 281, 283, 5, 150, 0, 0, 282, 284, 3, 4, 2, 0, 283, 282, 1, 0, 0, 0, 283, 284, 1, 0, 0, 0, 284, 285, 1, 0, 0, 0, 285, 289, 5, 150, 0, 0, 286, 290, 3, 6, 3, 0, 287, 290, 3, 30, 15, 0, 288, 290, 3, 4, 2, 0, 289, 286, 1, 0, 0, 0, 289, 287, 1, 0, 0, 0, 289, 288, 1, 0, 0, 0, 289, 290, 1, 0, 0, 0, 290, 291, 1, 0, 0, 0, 291, 292, 5, 149, 0, 0, 292, 294, 3, 10, 5, 0, 293, 295, 5, 150, 0, 0, 294, 293, 1, 0, 0, 0, 294, 295, 1, 0, 0, 0, 295, 25, 1, 0, 0, 0, 296, 297, 5, 33, 0, 0, 297, 298, 5, 130, 0, 0, 298, 299, 5, 52, 0, 0, 299, 302, 3, 156, 78, 0, 300, 301, 5, 116, 0, 0, 301, 303, 3, 156, 78, 0, 302, 300, 1, 0, 0, 0, 302, 303, 1, 0, 0, 0, 303, 304, 1, 0, 0, 0, 304, 305, 5, 42, 0, 0, 305, 306, 3, 4, 2, 0, 306, 307, 5, 149, 0, 0, 307, 309, 3, 10, 5, 0, 308, 310, 5, 150, 0, 0, 309, 308, 1, 0, 0, 0, 309, 310, 1, 0, 0, 0, 310, 27, 1, 0, 0, 0, 311, 312, 5, 31, 0, 0, 312, 313, 3, 156, 78, 0, 313, 315, 5, 130, 0, 0, 314, 316, 3, 8, 4, 0, 315, 314, 1, 0, 0, 0, 315, 316, 1, 0, 0, 0, 316, 317, 1, 0, 0, 0, 317, 318, 5, 149, 0, 0, 318, 319, 3, 36, 18, 0, 319, 29, 1, 0, 0, 0, 320, 321, 3, 4, 2, 0, 321, 322, 5, 115, 0, 0, 322, 323, 5, 122, 0, 0, 323, 324, 3, 4, 2, 0, 324, 31, 1, 0, 0, 0, 325, 327, 3, 4, 2, 0, 326, 328, 5, 150, 0, 0, 327, 326, 1, 0, 0, 0, 327, 328, 1, 0, 0, 0, 328, 33, 1, 0, 0, 0, 329, 330, 5, 150, 0, 0, 330, 35, 1, 0, 0, 0, 331, 335, 5, 128, 0, 0, 332, 334, 3, 2, 1, 0, 333, 332, 1, 0, 0, 0, 334, 337, 1, 0, 0, 0, 335, 333, 1, 0, 0, 0, 335, 336, 1, 0, 0, 0, 336, 338, 1, 0, 0, 0, 337, 335, 1, 0, 0, 0, 338, 339, 5, 147, 0, 0, 339, 37, 1, 0, 0, 0, 340, 341, 3, 4, 2, 0, 341, 342, 5, 115, 0, 0, 342, 343, 3, 4, 2, 0, 343, 39, 1, 0, 0, 0, 344, 349, 3, 38, 19, 0, 345, 346, 5, 116, 0, 0, 346, 348, 3, 38, 19, 0, 347, 345, 1, 0, 0, 0, 348, 351, 1, 0, 0, 0, 349, 347, 1, 0, 0, 0, 349, 350, 1, 0, 0, 0, 350, 353, 1, 0, 0, 0, 351, 349, 1, 0, 0, 0, 352, 354, 5, 116, 0, 0, 353, 352, 1, 0, 0, 0, 353, 354, 1, 0, 0, 0, 354, 41, 1, 0, 0, 0, 355, 359, 3, 44, 22, 0, 356, 359, 3, 48, 24, 0, 357, 359, 3, 120, 60, 0, 358, 355, 1, 0, 0, 0, 358, 356, 1, 0, 0, 0, 358, 357, 1, 0, 0, 0, 359, 360, 1, 0, 0, 0, 360, 361, 5, 0, 0, 1, 361, 43, 1, 0, 0, 0, 362, 368, 3, 46, 23, 0, 363, 364, 5, 95, 0, 0, 364, 365, 5, 1, 0, 0, 365, 367, 3, 46, 23, 0, 366, 363, 1, 0, 0, 0, 367, 370, 1, 0, 0, 0, 368, 366, 1, 0, 0, 0, 368, 369, 1, 0, 0, 0, 369, 45, 1, 0, 0, 0, 370, 368, 1, 0, 0, 0, 371, 378, 3, 48, 24, 0, 372, 373, 5, 130, 0, 0, 373, 374, 3, 44, 22, 0, 374, 375, 5, 149, 0, 0, 375, 378, 1, 0, 0, 0, 376, 378, 3, 160, 80, 0, 377, 371, 1, 0, 0, 0, 377, 372, 1, 0, 0, 0, 377, 376, 1, 0, 0, 0, 378, 47, 1, 0, 0, 0, 379, 381, 3, 50, 25, 0, 380, 379, 1, 0, 0, 0, 380, 381, 1, 0, 0, 0, 381, 382, 1, 0, 0, 0, 382, 384, 5, 79, 0, 0, 383, 385, 5, 24, 0, 0, 384, 383, 1, 0, 0, 0, 384, 385, 1, 0, 0, 0, 385, 387, 1, 0, 0, 0, 386, 388, 3, 52, 26, 0, 387, 386, 1, 0, 0, 0, 387, 388, 1, 0, 0, 0, 388, 389, 1, 0, 0, 0, 389, 391, 3, 114, 57, 0, 390, 392, 3, 54, 27, 0, 391, 390, 1, 0, 0, 0, 391, 392, 1, 0, 0, 0, 392, 394, 1, 0, 0, 0, 393, 395, 3, 56, 28, 0, 394, 393, 1, 0, 0, 0, 394, 395, 1, 0, 0, 0, 395, 397, 1, 0, 0, 0, 396, 398, 3, 60, 30, 0, 397, 396, 1, 0, 0, 0, 397, 398, 1, 0, 0, 0, 398, 400, 1, 0, 0, 0, 399, 401, 3, 62, 31, 0, 400, 399, 1, 0, 0, 0, 400, 401, 1, 0, 0, 0, 401, 403, 1, 0, 0, 0, 402, 404, 3, 64, 32, 0, 403, 402, 1, 0, 0, 0, 403, 404, 1, 0, 0, 0, 404, 407, 1, 0, 0, 0, 405, 406, 5, 102, 0, 0, 406, 408, 7, 0, 0, 0, 407, 405, 1, 0, 0, 0, 407, 408, 1, 0, 0, 0, 408, 411, 1, 0, 0, 0, 409, 410, 5, 102, 0, 0, 410, 412, 5, 89, 0, 0, 411, 409, 1, 0, 0, 0, 411, 412, 1, 0, 0, 0, 412, 414, 1, 0, 0, 0, 413, 415, 3, 66, 33, 0, 414, 413, 1, 0, 0, 0, 414, 415, 1, 0, 0, 0, 415, 417, 1, 0, 0, 0, 416, 418, 3, 58, 29, 0, 417, 416, 1, 0, 0, 0, 417, 418, 1, 0, 0, 0, 418, 420, 1, 0, 0, 0, 419, 421, 3, 68, 34, 0, 420, 419, 1, 0, 0, 0, 420, 421, 1, 0, 0, 0, 421, 424, 1, 0, 0, 0, 422, 425, 3, 72, 36, 0, 423, 425, 3, 74, 37, 0, 424, 422, 1, 0, 0, 0, 424, 423, 1, 0, 0, 0, 424, 425, 1, 0, 0, 0, 425, 427, 1, 0, 0, 0, 426, 428, 3, 76, 38, 0, 427, 426, 1, 0, 0, 0, 427, 428, 1, 0, 0, 0, 428, 49, 1, 0, 0, 0, 429, 430, 5, 102, 0, 0, 430, 431, 3, 124, 62, 0, 431, 51, 1, 0, 0, 0, 432, 433, 5, 88, 0, 0, 433, 436, 5, 108, 0, 0, 434, 435, 5, 102, 0, 0, 435, 437, 5, 85, 0, 0, 436, 434, 1, 0, 0, 0, 436, 437, 1, 0, 0, 0, 437, 53, 1, 0, 0, 0, 438, 439, 5, 34, 0, 0, 439, 440, 3, 78, 39, 0, 440, 55, 1, 0, 0, 0, 441, 443, 7, 1, 0, 0, 442, 441, 1, 0, 0, 0, 442, 443, 1, 0, 0, 0, 443, 444, 1, 0, 0, 0, 444, 445, 5, 5, 0, 0, 445, 446, 5, 47, 0, 0, 446, 447, 3, 114, 57, 0, 447, 57, 1, 0, 0, 0, 448, 449, 5, 101, 0, 0, 449, 450, 3, 156, 78, 0, 450, 451, 5, 6, 0, 0, 451, 452, 5, 130, 0, 0, 452, 453, 3, 98, 49, 0, 453, 463, 5, 149, 0, 0, 454, 455, 5, 116, 0, 0, 455, 456, 3, 156, 78, 0, 456, 457, 5, 6, 0, 0, 457, 458, 5, 130, 0, 0, 458, 459, 3, 98, 49, 0, 459, 460, 5, 149, 0, 0, 460, 462, 1, 0, 0, 0, 461, 454, 1, 0, 0, 0, 462, 465, 1, 0, 0, 0, 463, 461, 1, 0, 0, 0, 463, 464, 1, 0, 0, 0, 464, 59, 1, 0, 0, 0, 465, 463, 1, 0, 0, 0, 466, 467, 5, 69, 0, 0, 467, 468, 3, 116, 58, 0, 468, 61, 1, 0, 0, 0, 469, 470, 5, 99, 0, 0, 470, 471, 3, 116, 58, 0, 471, 63, 1, 0, 0, 0, 472, 473, 5, 36, 0, 0, 473, 480, 5, 11, 0, 0, 474, 475, 7, 0, 0, 0, 475, 476, 5, 130, 0, 0, 476, 477, 3, 114, 57, 0, 477, 478, 5, 149, 0, 0, 478, 481, 1, 0, 0, 0, 479, 481, 3, 114, 57, 0, 480, 474, 1, 0, 0, 0, 480, 479, 1, 0, 0, 0, 481, 65, 1, 0, 0, 0, 482, 483, 5, 37, 0, 0, 483, 484, 3, 116, 58, 0, 484, 67, 1, 0, 0, 0, 485, 486, 5, 64, 0, 0, 486, 487, 5, 11, 0, 0, 487, 488, 3, 88, 44, 0, 488, 69, 1, 0, 0, 0, 489, 490, 5, 64, 0, 0, 490, 491, 5, 11, 0, 0, 491, 492, 3, 114, 57, 0, 492, 71, 1, 0, 0, 0, 493, 494, 5, 54, 0, 0, 494, 497, 3, 116, 58, 0, 495, 496, 5, 116, 0, 0, 496, 498, 3, 116, 58, 0, 497, 495, 1, 0, 0, 0, 497, 498, 1, 0, 0, 0, 498, 503, 1, 0, 0, 0, 499, 500, 5, 102, 0, 0, 500, 504, 5, 85, 0, 0, 501, 502, 5, 11, 0, 0, 502, 504, 3, 114, 57, 0, 503, 499, 1, 0, 0, 0, 503, 501, 1, 0, 0, 0, 503, 504, 1, 0, 0, 0, 504, 523, 1, 0, 0, 0, 505, 506, 5, 54, 0, 0, 506, 509, 3, 116, 58, 0, 507, 508, 5, 102, 0, 0, 508, 510, 5, 85, 0, 0, 509, 507, 1, 0, 0, 0, 509, 510, 1, 0, 0, 0, 510, 511, 1, 0, 0, 0, 511, 512, 5, 61, 0, 0, 512, 513, 3, 116, 58, 0, 513, 523, 1, 0, 0, 0, 514, 515, 5, 54, 0, 0, 515, 516, 3, 116, 58, 0, 516, 517, 5, 61, 0, 0, 517, 520, 3, 116, 58, 0, 518, 519, 5, 11, 0, 0, 519, 521, 3, 114, 57, 0, 520, 518, 1, 0, 0, 0, 520, 521, 1, 0, 0, 0, 521, 523, 1, 0, 0, 0, 522, 493, 1, 0, 0, 0, 522, 505, 1, 0, 0, 0, 522, 514, 1, 0, 0, 0, 523, 73, 1, 0, 0, 0, 524, 525, 5, 61, 0, 0, 525, 526, 3, 116, 58, 0, 526, 75, 1, 0, 0, 0, 527, 528, 5, 81, 0, 0, 528, 529, 3, 94, 47, 0, 529, 77, 1, 0, 0, 0, 530, 531, 6, 39, -1, 0, 531, 533, 3, 132, 66, 0, 532, 534, 5, 28, 0, 0, 533, 532, 1, 0, 0, 0, 533, 534, 1, 0, 0, 0, 534, 536, 1, 0, 0, 0, 535, 537, 3, 86, 43, 0, 536, 535, 1, 0, 0, 0, 536, 537, 1, 0, 0, 0, 537, 543, 1, 0, 0, 0, 538, 539, 5, 130, 0, 0, 539, 540, 3, 78, 39, 0, 540, 541, 5, 149, 0, 0, 541, 543, 1, 0, 0, 0, 542, 530, 1, 0, 0, 0, 542, 538, 1, 0, 0, 0, 543, 558, 1, 0, 0, 0, 544, 545, 10, 3, 0, 0, 545, 546, 3, 82, 41, 0, 546, 547, 3, 78, 39, 4, 547, 557, 1, 0, 0, 0, 548, 550, 10, 4, 0, 0, 549, 551, 3, 80, 40, 0, 550, 549, 1, 0, 0, 0, 550, 551, 1, 0, 0, 0, 551, 552, 1, 0, 0, 0, 552, 553, 5, 47, 0, 0, 553, 554, 3, 78, 39, 0, 554, 555, 3, 84, 42, 0, 555, 557, 1, 0, 0, 0, 556, 544, 1, 0, 0, 0, 556, 548, 1, 0, 0, 0, 557, 560, 1, 0, 0, 0, 558, 556, 1, 0, 0, 0, 558, 559, 1, 0, 0, 0, 559, 79, 1, 0, 0, 0, 560, 558, 1, 0, 0, 0, 561, 563, 7, 2, 0, 0, 562, 561, 1, 0, 0, 0, 562, 563, 1, 0, 0, 0, 563, 564, 1, 0, 0, 0, 564, 571, 5, 44, 0, 0, 565, 567, 5, 44, 0, 0, 566, 568, 7, 2, 0, 0, 567, 566, 1, 0, 0, 0, 567, 568, 1, 0, 0, 0, 568, 571, 1, 0, 0, 0, 569, 571, 7, 2, 0, 0, 570, 562, 1, 0, 0, 0, 570, 565, 1, 0, 0, 0, 570, 569, 1, 0, 0, 0, 571, 605, 1, 0, 0, 0, 572, 574, 7, 3, 0, 0, 573, 572, 1, 0, 0, 0, 573, 574, 1, 0, 0, 0, 574, 575, 1, 0, 0, 0, 575, 577, 7, 4, 0, 0, 576, 578, 5, 65, 0, 0, 577, 576, 1, 0, 0, 0, 577, 578, 1, 0, 0, 0, 578, 587, 1, 0, 0, 0, 579, 581, 7, 4, 0, 0, 580, 582, 5, 65, 0, 0, 581, 580, 1, 0, 0, 0, 581, 582, 1, 0, 0, 0, 582, 584, 1, 0, 0, 0, 583, 585, 7, 3, 0, 0, 584, 583, 1, 0, 0, 0, 584, 585, 1, 0, 0, 0, 585, 587, 1, 0, 0, 0, 586, 573, 1, 0, 0, 0, 586, 579, 1, 0, 0, 0, 587, 605, 1, 0, 0, 0, 588, 590, 7, 5, 0, 0, 589, 588, 1, 0, 0, 0, 589, 590, 1, 0, 0, 0, 590, 591, 1, 0, 0, 0, 591, 593, 5, 35, 0, 0, 592, 594, 5, 65, 0, 0, 593, 592, 1, 0, 0, 0, 593, 594, 1, 0, 0, 0, 594, 603, 1, 0, 0, 0, 595, 597, 5, 35, 0, 0, 596, 598, 5, 65, 0, 0, 597, 596, 1, 0, 0, 0, 597, 598, 1, 0, 0, 0, 598, 600, 1, 0, 0, 0, 599, 601, 7, 5, 0, 0, 600, 599, 1, 0, 0, 0, 600, 601, 1, 0, 0, 0, 601, 603, 1, 0, 0, 0, 602, 589, 1, 0, 0, 0, 602, 595, 1, 0, 0, 0, 603, 605, 1, 0, 0, 0, 604, 570, 1, 0, 0, 0, 604, 586, 1, 0, 0, 0, 604, 602, 1, 0, 0, 0, 605, 81, 1, 0, 0, 0, 606, 607, 5, 17, 0, 0, 607, 610, 5, 47, 0, 0, 608, 610, 5, 116, 0, 0, 609, 606, 1, 0, 0, 0, 609, 608, 1, 0, 0, 0, 610, 83, 1, 0, 0, 0, 611, 612, 5, 62, 0, 0, 612, 621, 3, 114, 57, 0, 613, 614, 5, 96, 0, 0, 614, 615, 5, 130, 0, 0, 615, 616, 3, 114, 57, 0, 616, 617, 5, 149, 0, 0, 617, 621, 1, 0, 0, 0, 618, 619, 5, 96, 0, 0, 619, 621, 3, 114, 57, 0, 620, 611, 1, 0, 0, 0, 620, 613, 1, 0, 0, 0, 620, 618, 1, 0, 0, 0, 621, 85, 1, 0, 0, 0, 622, 623, 5, 77, 0, 0, 623, 626, 3, 92, 46, 0, 624, 625, 5, 61, 0, 0, 625, 627, 3, 92, 46, 0, 626, 624, 1, 0, 0, 0, 626, 627, 1, 0, 0, 0, 627, 87, 1, 0, 0, 0, 628, 633, 3, 90, 45, 0, 629, 630, 5, 116, 0, 0, 630, 632, 3, 90, 45, 0, 631, 629, 1, 0, 0, 0, 632, 635, 1, 0, 0, 0, 633, 631, 1, 0, 0, 0, 633, 634, 1, 0, 0, 0, 634, 89, 1, 0, 0, 0, 635, 633, 1, 0, 0, 0, 636, 638, 3, 116, 58, 0, 637, 639, 7, 6, 0, 0, 638, 637, 1, 0, 0, 0, 638, 639, 1, 0, 0, 0, 639, 642, 1, 0, 0, 0, 640, 641, 5, 60, 0, 0, 641, 643, 7, 7, 0, 0, 642, 640, 1, 0, 0, 0, 642, 643, 1, 0, 0, 0, 643, 646, 1, 0, 0, 0, 644, 645, 5, 16, 0, 0, 645, 647, 5, 110, 0, 0, 646, 644, 1, 0, 0, 0, 646, 647, 1, 0, 0, 0, 647, 91, 1, 0, 0, 0, 648, 655, 3, 160, 80, 0, 649, 652, 3, 144, 72, 0, 650, 651, 5, 151, 0, 0, 651, 653, 3, 144, 72, 0, 652, 650, 1, 0, 0, 0, 652, 653, 1, 0, 0, 0, 653, 655, 1, 0, 0, 0, 654, 648, 1, 0, 0, 0, 654, 649, 1, 0, 0, 0, 655, 93, 1, 0, 0, 0, 656, 661, 3, 96, 48, 0, 657, 658, 5, 116, 0, 0, 658, 660, 3, 96, 48, 0, 659, 657, 1, 0, 0, 0, 660, 663, 1, 0, 0, 0, 661, 659, 1, 0, 0, 0, 661, 662, 1, 0, 0, 0, 662, 95, 1, 0, 0, 0, 663, 661, 1, 0, 0, 0, 664, 665, 3, 156, 78, 0, 665, 666, 5, 122, 0, 0, 666, 667, 3, 146, 73, 0, 667, 97, 1, 0, 0, 0, 668, 670, 3, 100, 50, 0, 669, 668, 1, 0, 0, 0, 669, 670, 1, 0, 0, 0, 670, 672, 1, 0, 0, 0, 671, 673, 3, 102, 51, 0, 672, 671, 1, 0, 0, 0, 672, 673, 1, 0, 0, 0, 673, 675, 1, 0, 0, 0, 674, 676, 3, 104, 52, 0, 675, 674, 1, 0, 0, 0, 675, 676, 1, 0, 0, 0, 676, 99, 1, 0, 0, 0, 677, 678, 5, 67, 0, 0, 678, 679, 5, 11, 0, 0, 679, 680, 3, 114, 57, 0, 680, 101, 1, 0, 0, 0, 681, 682, 5, 64, 0, 0, 682, 683, 5, 11, 0, 0, 683, 684, 3, 88, 44, 0, 684, 103, 1, 0, 0, 0, 685, 686, 7, 8, 0, 0, 686, 687, 3, 106, 53, 0, 687, 105, 1, 0, 0, 0, 688, 695, 3, 108, 54, 0, 689, 690, 5, 9, 0, 0, 690, 691, 3, 108, 54, 0, 691, 692, 5, 2, 0, 0, 692, 693, 3, 108, 54, 0, 693, 695, 1, 0, 0, 0, 694, 688, 1, 0, 0, 0, 694, 689, 1, 0, 0, 0, 695, 107, 1, 0, 0, 0, 696, 697, 5, 19, 0, 0, 697, 709, 5, 75, 0, 0, 698, 699, 5, 94, 0, 0, 699, 709, 5, 68, 0, 0, 700, 701, 5, 94, 0, 0, 701, 709, 5, 32, 0, 0, 702, 703, 3, 144, 72, 0, 703, 704, 5, 68, 0, 0, 704, 709, 1, 0, 0, 0, 705, 706, 3, 144, 72, 0, 706, 707, 5, 32, 0, 0, 707, 709, 1, 0, 0, 0, 708, 696, 1, 0, 0, 0, 708, 698, 1, 0, 0, 0, 708, 700, 1, 0, 0, 0, 708, 702, 1, 0, 0, 0, 708, 705, 1, 0, 0, 0, 709, 109, 1, 0, 0, 0, 710, 711, 3, 116, 58, 0, 711, 712, 5, 0, 0, 1, 712, 111, 1, 0, 0, 0, 713, 770, 3, 156, 78, 0, 714, 715, 3, 156, 78, 0, 715, 716, 5, 130, 0, 0, 716, 717, 3, 156, 78, 0, 717, 724, 3, 112, 56, 0, 718, 719, 5, 116, 0, 0, 719, 720, 3, 156, 78, 0, 720, 721, 3, 112, 56, 0, 721, 723, 1, 0, 0, 0, 722, 718, 1, 0, 0, 0, 723, 726, 1, 0, 0, 0, 724, 722, 1, 0, 0, 0, 724, 725, 1, 0, 0, 0, 725, 728, 1, 0, 0, 0, 726, 724, 1, 0, 0, 0, 727, 729, 5, 116, 0, 0, 728, 727, 1, 0, 0, 0, 728, 729, 1, 0, 0, 0, 729, 730, 1, 0, 0, 0, 730, 731, 5, 149, 0, 0, 731, 770, 1, 0, 0, 0, 732, 733, 3, 156, 78, 0, 733, 734, 5, 130, 0, 0, 734, 739, 3, 158, 79, 0, 735, 736, 5, 116, 0, 0, 736, 738, 3, 158, 79, 0, 737, 735, 1, 0, 0, 0, 738, 741, 1, 0, 0, 0, 739, 737, 1, 0, 0, 0, 739, 740, 1, 0, 0, 0, 740, 743, 1, 0, 0, 0, 741, 739, 1, 0, 0, 0, 742, 744, 5, 116, 0, 0, 743, 742, 1, 0, 0, 0, 743, 744, 1, 0, 0, 0, 744, 745, 1, 0, 0, 0, 745, 746, 5, 149, 0, 0, 746, 770, 1, 0, 0, 0, 747, 748, 3, 156, 78, 0, 748, 749, 5, 130, 0, 0, 749, 754, 3, 112, 56, 0, 750, 751, 5, 116, 0, 0, 751, 753, 3, 112, 56, 0, 752, 750, 1, 0, 0, 0, 753, 756, 1, 0, 0, 0, 754, 752, 1, 0, 0, 0, 754, 755, 1, 0, 0, 0, 755, 758, 1, 0, 0, 0, 756, 754, 1, 0, 0, 0, 757, 759, 5, 116, 0, 0, 758, 757, 1, 0, 0, 0, 758, 759, 1, 0, 0, 0, 759, 760, 1, 0, 0, 0, 760, 761, 5, 149, 0, 0, 761, 770, 1, 0, 0, 0, 762, 763, 3, 156, 78, 0, 763, 765, 5, 130, 0, 0, 764, 766, 3, 114, 57, 0, 765, 764, 1, 0, 0, 0, 765, 766, 1, 0, 0, 0, 766, 767, 1, 0, 0, 0, 767, 768, 5, 149, 0, 0, 768, 770, 1, 0, 0, 0, 769, 713, 1, 0, 0, 0, 769, 714, 1, 0, 0, 0, 769, 732, 1, 0, 0, 0, 769, 747, 1, 0, 0, 0, 769, 762, 1, 0, 0, 0, 770, 113, 1, 0, 0, 0, 771, 776, 3, 116, 58, 0, 772, 773, 5, 116, 0, 0, 773, 775, 3, 116, 58, 0, 774, 772, 1, 0, 0, 0, 775, 778, 1, 0, 0, 0, 776, 774, 1, 0, 0, 0, 776, 777, 1, 0, 0, 0, 777, 780, 1, 0, 0, 0, 778, 776, 1, 0, 0, 0, 779, 781, 5, 116, 0, 0, 780, 779, 1, 0, 0, 0, 780, 781, 1, 0, 0, 0, 781, 115, 1, 0, 0, 0, 782, 783, 6, 58, -1, 0, 783, 785, 5, 12, 0, 0, 784, 786, 3, 116, 58, 0, 785, 784, 1, 0, 0, 0, 785, 786, 1, 0, 0, 0, 786, 792, 1, 0, 0, 0, 787, 788, 5, 98, 0, 0, 788, 789, 3, 116, 58, 0, 789, 790, 5, 83, 0, 0, 790, 791, 3, 116, 58, 0, 791, 793, 1, 0, 0, 0, 792, 787, 1, 0, 0, 0, 793, 794, 1, 0, 0, 0, 794, 792, 1, 0, 0, 0, 794, 795, 1, 0, 0, 0, 795, 798, 1, 0, 0, 0, 796, 797, 5, 25, 0, 0, 797, 799, 3, 116, 58, 0, 798, 796, 1, 0, 0, 0, 798, 799, 1, 0, 0, 0, 799, 800, 1, 0, 0, 0, 800, 801, 5, 26, 0, 0, 801, 933, 1, 0, 0, 0, 802, 803, 5, 13, 0, 0, 803, 804, 5, 130, 0, 0, 804, 805, 3, 116, 58, 0, 805, 806, 5, 6, 0, 0, 806, 807, 3, 112, 56, 0, 807, 808, 5, 149, 0, 0, 808, 933, 1, 0, 0, 0, 809, 810, 5, 20, 0, 0, 810, 933, 5, 110, 0, 0, 811, 812, 5, 45, 0, 0, 812, 813, 3, 116, 58, 0, 813, 814, 3, 148, 74, 0, 814, 933, 1, 0, 0, 0, 815, 816, 5, 82, 0, 0, 816, 817, 5, 130, 0, 0, 817, 818, 3, 116, 58, 0, 818, 819, 5, 34, 0, 0, 819, 822, 3, 116, 58, 0, 820, 821, 5, 33, 0, 0, 821, 823, 3, 116, 58, 0, 822, 820, 1, 0, 0, 0, 822, 823, 1, 0, 0, 0, 823, 824, 1, 0, 0, 0, 824, 825, 5, 149, 0, 0, 825, 933, 1, 0, 0, 0, 826, 827, 5, 86, 0, 0, 827, 933, 5, 110, 0, 0, 828, 829, 5, 91, 0, 0, 829, 830, 5, 130, 0, 0, 830, 831, 7, 9, 0, 0, 831, 832, 3, 162, 81, 0, 832, 833, 5, 34, 0, 0, 833, 834, 3, 116, 58, 0, 834, 835, 5, 149, 0, 0, 835, 933, 1, 0, 0, 0, 836, 837, 3, 156, 78, 0, 837, 839, 5, 130, 0, 0, 838, 840, 3, 114, 57, 0, 839, 838, 1, 0, 0, 0, 839, 840, 1, 0, 0, 0, 840, 841, 1, 0, 0, 0, 841, 842, 5, 149, 0, 0, 842, 851, 1, 0, 0, 0, 843, 845, 5, 130, 0, 0, 844, 846, 5, 24, 0, 0, 845, 844, 1, 0, 0, 0, 845, 846, 1, 0, 0, 0, 846, 848, 1, 0, 0, 0, 847, 849, 3, 114, 57, 0, 848, 847, 1, 0, 0, 0, 848, 849, 1, 0, 0, 0, 849, 850, 1, 0, 0, 0, 850, 852, 5, 149, 0, 0, 851, 843, 1, 0, 0, 0, 851, 852, 1, 0, 0, 0, 852, 853, 1, 0, 0, 0, 853, 854, 5, 66, 0, 0, 854, 855, 5, 130, 0, 0, 855, 856, 3, 98, 49, 0, 856, 857, 5, 149, 0, 0, 857, 933, 1, 0, 0, 0, 858, 859, 3, 156, 78, 0, 859, 861, 5, 130, 0, 0, 860, 862, 3, 114, 57, 0, 861, 860, 1, 0, 0, 0, 861, 862, 1, 0, 0, 0, 862, 863, 1, 0, 0, 0, 863, 864, 5, 149, 0, 0, 864, 873, 1, 0, 0, 0, 865, 867, 5, 130, 0, 0, 866, 868, 5, 24, 0, 0, 867, 866, 1, 0, 0, 0, 867, 868, 1, 0, 0, 0, 868, 870, 1, 0, 0, 0, 869, 871, 3, 114, 57, 0, 870, 869, 1, 0, 0, 0, 870, 871, 1, 0, 0, 0, 871, 872, 1, 0, 0, 0, 872, 874, 5, 149, 0, 0, 873, 865, 1, 0, 0, 0, 873, 874, 1, 0, 0, 0, 874, 875, 1, 0, 0, 0, 875, 876, 5, 66, 0, 0, 876, 877, 3, 156, 78, 0, 877, 933, 1, 0, 0, 0, 878, 884, 3, 156, 78, 0, 879, 881, 5, 130, 0, 0, 880, 882, 3, 114, 57, 0, 881, 880, 1, 0, 0, 0, 881, 882, 1, 0, 0, 0, 882, 883, 1, 0, 0, 0, 883, 885, 5, 149, 0, 0, 884, 879, 1, 0, 0, 0, 884, 885, 1, 0, 0, 0, 885, 886, 1, 0, 0, 0, 886, 888, 5, 130, 0, 0, 887, 889, 5, 24, 0, 0, 888, 887, 1, 0, 0, 0, 888, 889, 1, 0, 0, 0, 889, 891, 1, 0, 0, 0, 890, 892, 3, 114, 57, 0, 891, 890, 1, 0, 0, 0, 891, 892, 1, 0, 0, 0, 892, 893, 1, 0, 0, 0, 893, 894, 5, 149, 0, 0, 894, 933, 1, 0, 0, 0, 895, 933, 3, 120, 60, 0, 896, 933, 3, 164, 82, 0, 897, 933, 3, 146, 73, 0, 898, 899, 5, 118, 0, 0, 899, 933, 3, 116, 58, 20, 900, 901, 5, 58, 0, 0, 901, 933, 3, 116, 58, 14, 902, 903, 3, 136, 68, 0, 903, 904, 5, 120, 0, 0, 904, 906, 1, 0, 0, 0, 905, 902, 1, 0, 0, 0, 905, 906, 1, 0, 0, 0, 906, 907, 1, 0, 0, 0, 907, 933, 5, 112, 0, 0, 908, 909, 5, 130, 0, 0, 909, 910, 3, 44, 22, 0, 910, 911, 5, 149, 0, 0, 911, 933, 1, 0, 0, 0, 912, 913, 5, 130, 0, 0, 913, 914, 3, 116, 58, 0, 914, 915, 5, 149, 0, 0, 915, 933, 1, 0, 0, 0, 916, 917, 5, 130, 0, 0, 917, 918, 3, 114, 57, 0, 918, 919, 5, 149, 0, 0, 919, 933, 1, 0, 0, 0, 920, 922, 5, 129, 0, 0, 921, 923, 3, 114, 57, 0, 922, 921, 1, 0, 0, 0, 922, 923, 1, 0, 0, 0, 923, 924, 1, 0, 0, 0, 924, 933, 5, 148, 0, 0, 925, 927, 5, 128, 0, 0, 926, 928, 3, 40, 20, 0, 927, 926, 1, 0, 0, 0, 927, 928, 1, 0, 0, 0, 928, 929, 1, 0, 0, 0, 929, 933, 5, 147, 0, 0, 930, 933, 3, 118, 59, 0, 931, 933, 3, 128, 64, 0, 932, 782, 1, 0, 0, 0, 932, 802, 1, 0, 0, 0, 932, 809, 1, 0, 0, 0, 932, 811, 1, 0, 0, 0, 932, 815, 1, 0, 0, 0, 932, 826, 1, 0, 0, 0, 932, 828, 1, 0, 0, 0, 932, 836, 1, 0, 0, 0, 932, 858, 1, 0, 0, 0, 932, 878, 1, 0, 0, 0, 932, 895, 1, 0, 0, 0, 932, 896, 1, 0, 0, 0, 932, 897, 1, 0, 0, 0, 932, 898, 1, 0, 0, 0, 932, 900, 1, 0, 0, 0, 932, 905, 1, 0, 0, 0, 932, 908, 1, 0, 0, 0, 932, 912, 1, 0, 0, 0, 932, 916, 1, 0, 0, 0, 932, 920, 1, 0, 0, 0, 932, 925, 1, 0, 0, 0, 932, 930, 1, 0, 0, 0, 932, 931, 1, 0, 0, 0, 933, 1044, 1, 0, 0, 0, 934, 938, 10, 19, 0, 0, 935, 939, 5, 112, 0, 0, 936, 939, 5, 151, 0, 0, 937, 939, 5, 138, 0, 0, 938, 935, 1, 0, 0, 0, 938, 936, 1, 0, 0, 0, 938, 937, 1, 0, 0, 0, 939, 940, 1, 0, 0, 0, 940, 1043, 3, 116, 58, 20, 941, 945, 10, 18, 0, 0, 942, 946, 5, 139, 0, 0, 943, 946, 5, 118, 0, 0, 944, 946, 5, 117, 0, 0, 945, 942, 1, 0, 0, 0, 945, 943, 1, 0, 0, 0, 945, 944, 1, 0, 0, 0, 946, 947, 1, 0, 0, 0, 947, 1043, 3, 116, 58, 19, 948, 973, 10, 17, 0, 0, 949, 974, 5, 121, 0, 0, 950, 974, 5, 122, 0, 0, 951, 974, 5, 133, 0, 0, 952, 974, 5, 131, 0, 0, 953, 974, 5, 132, 0, 0, 954, 974, 5, 123, 0, 0, 955, 974, 5, 124, 0, 0, 956, 958, 5, 58, 0, 0, 957, 956, 1, 0, 0, 0, 957, 958, 1, 0, 0, 0, 958, 959, 1, 0, 0, 0, 959, 961, 5, 42, 0, 0, 960, 962, 5, 15, 0, 0, 961, 960, 1, 0, 0, 0, 961, 962, 1, 0, 0, 0, 962, 974, 1, 0, 0, 0, 963, 965, 5, 58, 0, 0, 964, 963, 1, 0, 0, 0, 964, 965, 1, 0, 0, 0, 965, 966, 1, 0, 0, 0, 966, 974, 7, 10, 0, 0, 967, 974, 5, 145, 0, 0, 968, 974, 5, 146, 0, 0, 969, 974, 5, 135, 0, 0, 970, 974, 5, 126, 0, 0, 971, 974, 5, 127, 0, 0, 972, 974, 5, 134, 0, 0, 973, 949, 1, 0, 0, 0, 973, 950, 1, 0, 0, 0, 973, 951, 1, 0, 0, 0, 973, 952, 1, 0, 0, 0, 973, 953, 1, 0, 0, 0, 973, 954, 1, 0, 0, 0, 973, 955, 1, 0, 0, 0, 973, 957, 1, 0, 0, 0, 973, 964, 1, 0, 0, 0, 973, 967, 1, 0, 0, 0, 973, 968, 1, 0, 0, 0, 973, 969, 1, 0, 0, 0, 973, 970, 1, 0, 0, 0, 973, 971, 1, 0, 0, 0, 973, 972, 1, 0, 0, 0, 974, 975, 1, 0, 0, 0, 975, 1043, 3, 116, 58, 18, 976, 977, 10, 15, 0, 0, 977, 978, 5, 137, 0, 0, 978, 1043, 3, 116, 58, 16, 979, 980, 10, 13, 0, 0, 980, 981, 5, 2, 0, 0, 981, 1043, 3, 116, 58, 14, 982, 983, 10, 12, 0, 0, 983, 984, 5, 63, 0, 0, 984, 1043, 3, 116, 58, 13, 985, 987, 10, 11, 0, 0, 986, 988, 5, 58, 0, 0, 987, 986, 1, 0, 0, 0, 987, 988, 1, 0, 0, 0, 988, 989, 1, 0, 0, 0, 989, 990, 5, 9, 0, 0, 990, 991, 3, 116, 58, 0, 991, 992, 5, 2, 0, 0, 992, 993, 3, 116, 58, 12, 993, 1043, 1, 0, 0, 0, 994, 995, 10, 10, 0, 0, 995, 996, 5, 140, 0, 0, 996, 997, 3, 116, 58, 0, 997, 998, 5, 115, 0, 0, 998, 999, 3, 116, 58, 10, 999, 1043, 1, 0, 0, 0, 1000, 1001, 10, 30, 0, 0, 1001, 1003, 5, 130, 0, 0, 1002, 1004, 3, 114, 57, 0, 1003, 1002, 1, 0, 0, 0, 1003, 1004, 1, 0, 0, 0, 1004, 1005, 1, 0, 0, 0, 1005, 1043, 5, 149, 0, 0, 1006, 1007, 10, 26, 0, 0, 1007, 1008, 5, 129, 0, 0, 1008, 1009, 3, 116, 58, 0, 1009, 1010, 5, 148, 0, 0, 1010, 1043, 1, 0, 0, 0, 1011, 1012, 10, 25, 0, 0, 1012, 1013, 5, 120, 0, 0, 1013, 1043, 5, 108, 0, 0, 1014, 1015, 10, 24, 0, 0, 1015, 1016, 5, 120, 0, 0, 1016, 1043, 3, 156, 78, 0, 1017, 1018, 10, 23, 0, 0, 1018, 1019, 5, 136, 0, 0, 1019, 1020, 5, 129, 0, 0, 1020, 1021, 3, 116, 58, 0, 1021, 1022, 5, 148, 0, 0, 1022, 1043, 1, 0, 0, 0, 1023, 1024, 10, 22, 0, 0, 1024, 1025, 5, 136, 0, 0, 1025, 1043, 5, 108, 0, 0, 1026, 1027, 10, 21, 0, 0, 1027, 1028, 5, 136, 0, 0, 1028, 1043, 3, 156, 78, 0, 1029, 1030, 10, 16, 0, 0, 1030, 1032, 5, 46, 0, 0, 1031, 1033, 5, 58, 0, 0, 1032, 1031, 1, 0, 0, 0, 1032, 1033, 1, 0, 0, 0, 1033, 1034, 1, 0, 0, 0, 1034, 1043, 5, 59, 0, 0, 1035, 1040, 10, 9, 0, 0, 1036, 1037, 5, 6, 0, 0, 1037, 1041, 3, 156, 78, 0, 1038, 1039, 5, 6, 0, 0, 1039, 1041, 5, 110, 0, 0, 1040, 1036, 1, 0, 0, 0, 1040, 1038, 1, 0, 0, 0, 1041, 1043, 1, 0, 0, 0, 1042, 934, 1, 0, 0, 0, 1042, 941, 1, 0, 0, 0, 1042, 948, 1, 0, 0, 0, 1042, 976, 1, 0, 0, 0, 1042, 979, 1, 0, 0, 0, 1042, 982, 1, 0, 0, 0, 1042, 985, 1, 0, 0, 0, 1042, 994, 1, 0, 0, 0, 1042, 1000, 1, 0, 0, 0, 1042, 1006, 1, 0, 0, 0, 1042, 1011, 1, 0, 0, 0, 1042, 1014, 1, 0, 0, 0, 1042, 1017, 1, 0, 0, 0, 1042, 1023, 1, 0, 0, 0, 1042, 1026, 1, 0, 0, 0, 1042, 1029, 1, 0, 0, 0, 1042, 1035, 1, 0, 0, 0, 1043, 1046, 1, 0, 0, 0, 1044, 1042, 1, 0, 0, 0, 1044, 1045, 1, 0, 0, 0, 1045, 117, 1, 0, 0, 0, 1046, 1044, 1, 0, 0, 0, 1047, 1048, 5, 130, 0, 0, 1048, 1053, 3, 156, 78, 0, 1049, 1050, 5, 116, 0, 0, 1050, 1052, 3, 156, 78, 0, 1051, 1049, 1, 0, 0, 0, 1052, 1055, 1, 0, 0, 0, 1053, 1051, 1, 0, 0, 0, 1053, 1054, 1, 0, 0, 0, 1054, 1057, 1, 0, 0, 0, 1055, 1053, 1, 0, 0, 0, 1056, 1058, 5, 116, 0, 0, 1057, 1056, 1, 0, 0, 0, 1057, 1058, 1, 0, 0, 0, 1058, 1059, 1, 0, 0, 0, 1059, 1060, 5, 149, 0, 0, 1060, 1075, 1, 0, 0, 0, 1061, 1066, 3, 156, 78, 0, 1062, 1063, 5, 116, 0, 0, 1063, 1065, 3, 156, 78, 0, 1064, 1062, 1, 0, 0, 0, 1065, 1068, 1, 0, 0, 0, 1066, 1064, 1, 0, 0, 0, 1066, 1067, 1, 0, 0, 0, 1067, 1070, 1, 0, 0, 0, 1068, 1066, 1, 0, 0, 0, 1069, 1071, 5, 116, 0, 0, 1070, 1069, 1, 0, 0, 0, 1070, 1071, 1, 0, 0, 0, 1071, 1075, 1, 0, 0, 0, 1072, 1073, 5, 130, 0, 0, 1073, 1075, 5, 149, 0, 0, 1074, 1047, 1, 0, 0, 0, 1074, 1061, 1, 0, 0, 0, 1074, 1072, 1, 0, 0, 0, 1075, 1076, 1, 0, 0, 0, 1076, 1079, 5, 111, 0, 0, 1077, 1080, 3, 116, 58, 0, 1078, 1080, 3, 36, 18, 0, 1079, 1077, 1, 0, 0, 0, 1079, 1078, 1, 0, 0, 0, 1080, 119, 1, 0, 0, 0, 1081, 1082, 5, 132, 0, 0, 1082, 1086, 3, 156, 78, 0, 1083, 1085, 3, 122, 61, 0, 1084, 1083, 1, 0, 0, 0, 1085, 1088, 1, 0, 0, 0, 1086, 1084, 1, 0, 0, 0, 1086, 1087, 1, 0, 0, 0, 1087, 1089, 1, 0, 0, 0, 1088, 1086, 1, 0, 0, 0, 1089, 1090, 5, 151, 0, 0, 1090, 1091, 5, 124, 0, 0, 1091, 1114, 1, 0, 0, 0, 1092, 1093, 5, 132, 0, 0, 1093, 1097, 3, 156, 78, 0, 1094, 1096, 3, 122, 61, 0, 1095, 1094, 1, 0, 0, 0, 1096, 1099, 1, 0, 0, 0, 1097, 1095, 1, 0, 0, 0, 1097, 1098, 1, 0, 0, 0, 1098, 1100, 1, 0, 0, 0, 1099, 1097, 1, 0, 0, 0, 1100, 1106, 5, 124, 0, 0, 1101, 1107, 3, 120, 60, 0, 1102, 1103, 5, 128, 0, 0, 1103, 1104, 3, 116, 58, 0, 1104, 1105, 5, 147, 0, 0, 1105, 1107, 1, 0, 0, 0, 1106, 1101, 1, 0, 0, 0, 1106, 1102, 1, 0, 0, 0, 1106, 1107, 1, 0, 0, 0, 1107, 1108, 1, 0, 0, 0, 1108, 1109, 5, 132, 0, 0, 1109, 1110, 5, 151, 0, 0, 1110, 1111, 3, 156, 78, 0, 1111, 1112, 5, 124, 0, 0, 1112, 1114, 1, 0, 0, 0, 1113, 1081, 1, 0, 0, 0, 1113, 1092, 1, 0, 0, 0, 1114, 121, 1, 0, 0, 0, 1115, 1116, 3, 156, 78, 0, 1116, 1117, 5, 122, 0, 0, 1117, 1118, 3, 162, 81, 0, 1118, 1127, 1, 0, 0, 0, 1119, 1120, 3, 156, 78, 0, 1120, 1121, 5, 122, 0, 0, 1121, 1122, 5, 128, 0, 0, 1122, 1123, 3, 116, 58, 0, 1123, 1124, 5, 147, 0, 0, 1124, 1127, 1, 0, 0, 0, 1125, 1127, 3, 156, 78, 0, 1126, 1115, 1, 0, 0, 0, 1126, 1119, 1, 0, 0, 0, 1126, 1125, 1, 0, 0, 0, 1127, 123, 1, 0, 0, 0, 1128, 1133, 3, 126, 63, 0, 1129, 1130, 5, 116, 0, 0, 1130, 1132, 3, 126, 63, 0, 1131, 1129, 1, 0, 0, 0, 1132, 1135, 1, 0, 0, 0, 1133, 1131, 1, 0, 0, 0, 1133, 1134, 1, 0, 0, 0, 1134, 1137, 1, 0, 0, 0, 1135, 1133, 1, 0, 0, 0, 1136, 1138, 5, 116, 0, 0, 1137, 1136, 1, 0, 0, 0, 1137, 1138, 1, 0, 0, 0, 1138, 125, 1, 0, 0, 0, 1139, 1140, 3, 156, 78, 0, 1140, 1141, 5, 6, 0, 0, 1141, 1142, 5, 130, 0, 0, 1142, 1143, 3, 44, 22, 0, 1143, 1144, 5, 149, 0, 0, 1144, 1150, 1, 0, 0, 0, 1145, 1146, 3, 116, 58, 0, 1146, 1147, 5, 6, 0, 0, 1147, 1148, 3, 156, 78, 0, 1148, 1150, 1, 0, 0, 0, 1149, 1139, 1, 0, 0, 0, 1149, 1145, 1, 0, 0, 0, 1150, 127, 1, 0, 0, 0, 1151, 1159, 3, 160, 80, 0, 1152, 1153, 3, 136, 68, 0, 1153, 1154, 5, 120, 0, 0, 1154, 1156, 1, 0, 0, 0, 1155, 1152, 1, 0, 0, 0, 1155, 1156, 1, 0, 0, 0, 1156, 1157, 1, 0, 0, 0, 1157, 1159, 3, 130, 65, 0, 1158, 1151, 1, 0, 0, 0, 1158, 1155, 1, 0, 0, 0, 1159, 129, 1, 0, 0, 0, 1160, 1165, 3, 156, 78, 0, 1161, 1162, 5, 120, 0, 0, 1162, 1164, 3, 156, 78, 0, 1163, 1161, 1, 0, 0, 0, 1164, 1167, 1, 0, 0, 0, 1165, 1163, 1, 0, 0, 0, 1165, 1166, 1, 0, 0, 0, 1166, 131, 1, 0, 0, 0, 1167, 1165, 1, 0, 0, 0, 1168, 1169, 6, 66, -1, 0, 1169, 1178, 3, 136, 68, 0, 1170, 1178, 3, 134, 67, 0, 1171, 1172, 5, 130, 0, 0, 1172, 1173, 3, 44, 22, 0, 1173, 1174, 5, 149, 0, 0, 1174, 1178, 1, 0, 0, 0, 1175, 1178, 3, 120, 60, 0, 1176, 1178, 3, 160, 80, 0, 1177, 1168, 1, 0, 0, 0, 1177, 1170, 1, 0, 0, 0, 1177, 1171, 1, 0, 0, 0, 1177, 1175, 1, 0, 0, 0, 1177, 1176, 1, 0, 0, 0, 1178, 1187, 1, 0, 0, 0, 1179, 1183, 10, 3, 0, 0, 1180, 1184, 3, 154, 77, 0, 1181, 1182, 5, 6, 0, 0, 1182, 1184, 3, 156, 78, 0, 1183, 1180, 1, 0, 0, 0, 1183, 1181, 1, 0, 0, 0, 1184, 1186, 1, 0, 0, 0, 1185, 1179, 1, 0, 0, 0, 1186, 1189, 1, 0, 0, 0, 1187, 1185, 1, 0, 0, 0, 1187, 1188, 1, 0, 0, 0, 1188, 133, 1, 0, 0, 0, 1189, 1187, 1, 0, 0, 0, 1190, 1191, 3, 156, 78, 0, 1191, 1193, 5, 130, 0, 0, 1192, 1194, 3, 138, 69, 0, 1193, 1192, 1, 0, 0, 0, 1193, 1194, 1, 0, 0, 0, 1194, 1195, 1, 0, 0, 0, 1195, 1196, 5, 149, 0, 0, 1196, 135, 1, 0, 0, 0, 1197, 1198, 3, 140, 70, 0, 1198, 1199, 5, 120, 0, 0, 1199, 1201, 1, 0, 0, 0, 1200, 1197, 1, 0, 0, 0, 1200, 1201, 1, 0, 0, 0, 1201, 1202, 1, 0, 0, 0, 1202, 1203, 3, 156, 78, 0, 1203, 137, 1, 0, 0, 0, 1204, 1209, 3, 116, 58, 0, 1205, 1206, 5, 116, 0, 0, 1206, 1208, 3, 116, 58, 0, 1207, 1205, 1, 0, 0, 0, 1208, 1211, 1, 0, 0, 0, 1209, 1207, 1, 0, 0, 0, 1209, 1210, 1, 0, 0, 0, 1210, 1213, 1, 0, 0, 0, 1211, 1209, 1, 0, 0, 0, 1212, 1214, 5, 116, 0, 0, 1213, 1212, 1, 0, 0, 0, 1213, 1214, 1, 0, 0, 0, 1214, 139, 1, 0, 0, 0, 1215, 1216, 3, 156, 78, 0, 1216, 141, 1, 0, 0, 0, 1217, 1226, 5, 106, 0, 0, 1218, 1219, 5, 120, 0, 0, 1219, 1226, 7, 11, 0, 0, 1220, 1221, 5, 108, 0, 0, 1221, 1223, 5, 120, 0, 0, 1222, 1224, 7, 11, 0, 0, 1223, 1222, 1, 0, 0, 0, 1223, 1224, 1, 0, 0, 0, 1224, 1226, 1, 0, 0, 0, 1225, 1217, 1, 0, 0, 0, 1225, 1218, 1, 0, 0, 0, 1225, 1220, 1, 0, 0, 0, 1226, 143, 1, 0, 0, 0, 1227, 1229, 7, 12, 0, 0, 1228, 1227, 1, 0, 0, 0, 1228, 1229, 1, 0, 0, 0, 1229, 1236, 1, 0, 0, 0, 1230, 1237, 3, 142, 71, 0, 1231, 1237, 5, 107, 0, 0, 1232, 1237, 5, 108, 0, 0, 1233, 1237, 5, 109, 0, 0, 1234, 1237, 5, 43, 0, 0, 1235, 1237, 5, 57, 0, 0, 1236, 1230, 1, 0, 0, 0, 1236, 1231, 1, 0, 0, 0, 1236, 1232, 1, 0, 0, 0, 1236, 1233, 1, 0, 0, 0, 1236, 1234, 1, 0, 0, 0, 1236, 1235, 1, 0, 0, 0, 1237, 145, 1, 0, 0, 0, 1238, 1242, 3, 144, 72, 0, 1239, 1242, 5, 110, 0, 0, 1240, 1242, 5, 59, 0, 0, 1241, 1238, 1, 0, 0, 0, 1241, 1239, 1, 0, 0, 0, 1241, 1240, 1, 0, 0, 0, 1242, 147, 1, 0, 0, 0, 1243, 1244, 7, 13, 0, 0, 1244, 149, 1, 0, 0, 0, 1245, 1246, 7, 14, 0, 0, 1246, 151, 1, 0, 0, 0, 1247, 1248, 7, 15, 0, 0, 1248, 153, 1, 0, 0, 0, 1249, 1252, 5, 105, 0, 0, 1250, 1252, 3, 152, 76, 0, 1251, 1249, 1, 0, 0, 0, 1251, 1250, 1, 0, 0, 0, 1252, 155, 1, 0, 0, 0, 1253, 1257, 5, 105, 0, 0, 1254, 1257, 3, 148, 74, 0, 1255, 1257, 3, 150, 75, 0, 1256, 1253, 1, 0, 0, 0, 1256, 1254, 1, 0, 0, 0, 1256, 1255, 1, 0, 0, 0, 1257, 157, 1, 0, 0, 0, 1258, 1259, 3, 162, 81, 0, 1259, 1260, 5, 122, 0, 0, 1260, 1261, 3, 144, 72, 0, 1261, 159, 1, 0, 0, 0, 1262, 1263, 5, 128, 0, 0, 1263, 1264, 3, 130, 65, 0, 1264, 1265, 5, 147, 0, 0, 1265, 161, 1, 0, 0, 0, 1266, 1269, 5, 110, 0, 0, 1267, 1269, 3, 164, 82, 0, 1268, 1266, 1, 0, 0, 0, 1268, 1267, 1, 0, 0, 0, 1269, 163, 1, 0, 0, 0, 1270, 1274, 5, 142, 0, 0, 1271, 1273, 3, 166, 83, 0, 1272, 1271, 1, 0, 0, 0, 1273, 1276, 1, 0, 0, 0, 1274, 1272, 1, 0, 0, 0, 1274, 1275, 1, 0, 0, 0, 1275, 1277, 1, 0, 0, 0, 1276, 1274, 1, 0, 0, 0, 1277, 1278, 5, 144, 0, 0, 1278, 165, 1, 0, 0, 0, 1279, 1280, 5, 157, 0, 0, 1280, 1281, 3, 116, 58, 0, 1281, 1282, 5, 147, 0, 0, 1282, 1285, 1, 0, 0, 0, 1283, 1285, 5, 156, 0, 0, 1284, 1279, 1, 0, 0, 0, 1284, 1283, 1, 0, 0, 0, 1285, 167, 1, 0, 0, 0, 1286, 1290, 5, 143, 0, 0, 1287, 1289, 3, 170, 85, 0, 1288, 1287, 1, 0, 0, 0, 1289, 1292, 1, 0, 0, 0, 1290, 1288, 1, 0, 0, 0, 1290, 1291, 1, 0, 0, 0, 1291, 1293, 1, 0, 0, 0, 1292, 1290, 1, 0, 0, 0, 1293, 1294, 5, 0, 0, 1, 1294, 169, 1, 0, 0, 0, 1295, 1296, 5, 159, 0, 0, 1296, 1297, 3, 116, 58, 0, 1297, 1298, 5, 147, 0, 0, 1298, 1301, 1, 0, 0, 0, 1299, 1301, 5, 158, 0, 0, 1300, 1295, 1, 0, 0, 0, 1300, 1299, 1, 0, 0, 0, 1301, 171, 1, 0, 0, 0, 167, 175, 182, 191, 198, 202, 216, 220, 223, 227, 230, 237, 241, 250, 255, 264, 272, 279, 283, 289, 294, 302, 309, 315, 327, 335, 349, 353, 358, 368, 377, 380, 384, 387, 391, 394, 397, 400, 403, 407, 411, 414, 417, 420, 424, 427, 436, 442, 463, 480, 497, 503, 509, 520, 522, 533, 536, 542, 550, 556, 558, 562, 567, 570, 573, 577, 581, 584, 586, 589, 593, 597, 600, 602, 604, 609, 620, 626, 633, 638, 642, 646, 652, 654, 661, 669, 672, 675, 694, 708, 724, 728, 739, 743, 754, 758, 765, 769, 776, 780, 785, 794, 798, 822, 839, 845, 848, 851, 861, 867, 870, 873, 881, 884, 888, 891, 905, 922, 927, 932, 938, 945, 957, 961, 964, 973, 987, 1003, 1032, 1040, 1042, 1044, 1053, 1057, 1066, 1070, 1074, 1079, 1086, 1097, 1106, 1113, 1126, 1133, 1137, 1149, 1155, 1158, 1165, 1177, 1183, 1187, 1193, 1200, 1209, 1213, 1223, 1225, 1228, 1236, 1241, 1251, 1256, 1268, 1274, 1284, 1290, 1300] \ No newline at end of file diff --git a/posthog/hogql/grammar/HogQLParser.py b/posthog/hogql/grammar/HogQLParser.py index f3a1844f404d3..e983a673610fb 100644 --- a/posthog/hogql/grammar/HogQLParser.py +++ b/posthog/hogql/grammar/HogQLParser.py @@ -1,4 +1,4 @@ -# Generated from HogQLParser.g4 by ANTLR 4.13.1 +# Generated from HogQLParser.g4 by ANTLR 4.13.2 # encoding: utf-8 from antlr4 import * from io import StringIO @@ -10,7 +10,7 @@ def serializedATN(): return [ - 4,1,159,1311,2,0,7,0,2,1,7,1,2,2,7,2,2,3,7,3,2,4,7,4,2,5,7,5,2,6, + 4,1,159,1303,2,0,7,0,2,1,7,1,2,2,7,2,2,3,7,3,2,4,7,4,2,5,7,5,2,6, 7,6,2,7,7,7,2,8,7,8,2,9,7,9,2,10,7,10,2,11,7,11,2,12,7,12,2,13,7, 13,2,14,7,14,2,15,7,15,2,16,7,16,2,17,7,17,2,18,7,18,2,19,7,19,2, 20,7,20,2,21,7,21,2,22,7,22,2,23,7,23,2,24,7,24,2,25,7,25,2,26,7, @@ -23,520 +23,519 @@ def serializedATN(): 65,2,66,7,66,2,67,7,67,2,68,7,68,2,69,7,69,2,70,7,70,2,71,7,71,2, 72,7,72,2,73,7,73,2,74,7,74,2,75,7,75,2,76,7,76,2,77,7,77,2,78,7, 78,2,79,7,79,2,80,7,80,2,81,7,81,2,82,7,82,2,83,7,83,2,84,7,84,2, - 85,7,85,2,86,7,86,2,87,7,87,1,0,5,0,178,8,0,10,0,12,0,181,9,0,1, - 0,1,0,1,1,1,1,3,1,187,8,1,1,2,1,2,1,3,1,3,1,3,1,3,1,3,3,3,196,8, - 3,1,4,1,4,1,4,5,4,201,8,4,10,4,12,4,204,9,4,1,4,3,4,207,8,4,1,5, - 1,5,1,5,1,5,1,5,1,5,1,5,1,5,1,5,1,5,1,5,1,5,3,5,221,8,5,1,6,1,6, - 3,6,225,8,6,1,6,3,6,228,8,6,1,7,1,7,3,7,232,8,7,1,7,3,7,235,8,7, - 1,8,1,8,1,8,1,8,1,8,3,8,242,8,8,1,8,1,8,3,8,246,8,8,1,8,1,8,1,9, - 1,9,1,9,5,9,253,8,9,10,9,12,9,256,9,9,1,9,1,9,3,9,260,8,9,1,10,1, - 10,1,10,1,10,1,10,1,10,1,10,3,10,269,8,10,1,11,1,11,1,11,1,11,1, - 11,1,11,3,11,277,8,11,1,12,1,12,1,12,1,12,1,12,3,12,284,8,12,1,12, - 1,12,3,12,288,8,12,1,12,1,12,1,12,1,12,3,12,294,8,12,1,12,1,12,1, - 12,3,12,299,8,12,1,13,1,13,1,13,1,13,1,13,1,13,3,13,307,8,13,1,13, - 1,13,1,13,1,13,1,13,3,13,314,8,13,1,14,1,14,1,14,1,14,3,14,320,8, - 14,1,14,1,14,1,14,1,15,1,15,1,15,1,15,1,15,1,16,1,16,3,16,332,8, - 16,1,17,1,17,1,18,1,18,5,18,338,8,18,10,18,12,18,341,9,18,1,18,1, - 18,1,19,1,19,1,19,1,19,1,20,1,20,1,20,5,20,352,8,20,10,20,12,20, - 355,9,20,1,20,3,20,358,8,20,1,21,1,21,1,21,3,21,363,8,21,1,21,1, - 21,1,22,1,22,1,22,1,22,5,22,371,8,22,10,22,12,22,374,9,22,1,23,1, - 23,1,23,1,23,1,23,1,23,3,23,382,8,23,1,24,3,24,385,8,24,1,24,1,24, - 3,24,389,8,24,1,24,3,24,392,8,24,1,24,1,24,3,24,396,8,24,1,24,3, - 24,399,8,24,1,24,3,24,402,8,24,1,24,3,24,405,8,24,1,24,3,24,408, - 8,24,1,24,1,24,3,24,412,8,24,1,24,1,24,3,24,416,8,24,1,24,3,24,419, - 8,24,1,24,3,24,422,8,24,1,24,3,24,425,8,24,1,24,1,24,3,24,429,8, - 24,1,24,3,24,432,8,24,1,25,1,25,1,25,1,26,1,26,1,26,1,26,3,26,441, - 8,26,1,27,1,27,1,27,1,28,3,28,447,8,28,1,28,1,28,1,28,1,28,1,29, - 1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,5,29, - 466,8,29,10,29,12,29,469,9,29,1,30,1,30,1,30,1,31,1,31,1,31,1,32, - 1,32,1,32,1,32,1,32,1,32,1,32,1,32,3,32,485,8,32,1,33,1,33,1,33, - 1,34,1,34,1,34,1,34,1,35,1,35,1,35,1,35,1,36,1,36,1,36,1,36,3,36, - 502,8,36,1,36,1,36,1,36,1,36,3,36,508,8,36,1,36,1,36,1,36,1,36,3, - 36,514,8,36,1,36,1,36,1,36,1,36,1,36,1,36,1,36,1,36,1,36,3,36,525, - 8,36,3,36,527,8,36,1,37,1,37,1,37,1,38,1,38,1,38,1,39,1,39,1,39, - 3,39,538,8,39,1,39,3,39,541,8,39,1,39,1,39,1,39,1,39,3,39,547,8, - 39,1,39,1,39,1,39,1,39,1,39,1,39,3,39,555,8,39,1,39,1,39,1,39,1, - 39,5,39,561,8,39,10,39,12,39,564,9,39,1,40,3,40,567,8,40,1,40,1, - 40,1,40,3,40,572,8,40,1,40,3,40,575,8,40,1,40,3,40,578,8,40,1,40, - 1,40,3,40,582,8,40,1,40,1,40,3,40,586,8,40,1,40,3,40,589,8,40,3, - 40,591,8,40,1,40,3,40,594,8,40,1,40,1,40,3,40,598,8,40,1,40,1,40, - 3,40,602,8,40,1,40,3,40,605,8,40,3,40,607,8,40,3,40,609,8,40,1,41, - 1,41,1,41,3,41,614,8,41,1,42,1,42,1,42,1,42,1,42,1,42,1,42,1,42, - 1,42,3,42,625,8,42,1,43,1,43,1,43,1,43,3,43,631,8,43,1,44,1,44,1, - 44,5,44,636,8,44,10,44,12,44,639,9,44,1,45,1,45,3,45,643,8,45,1, - 45,1,45,3,45,647,8,45,1,45,1,45,3,45,651,8,45,1,46,1,46,1,46,1,46, - 3,46,657,8,46,3,46,659,8,46,1,47,1,47,1,47,5,47,664,8,47,10,47,12, - 47,667,9,47,1,48,1,48,1,48,1,48,1,49,3,49,674,8,49,1,49,3,49,677, - 8,49,1,49,3,49,680,8,49,1,50,1,50,1,50,1,50,1,51,1,51,1,51,1,51, - 1,52,1,52,1,52,1,53,1,53,1,53,1,53,1,53,1,53,3,53,699,8,53,1,54, - 1,54,1,54,1,54,1,54,1,54,1,54,1,54,1,54,1,54,1,54,1,54,3,54,713, - 8,54,1,55,1,55,1,55,1,56,1,56,1,56,1,56,1,56,1,56,1,56,1,56,1,56, - 5,56,727,8,56,10,56,12,56,730,9,56,1,56,3,56,733,8,56,1,56,1,56, - 1,56,1,56,1,56,1,56,1,56,5,56,742,8,56,10,56,12,56,745,9,56,1,56, - 3,56,748,8,56,1,56,1,56,1,56,1,56,1,56,1,56,1,56,5,56,757,8,56,10, - 56,12,56,760,9,56,1,56,3,56,763,8,56,1,56,1,56,1,56,1,56,1,56,3, - 56,770,8,56,1,56,1,56,3,56,774,8,56,1,57,1,57,1,57,5,57,779,8,57, - 10,57,12,57,782,9,57,1,57,3,57,785,8,57,1,58,1,58,1,58,3,58,790, - 8,58,1,58,1,58,1,58,1,58,1,58,4,58,797,8,58,11,58,12,58,798,1,58, - 1,58,3,58,803,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, + 85,7,85,1,0,5,0,174,8,0,10,0,12,0,177,9,0,1,0,1,0,1,1,1,1,3,1,183, + 8,1,1,2,1,2,1,3,1,3,1,3,1,3,1,3,3,3,192,8,3,1,4,1,4,1,4,5,4,197, + 8,4,10,4,12,4,200,9,4,1,4,3,4,203,8,4,1,5,1,5,1,5,1,5,1,5,1,5,1, + 5,1,5,1,5,1,5,1,5,1,5,3,5,217,8,5,1,6,1,6,3,6,221,8,6,1,6,3,6,224, + 8,6,1,7,1,7,3,7,228,8,7,1,7,3,7,231,8,7,1,8,1,8,1,8,1,8,1,8,3,8, + 238,8,8,1,8,1,8,3,8,242,8,8,1,8,1,8,1,9,1,9,1,9,5,9,249,8,9,10,9, + 12,9,252,9,9,1,9,1,9,3,9,256,8,9,1,10,1,10,1,10,1,10,1,10,1,10,1, + 10,3,10,265,8,10,1,11,1,11,1,11,1,11,1,11,1,11,3,11,273,8,11,1,12, + 1,12,1,12,1,12,1,12,3,12,280,8,12,1,12,1,12,3,12,284,8,12,1,12,1, + 12,1,12,1,12,3,12,290,8,12,1,12,1,12,1,12,3,12,295,8,12,1,13,1,13, + 1,13,1,13,1,13,1,13,3,13,303,8,13,1,13,1,13,1,13,1,13,1,13,3,13, + 310,8,13,1,14,1,14,1,14,1,14,3,14,316,8,14,1,14,1,14,1,14,1,15,1, + 15,1,15,1,15,1,15,1,16,1,16,3,16,328,8,16,1,17,1,17,1,18,1,18,5, + 18,334,8,18,10,18,12,18,337,9,18,1,18,1,18,1,19,1,19,1,19,1,19,1, + 20,1,20,1,20,5,20,348,8,20,10,20,12,20,351,9,20,1,20,3,20,354,8, + 20,1,21,1,21,1,21,3,21,359,8,21,1,21,1,21,1,22,1,22,1,22,1,22,5, + 22,367,8,22,10,22,12,22,370,9,22,1,23,1,23,1,23,1,23,1,23,1,23,3, + 23,378,8,23,1,24,3,24,381,8,24,1,24,1,24,3,24,385,8,24,1,24,3,24, + 388,8,24,1,24,1,24,3,24,392,8,24,1,24,3,24,395,8,24,1,24,3,24,398, + 8,24,1,24,3,24,401,8,24,1,24,3,24,404,8,24,1,24,1,24,3,24,408,8, + 24,1,24,1,24,3,24,412,8,24,1,24,3,24,415,8,24,1,24,3,24,418,8,24, + 1,24,3,24,421,8,24,1,24,1,24,3,24,425,8,24,1,24,3,24,428,8,24,1, + 25,1,25,1,25,1,26,1,26,1,26,1,26,3,26,437,8,26,1,27,1,27,1,27,1, + 28,3,28,443,8,28,1,28,1,28,1,28,1,28,1,29,1,29,1,29,1,29,1,29,1, + 29,1,29,1,29,1,29,1,29,1,29,1,29,1,29,5,29,462,8,29,10,29,12,29, + 465,9,29,1,30,1,30,1,30,1,31,1,31,1,31,1,32,1,32,1,32,1,32,1,32, + 1,32,1,32,1,32,3,32,481,8,32,1,33,1,33,1,33,1,34,1,34,1,34,1,34, + 1,35,1,35,1,35,1,35,1,36,1,36,1,36,1,36,3,36,498,8,36,1,36,1,36, + 1,36,1,36,3,36,504,8,36,1,36,1,36,1,36,1,36,3,36,510,8,36,1,36,1, + 36,1,36,1,36,1,36,1,36,1,36,1,36,1,36,3,36,521,8,36,3,36,523,8,36, + 1,37,1,37,1,37,1,38,1,38,1,38,1,39,1,39,1,39,3,39,534,8,39,1,39, + 3,39,537,8,39,1,39,1,39,1,39,1,39,3,39,543,8,39,1,39,1,39,1,39,1, + 39,1,39,1,39,3,39,551,8,39,1,39,1,39,1,39,1,39,5,39,557,8,39,10, + 39,12,39,560,9,39,1,40,3,40,563,8,40,1,40,1,40,1,40,3,40,568,8,40, + 1,40,3,40,571,8,40,1,40,3,40,574,8,40,1,40,1,40,3,40,578,8,40,1, + 40,1,40,3,40,582,8,40,1,40,3,40,585,8,40,3,40,587,8,40,1,40,3,40, + 590,8,40,1,40,1,40,3,40,594,8,40,1,40,1,40,3,40,598,8,40,1,40,3, + 40,601,8,40,3,40,603,8,40,3,40,605,8,40,1,41,1,41,1,41,3,41,610, + 8,41,1,42,1,42,1,42,1,42,1,42,1,42,1,42,1,42,1,42,3,42,621,8,42, + 1,43,1,43,1,43,1,43,3,43,627,8,43,1,44,1,44,1,44,5,44,632,8,44,10, + 44,12,44,635,9,44,1,45,1,45,3,45,639,8,45,1,45,1,45,3,45,643,8,45, + 1,45,1,45,3,45,647,8,45,1,46,1,46,1,46,1,46,3,46,653,8,46,3,46,655, + 8,46,1,47,1,47,1,47,5,47,660,8,47,10,47,12,47,663,9,47,1,48,1,48, + 1,48,1,48,1,49,3,49,670,8,49,1,49,3,49,673,8,49,1,49,3,49,676,8, + 49,1,50,1,50,1,50,1,50,1,51,1,51,1,51,1,51,1,52,1,52,1,52,1,53,1, + 53,1,53,1,53,1,53,1,53,3,53,695,8,53,1,54,1,54,1,54,1,54,1,54,1, + 54,1,54,1,54,1,54,1,54,1,54,1,54,3,54,709,8,54,1,55,1,55,1,55,1, + 56,1,56,1,56,1,56,1,56,1,56,1,56,1,56,1,56,5,56,723,8,56,10,56,12, + 56,726,9,56,1,56,3,56,729,8,56,1,56,1,56,1,56,1,56,1,56,1,56,1,56, + 5,56,738,8,56,10,56,12,56,741,9,56,1,56,3,56,744,8,56,1,56,1,56, + 1,56,1,56,1,56,1,56,1,56,5,56,753,8,56,10,56,12,56,756,9,56,1,56, + 3,56,759,8,56,1,56,1,56,1,56,1,56,1,56,3,56,766,8,56,1,56,1,56,3, + 56,770,8,56,1,57,1,57,1,57,5,57,775,8,57,10,57,12,57,778,9,57,1, + 57,3,57,781,8,57,1,58,1,58,1,58,3,58,786,8,58,1,58,1,58,1,58,1,58, + 1,58,4,58,793,8,58,11,58,12,58,794,1,58,1,58,3,58,799,8,58,1,58, 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, - 3,58,827,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, - 1,58,1,58,1,58,1,58,1,58,3,58,844,8,58,1,58,1,58,1,58,1,58,3,58, - 850,8,58,1,58,3,58,853,8,58,1,58,3,58,856,8,58,1,58,1,58,1,58,1, - 58,1,58,1,58,1,58,1,58,3,58,866,8,58,1,58,1,58,1,58,1,58,3,58,872, - 8,58,1,58,3,58,875,8,58,1,58,3,58,878,8,58,1,58,1,58,1,58,1,58,1, - 58,1,58,3,58,886,8,58,1,58,3,58,889,8,58,1,58,1,58,3,58,893,8,58, - 1,58,3,58,896,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, - 1,58,1,58,1,58,3,58,910,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, - 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,927,8,58,1,58,1,58, - 1,58,3,58,932,8,58,1,58,1,58,3,58,936,8,58,1,58,1,58,1,58,1,58,3, - 58,942,8,58,1,58,1,58,1,58,1,58,1,58,3,58,949,8,58,1,58,1,58,1,58, - 1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,961,8,58,1,58,1,58,3,58, - 965,8,58,1,58,3,58,968,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3, - 58,977,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1, - 58,1,58,3,58,991,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1, - 58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1, - 58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1, - 58,1,58,1,58,3,58,1030,8,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,1038, - 8,58,5,58,1040,8,58,10,58,12,58,1043,9,58,1,59,1,59,1,59,5,59,1048, - 8,59,10,59,12,59,1051,9,59,1,59,3,59,1054,8,59,1,60,1,60,3,60,1058, - 8,60,1,61,1,61,1,61,1,61,5,61,1064,8,61,10,61,12,61,1067,9,61,1, - 61,3,61,1070,8,61,1,61,1,61,1,61,1,61,1,61,5,61,1077,8,61,10,61, - 12,61,1080,9,61,1,61,3,61,1083,8,61,3,61,1085,8,61,1,61,1,61,1,61, - 1,62,1,62,1,62,5,62,1093,8,62,10,62,12,62,1096,9,62,1,62,1,62,1, - 62,1,62,1,62,1,62,5,62,1104,8,62,10,62,12,62,1107,9,62,1,62,1,62, - 1,62,1,62,1,62,1,62,3,62,1115,8,62,1,62,1,62,1,62,1,62,1,62,3,62, - 1122,8,62,1,63,1,63,1,63,1,63,1,63,1,63,1,63,1,63,1,63,1,63,1,63, - 3,63,1135,8,63,1,64,1,64,1,64,5,64,1140,8,64,10,64,12,64,1143,9, - 64,1,64,3,64,1146,8,64,1,65,1,65,1,65,1,65,1,65,1,65,1,65,1,65,1, - 65,1,65,3,65,1158,8,65,1,66,1,66,1,66,1,66,3,66,1164,8,66,1,66,3, - 66,1167,8,66,1,67,1,67,1,67,5,67,1172,8,67,10,67,12,67,1175,9,67, - 1,68,1,68,1,68,1,68,1,68,1,68,1,68,1,68,1,68,3,68,1186,8,68,1,68, - 1,68,1,68,1,68,3,68,1192,8,68,5,68,1194,8,68,10,68,12,68,1197,9, - 68,1,69,1,69,1,69,3,69,1202,8,69,1,69,1,69,1,70,1,70,1,70,3,70,1209, - 8,70,1,70,1,70,1,71,1,71,1,71,5,71,1216,8,71,10,71,12,71,1219,9, - 71,1,71,3,71,1222,8,71,1,72,1,72,1,73,1,73,1,73,1,73,1,73,1,73,3, - 73,1232,8,73,3,73,1234,8,73,1,74,3,74,1237,8,74,1,74,1,74,1,74,1, - 74,1,74,1,74,3,74,1245,8,74,1,75,1,75,1,75,3,75,1250,8,75,1,76,1, - 76,1,77,1,77,1,78,1,78,1,79,1,79,3,79,1260,8,79,1,80,1,80,1,80,3, - 80,1265,8,80,1,81,1,81,1,81,1,81,1,82,1,82,1,82,1,82,1,83,1,83,3, - 83,1277,8,83,1,84,1,84,5,84,1281,8,84,10,84,12,84,1284,9,84,1,84, - 1,84,1,85,1,85,1,85,1,85,1,85,3,85,1293,8,85,1,86,1,86,5,86,1297, - 8,86,10,86,12,86,1300,9,86,1,86,1,86,1,87,1,87,1,87,1,87,1,87,3, - 87,1309,8,87,1,87,0,3,78,116,136,88,0,2,4,6,8,10,12,14,16,18,20, - 22,24,26,28,30,32,34,36,38,40,42,44,46,48,50,52,54,56,58,60,62,64, - 66,68,70,72,74,76,78,80,82,84,86,88,90,92,94,96,98,100,102,104,106, - 108,110,112,114,116,118,120,122,124,126,128,130,132,134,136,138, - 140,142,144,146,148,150,152,154,156,158,160,162,164,166,168,170, - 172,174,0,16,2,0,18,18,74,74,2,0,44,44,51,51,3,0,1,1,4,4,8,8,4,0, - 1,1,3,4,8,8,80,80,2,0,51,51,73,73,2,0,1,1,4,4,2,0,7,7,22,23,2,0, - 30,30,49,49,2,0,71,71,76,76,3,0,10,10,50,50,90,90,2,0,41,41,53,53, - 1,0,107,108,2,0,118,118,139,139,7,0,21,21,38,38,55,56,70,70,78,78, - 97,97,103,103,16,0,1,13,15,20,22,28,30,30,32,37,39,42,44,51,53,54, - 58,58,60,69,71,77,79,83,85,92,94,96,98,99,101,102,4,0,20,20,30,30, - 39,39,48,48,1479,0,179,1,0,0,0,2,186,1,0,0,0,4,188,1,0,0,0,6,190, - 1,0,0,0,8,197,1,0,0,0,10,220,1,0,0,0,12,222,1,0,0,0,14,229,1,0,0, - 0,16,236,1,0,0,0,18,249,1,0,0,0,20,261,1,0,0,0,22,270,1,0,0,0,24, - 278,1,0,0,0,26,300,1,0,0,0,28,315,1,0,0,0,30,324,1,0,0,0,32,329, - 1,0,0,0,34,333,1,0,0,0,36,335,1,0,0,0,38,344,1,0,0,0,40,348,1,0, - 0,0,42,362,1,0,0,0,44,366,1,0,0,0,46,381,1,0,0,0,48,384,1,0,0,0, - 50,433,1,0,0,0,52,436,1,0,0,0,54,442,1,0,0,0,56,446,1,0,0,0,58,452, - 1,0,0,0,60,470,1,0,0,0,62,473,1,0,0,0,64,476,1,0,0,0,66,486,1,0, - 0,0,68,489,1,0,0,0,70,493,1,0,0,0,72,526,1,0,0,0,74,528,1,0,0,0, - 76,531,1,0,0,0,78,546,1,0,0,0,80,608,1,0,0,0,82,613,1,0,0,0,84,624, - 1,0,0,0,86,626,1,0,0,0,88,632,1,0,0,0,90,640,1,0,0,0,92,658,1,0, - 0,0,94,660,1,0,0,0,96,668,1,0,0,0,98,673,1,0,0,0,100,681,1,0,0,0, - 102,685,1,0,0,0,104,689,1,0,0,0,106,698,1,0,0,0,108,712,1,0,0,0, - 110,714,1,0,0,0,112,773,1,0,0,0,114,775,1,0,0,0,116,935,1,0,0,0, - 118,1044,1,0,0,0,120,1057,1,0,0,0,122,1084,1,0,0,0,124,1121,1,0, - 0,0,126,1134,1,0,0,0,128,1136,1,0,0,0,130,1157,1,0,0,0,132,1166, - 1,0,0,0,134,1168,1,0,0,0,136,1185,1,0,0,0,138,1198,1,0,0,0,140,1208, - 1,0,0,0,142,1212,1,0,0,0,144,1223,1,0,0,0,146,1233,1,0,0,0,148,1236, - 1,0,0,0,150,1249,1,0,0,0,152,1251,1,0,0,0,154,1253,1,0,0,0,156,1255, - 1,0,0,0,158,1259,1,0,0,0,160,1264,1,0,0,0,162,1266,1,0,0,0,164,1270, - 1,0,0,0,166,1276,1,0,0,0,168,1278,1,0,0,0,170,1292,1,0,0,0,172,1294, - 1,0,0,0,174,1308,1,0,0,0,176,178,3,2,1,0,177,176,1,0,0,0,178,181, - 1,0,0,0,179,177,1,0,0,0,179,180,1,0,0,0,180,182,1,0,0,0,181,179, - 1,0,0,0,182,183,5,0,0,1,183,1,1,0,0,0,184,187,3,6,3,0,185,187,3, - 10,5,0,186,184,1,0,0,0,186,185,1,0,0,0,187,3,1,0,0,0,188,189,3,116, - 58,0,189,5,1,0,0,0,190,191,5,52,0,0,191,195,3,160,80,0,192,193,5, - 115,0,0,193,194,5,122,0,0,194,196,3,4,2,0,195,192,1,0,0,0,195,196, - 1,0,0,0,196,7,1,0,0,0,197,202,3,160,80,0,198,199,5,116,0,0,199,201, - 3,160,80,0,200,198,1,0,0,0,201,204,1,0,0,0,202,200,1,0,0,0,202,203, - 1,0,0,0,203,206,1,0,0,0,204,202,1,0,0,0,205,207,5,116,0,0,206,205, - 1,0,0,0,206,207,1,0,0,0,207,9,1,0,0,0,208,221,3,12,6,0,209,221,3, - 14,7,0,210,221,3,18,9,0,211,221,3,20,10,0,212,221,3,22,11,0,213, - 221,3,26,13,0,214,221,3,24,12,0,215,221,3,28,14,0,216,221,3,30,15, - 0,217,221,3,36,18,0,218,221,3,32,16,0,219,221,3,34,17,0,220,208, - 1,0,0,0,220,209,1,0,0,0,220,210,1,0,0,0,220,211,1,0,0,0,220,212, - 1,0,0,0,220,213,1,0,0,0,220,214,1,0,0,0,220,215,1,0,0,0,220,216, - 1,0,0,0,220,217,1,0,0,0,220,218,1,0,0,0,220,219,1,0,0,0,221,11,1, - 0,0,0,222,224,5,72,0,0,223,225,3,4,2,0,224,223,1,0,0,0,224,225,1, - 0,0,0,225,227,1,0,0,0,226,228,5,150,0,0,227,226,1,0,0,0,227,228, - 1,0,0,0,228,13,1,0,0,0,229,231,5,84,0,0,230,232,3,4,2,0,231,230, - 1,0,0,0,231,232,1,0,0,0,232,234,1,0,0,0,233,235,5,150,0,0,234,233, - 1,0,0,0,234,235,1,0,0,0,235,15,1,0,0,0,236,245,5,14,0,0,237,238, - 5,130,0,0,238,241,3,160,80,0,239,240,5,115,0,0,240,242,3,160,80, - 0,241,239,1,0,0,0,241,242,1,0,0,0,242,243,1,0,0,0,243,244,5,149, - 0,0,244,246,1,0,0,0,245,237,1,0,0,0,245,246,1,0,0,0,246,247,1,0, - 0,0,247,248,3,36,18,0,248,17,1,0,0,0,249,250,5,93,0,0,250,254,3, - 36,18,0,251,253,3,16,8,0,252,251,1,0,0,0,253,256,1,0,0,0,254,252, - 1,0,0,0,254,255,1,0,0,0,255,259,1,0,0,0,256,254,1,0,0,0,257,258, - 5,29,0,0,258,260,3,36,18,0,259,257,1,0,0,0,259,260,1,0,0,0,260,19, - 1,0,0,0,261,262,5,40,0,0,262,263,5,130,0,0,263,264,3,4,2,0,264,265, - 5,149,0,0,265,268,3,10,5,0,266,267,5,25,0,0,267,269,3,10,5,0,268, - 266,1,0,0,0,268,269,1,0,0,0,269,21,1,0,0,0,270,271,5,100,0,0,271, - 272,5,130,0,0,272,273,3,4,2,0,273,274,5,149,0,0,274,276,3,10,5,0, - 275,277,5,150,0,0,276,275,1,0,0,0,276,277,1,0,0,0,277,23,1,0,0,0, - 278,279,5,33,0,0,279,283,5,130,0,0,280,284,3,6,3,0,281,284,3,30, - 15,0,282,284,3,4,2,0,283,280,1,0,0,0,283,281,1,0,0,0,283,282,1,0, - 0,0,283,284,1,0,0,0,284,285,1,0,0,0,285,287,5,150,0,0,286,288,3, - 4,2,0,287,286,1,0,0,0,287,288,1,0,0,0,288,289,1,0,0,0,289,293,5, - 150,0,0,290,294,3,6,3,0,291,294,3,30,15,0,292,294,3,4,2,0,293,290, - 1,0,0,0,293,291,1,0,0,0,293,292,1,0,0,0,293,294,1,0,0,0,294,295, - 1,0,0,0,295,296,5,149,0,0,296,298,3,10,5,0,297,299,5,150,0,0,298, - 297,1,0,0,0,298,299,1,0,0,0,299,25,1,0,0,0,300,301,5,33,0,0,301, - 302,5,130,0,0,302,303,5,52,0,0,303,306,3,160,80,0,304,305,5,116, - 0,0,305,307,3,160,80,0,306,304,1,0,0,0,306,307,1,0,0,0,307,308,1, - 0,0,0,308,309,5,42,0,0,309,310,3,4,2,0,310,311,5,149,0,0,311,313, - 3,10,5,0,312,314,5,150,0,0,313,312,1,0,0,0,313,314,1,0,0,0,314,27, - 1,0,0,0,315,316,5,31,0,0,316,317,3,160,80,0,317,319,5,130,0,0,318, - 320,3,8,4,0,319,318,1,0,0,0,319,320,1,0,0,0,320,321,1,0,0,0,321, - 322,5,149,0,0,322,323,3,36,18,0,323,29,1,0,0,0,324,325,3,4,2,0,325, - 326,5,115,0,0,326,327,5,122,0,0,327,328,3,4,2,0,328,31,1,0,0,0,329, - 331,3,4,2,0,330,332,5,150,0,0,331,330,1,0,0,0,331,332,1,0,0,0,332, - 33,1,0,0,0,333,334,5,150,0,0,334,35,1,0,0,0,335,339,5,128,0,0,336, - 338,3,2,1,0,337,336,1,0,0,0,338,341,1,0,0,0,339,337,1,0,0,0,339, - 340,1,0,0,0,340,342,1,0,0,0,341,339,1,0,0,0,342,343,5,147,0,0,343, - 37,1,0,0,0,344,345,3,4,2,0,345,346,5,115,0,0,346,347,3,4,2,0,347, - 39,1,0,0,0,348,353,3,38,19,0,349,350,5,116,0,0,350,352,3,38,19,0, - 351,349,1,0,0,0,352,355,1,0,0,0,353,351,1,0,0,0,353,354,1,0,0,0, - 354,357,1,0,0,0,355,353,1,0,0,0,356,358,5,116,0,0,357,356,1,0,0, - 0,357,358,1,0,0,0,358,41,1,0,0,0,359,363,3,44,22,0,360,363,3,48, - 24,0,361,363,3,124,62,0,362,359,1,0,0,0,362,360,1,0,0,0,362,361, - 1,0,0,0,363,364,1,0,0,0,364,365,5,0,0,1,365,43,1,0,0,0,366,372,3, - 46,23,0,367,368,5,95,0,0,368,369,5,1,0,0,369,371,3,46,23,0,370,367, - 1,0,0,0,371,374,1,0,0,0,372,370,1,0,0,0,372,373,1,0,0,0,373,45,1, - 0,0,0,374,372,1,0,0,0,375,382,3,48,24,0,376,377,5,130,0,0,377,378, - 3,44,22,0,378,379,5,149,0,0,379,382,1,0,0,0,380,382,3,164,82,0,381, - 375,1,0,0,0,381,376,1,0,0,0,381,380,1,0,0,0,382,47,1,0,0,0,383,385, - 3,50,25,0,384,383,1,0,0,0,384,385,1,0,0,0,385,386,1,0,0,0,386,388, - 5,79,0,0,387,389,5,24,0,0,388,387,1,0,0,0,388,389,1,0,0,0,389,391, - 1,0,0,0,390,392,3,52,26,0,391,390,1,0,0,0,391,392,1,0,0,0,392,393, - 1,0,0,0,393,395,3,114,57,0,394,396,3,54,27,0,395,394,1,0,0,0,395, - 396,1,0,0,0,396,398,1,0,0,0,397,399,3,56,28,0,398,397,1,0,0,0,398, - 399,1,0,0,0,399,401,1,0,0,0,400,402,3,60,30,0,401,400,1,0,0,0,401, - 402,1,0,0,0,402,404,1,0,0,0,403,405,3,62,31,0,404,403,1,0,0,0,404, - 405,1,0,0,0,405,407,1,0,0,0,406,408,3,64,32,0,407,406,1,0,0,0,407, - 408,1,0,0,0,408,411,1,0,0,0,409,410,5,102,0,0,410,412,7,0,0,0,411, - 409,1,0,0,0,411,412,1,0,0,0,412,415,1,0,0,0,413,414,5,102,0,0,414, - 416,5,89,0,0,415,413,1,0,0,0,415,416,1,0,0,0,416,418,1,0,0,0,417, - 419,3,66,33,0,418,417,1,0,0,0,418,419,1,0,0,0,419,421,1,0,0,0,420, - 422,3,58,29,0,421,420,1,0,0,0,421,422,1,0,0,0,422,424,1,0,0,0,423, - 425,3,68,34,0,424,423,1,0,0,0,424,425,1,0,0,0,425,428,1,0,0,0,426, - 429,3,72,36,0,427,429,3,74,37,0,428,426,1,0,0,0,428,427,1,0,0,0, - 428,429,1,0,0,0,429,431,1,0,0,0,430,432,3,76,38,0,431,430,1,0,0, - 0,431,432,1,0,0,0,432,49,1,0,0,0,433,434,5,102,0,0,434,435,3,128, - 64,0,435,51,1,0,0,0,436,437,5,88,0,0,437,440,5,108,0,0,438,439,5, - 102,0,0,439,441,5,85,0,0,440,438,1,0,0,0,440,441,1,0,0,0,441,53, - 1,0,0,0,442,443,5,34,0,0,443,444,3,78,39,0,444,55,1,0,0,0,445,447, - 7,1,0,0,446,445,1,0,0,0,446,447,1,0,0,0,447,448,1,0,0,0,448,449, - 5,5,0,0,449,450,5,47,0,0,450,451,3,114,57,0,451,57,1,0,0,0,452,453, - 5,101,0,0,453,454,3,160,80,0,454,455,5,6,0,0,455,456,5,130,0,0,456, - 457,3,98,49,0,457,467,5,149,0,0,458,459,5,116,0,0,459,460,3,160, - 80,0,460,461,5,6,0,0,461,462,5,130,0,0,462,463,3,98,49,0,463,464, - 5,149,0,0,464,466,1,0,0,0,465,458,1,0,0,0,466,469,1,0,0,0,467,465, - 1,0,0,0,467,468,1,0,0,0,468,59,1,0,0,0,469,467,1,0,0,0,470,471,5, - 69,0,0,471,472,3,116,58,0,472,61,1,0,0,0,473,474,5,99,0,0,474,475, - 3,116,58,0,475,63,1,0,0,0,476,477,5,36,0,0,477,484,5,11,0,0,478, - 479,7,0,0,0,479,480,5,130,0,0,480,481,3,114,57,0,481,482,5,149,0, - 0,482,485,1,0,0,0,483,485,3,114,57,0,484,478,1,0,0,0,484,483,1,0, - 0,0,485,65,1,0,0,0,486,487,5,37,0,0,487,488,3,116,58,0,488,67,1, - 0,0,0,489,490,5,64,0,0,490,491,5,11,0,0,491,492,3,88,44,0,492,69, - 1,0,0,0,493,494,5,64,0,0,494,495,5,11,0,0,495,496,3,114,57,0,496, - 71,1,0,0,0,497,498,5,54,0,0,498,501,3,116,58,0,499,500,5,116,0,0, - 500,502,3,116,58,0,501,499,1,0,0,0,501,502,1,0,0,0,502,507,1,0,0, - 0,503,504,5,102,0,0,504,508,5,85,0,0,505,506,5,11,0,0,506,508,3, - 114,57,0,507,503,1,0,0,0,507,505,1,0,0,0,507,508,1,0,0,0,508,527, - 1,0,0,0,509,510,5,54,0,0,510,513,3,116,58,0,511,512,5,102,0,0,512, - 514,5,85,0,0,513,511,1,0,0,0,513,514,1,0,0,0,514,515,1,0,0,0,515, - 516,5,61,0,0,516,517,3,116,58,0,517,527,1,0,0,0,518,519,5,54,0,0, - 519,520,3,116,58,0,520,521,5,61,0,0,521,524,3,116,58,0,522,523,5, - 11,0,0,523,525,3,114,57,0,524,522,1,0,0,0,524,525,1,0,0,0,525,527, - 1,0,0,0,526,497,1,0,0,0,526,509,1,0,0,0,526,518,1,0,0,0,527,73,1, - 0,0,0,528,529,5,61,0,0,529,530,3,116,58,0,530,75,1,0,0,0,531,532, - 5,81,0,0,532,533,3,94,47,0,533,77,1,0,0,0,534,535,6,39,-1,0,535, - 537,3,136,68,0,536,538,5,28,0,0,537,536,1,0,0,0,537,538,1,0,0,0, - 538,540,1,0,0,0,539,541,3,86,43,0,540,539,1,0,0,0,540,541,1,0,0, - 0,541,547,1,0,0,0,542,543,5,130,0,0,543,544,3,78,39,0,544,545,5, - 149,0,0,545,547,1,0,0,0,546,534,1,0,0,0,546,542,1,0,0,0,547,562, - 1,0,0,0,548,549,10,3,0,0,549,550,3,82,41,0,550,551,3,78,39,4,551, - 561,1,0,0,0,552,554,10,4,0,0,553,555,3,80,40,0,554,553,1,0,0,0,554, - 555,1,0,0,0,555,556,1,0,0,0,556,557,5,47,0,0,557,558,3,78,39,0,558, - 559,3,84,42,0,559,561,1,0,0,0,560,548,1,0,0,0,560,552,1,0,0,0,561, - 564,1,0,0,0,562,560,1,0,0,0,562,563,1,0,0,0,563,79,1,0,0,0,564,562, - 1,0,0,0,565,567,7,2,0,0,566,565,1,0,0,0,566,567,1,0,0,0,567,568, - 1,0,0,0,568,575,5,44,0,0,569,571,5,44,0,0,570,572,7,2,0,0,571,570, - 1,0,0,0,571,572,1,0,0,0,572,575,1,0,0,0,573,575,7,2,0,0,574,566, - 1,0,0,0,574,569,1,0,0,0,574,573,1,0,0,0,575,609,1,0,0,0,576,578, - 7,3,0,0,577,576,1,0,0,0,577,578,1,0,0,0,578,579,1,0,0,0,579,581, - 7,4,0,0,580,582,5,65,0,0,581,580,1,0,0,0,581,582,1,0,0,0,582,591, - 1,0,0,0,583,585,7,4,0,0,584,586,5,65,0,0,585,584,1,0,0,0,585,586, - 1,0,0,0,586,588,1,0,0,0,587,589,7,3,0,0,588,587,1,0,0,0,588,589, - 1,0,0,0,589,591,1,0,0,0,590,577,1,0,0,0,590,583,1,0,0,0,591,609, - 1,0,0,0,592,594,7,5,0,0,593,592,1,0,0,0,593,594,1,0,0,0,594,595, + 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,823,8,58,1,58,1,58, + 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, + 3,58,840,8,58,1,58,1,58,1,58,1,58,3,58,846,8,58,1,58,3,58,849,8, + 58,1,58,3,58,852,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3, + 58,862,8,58,1,58,1,58,1,58,1,58,3,58,868,8,58,1,58,3,58,871,8,58, + 1,58,3,58,874,8,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,882,8,58,1, + 58,3,58,885,8,58,1,58,1,58,3,58,889,8,58,1,58,3,58,892,8,58,1,58, + 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,906, + 8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, + 1,58,1,58,1,58,3,58,923,8,58,1,58,1,58,1,58,3,58,928,8,58,1,58,1, + 58,1,58,3,58,933,8,58,1,58,1,58,1,58,1,58,3,58,939,8,58,1,58,1,58, + 1,58,1,58,1,58,3,58,946,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, + 1,58,1,58,1,58,3,58,958,8,58,1,58,1,58,3,58,962,8,58,1,58,3,58,965, + 8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,974,8,58,1,58,1,58, + 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,3,58,988,8,58, + 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, + 1,58,3,58,1004,8,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, + 1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58,1,58, + 1,58,1,58,1,58,1,58,1,58,3,58,1033,8,58,1,58,1,58,1,58,1,58,1,58, + 1,58,3,58,1041,8,58,5,58,1043,8,58,10,58,12,58,1046,9,58,1,59,1, + 59,1,59,1,59,5,59,1052,8,59,10,59,12,59,1055,9,59,1,59,3,59,1058, + 8,59,1,59,1,59,1,59,1,59,1,59,5,59,1065,8,59,10,59,12,59,1068,9, + 59,1,59,3,59,1071,8,59,1,59,1,59,3,59,1075,8,59,1,59,1,59,1,59,3, + 59,1080,8,59,1,60,1,60,1,60,5,60,1085,8,60,10,60,12,60,1088,9,60, + 1,60,1,60,1,60,1,60,1,60,1,60,5,60,1096,8,60,10,60,12,60,1099,9, + 60,1,60,1,60,1,60,1,60,1,60,1,60,3,60,1107,8,60,1,60,1,60,1,60,1, + 60,1,60,3,60,1114,8,60,1,61,1,61,1,61,1,61,1,61,1,61,1,61,1,61,1, + 61,1,61,1,61,3,61,1127,8,61,1,62,1,62,1,62,5,62,1132,8,62,10,62, + 12,62,1135,9,62,1,62,3,62,1138,8,62,1,63,1,63,1,63,1,63,1,63,1,63, + 1,63,1,63,1,63,1,63,3,63,1150,8,63,1,64,1,64,1,64,1,64,3,64,1156, + 8,64,1,64,3,64,1159,8,64,1,65,1,65,1,65,5,65,1164,8,65,10,65,12, + 65,1167,9,65,1,66,1,66,1,66,1,66,1,66,1,66,1,66,1,66,1,66,3,66,1178, + 8,66,1,66,1,66,1,66,1,66,3,66,1184,8,66,5,66,1186,8,66,10,66,12, + 66,1189,9,66,1,67,1,67,1,67,3,67,1194,8,67,1,67,1,67,1,68,1,68,1, + 68,3,68,1201,8,68,1,68,1,68,1,69,1,69,1,69,5,69,1208,8,69,10,69, + 12,69,1211,9,69,1,69,3,69,1214,8,69,1,70,1,70,1,71,1,71,1,71,1,71, + 1,71,1,71,3,71,1224,8,71,3,71,1226,8,71,1,72,3,72,1229,8,72,1,72, + 1,72,1,72,1,72,1,72,1,72,3,72,1237,8,72,1,73,1,73,1,73,3,73,1242, + 8,73,1,74,1,74,1,75,1,75,1,76,1,76,1,77,1,77,3,77,1252,8,77,1,78, + 1,78,1,78,3,78,1257,8,78,1,79,1,79,1,79,1,79,1,80,1,80,1,80,1,80, + 1,81,1,81,3,81,1269,8,81,1,82,1,82,5,82,1273,8,82,10,82,12,82,1276, + 9,82,1,82,1,82,1,83,1,83,1,83,1,83,1,83,3,83,1285,8,83,1,84,1,84, + 5,84,1289,8,84,10,84,12,84,1292,9,84,1,84,1,84,1,85,1,85,1,85,1, + 85,1,85,3,85,1301,8,85,1,85,0,3,78,116,132,86,0,2,4,6,8,10,12,14, + 16,18,20,22,24,26,28,30,32,34,36,38,40,42,44,46,48,50,52,54,56,58, + 60,62,64,66,68,70,72,74,76,78,80,82,84,86,88,90,92,94,96,98,100, + 102,104,106,108,110,112,114,116,118,120,122,124,126,128,130,132, + 134,136,138,140,142,144,146,148,150,152,154,156,158,160,162,164, + 166,168,170,0,16,2,0,18,18,74,74,2,0,44,44,51,51,3,0,1,1,4,4,8,8, + 4,0,1,1,3,4,8,8,80,80,2,0,51,51,73,73,2,0,1,1,4,4,2,0,7,7,22,23, + 2,0,30,30,49,49,2,0,71,71,76,76,3,0,10,10,50,50,90,90,2,0,41,41, + 53,53,1,0,107,108,2,0,118,118,139,139,7,0,21,21,38,38,55,56,70,70, + 78,78,97,97,103,103,16,0,1,13,15,20,22,28,30,30,32,37,39,42,44,51, + 53,54,58,58,60,69,71,77,79,83,85,92,94,96,98,99,101,102,4,0,20,20, + 30,30,39,39,48,48,1475,0,175,1,0,0,0,2,182,1,0,0,0,4,184,1,0,0,0, + 6,186,1,0,0,0,8,193,1,0,0,0,10,216,1,0,0,0,12,218,1,0,0,0,14,225, + 1,0,0,0,16,232,1,0,0,0,18,245,1,0,0,0,20,257,1,0,0,0,22,266,1,0, + 0,0,24,274,1,0,0,0,26,296,1,0,0,0,28,311,1,0,0,0,30,320,1,0,0,0, + 32,325,1,0,0,0,34,329,1,0,0,0,36,331,1,0,0,0,38,340,1,0,0,0,40,344, + 1,0,0,0,42,358,1,0,0,0,44,362,1,0,0,0,46,377,1,0,0,0,48,380,1,0, + 0,0,50,429,1,0,0,0,52,432,1,0,0,0,54,438,1,0,0,0,56,442,1,0,0,0, + 58,448,1,0,0,0,60,466,1,0,0,0,62,469,1,0,0,0,64,472,1,0,0,0,66,482, + 1,0,0,0,68,485,1,0,0,0,70,489,1,0,0,0,72,522,1,0,0,0,74,524,1,0, + 0,0,76,527,1,0,0,0,78,542,1,0,0,0,80,604,1,0,0,0,82,609,1,0,0,0, + 84,620,1,0,0,0,86,622,1,0,0,0,88,628,1,0,0,0,90,636,1,0,0,0,92,654, + 1,0,0,0,94,656,1,0,0,0,96,664,1,0,0,0,98,669,1,0,0,0,100,677,1,0, + 0,0,102,681,1,0,0,0,104,685,1,0,0,0,106,694,1,0,0,0,108,708,1,0, + 0,0,110,710,1,0,0,0,112,769,1,0,0,0,114,771,1,0,0,0,116,932,1,0, + 0,0,118,1074,1,0,0,0,120,1113,1,0,0,0,122,1126,1,0,0,0,124,1128, + 1,0,0,0,126,1149,1,0,0,0,128,1158,1,0,0,0,130,1160,1,0,0,0,132,1177, + 1,0,0,0,134,1190,1,0,0,0,136,1200,1,0,0,0,138,1204,1,0,0,0,140,1215, + 1,0,0,0,142,1225,1,0,0,0,144,1228,1,0,0,0,146,1241,1,0,0,0,148,1243, + 1,0,0,0,150,1245,1,0,0,0,152,1247,1,0,0,0,154,1251,1,0,0,0,156,1256, + 1,0,0,0,158,1258,1,0,0,0,160,1262,1,0,0,0,162,1268,1,0,0,0,164,1270, + 1,0,0,0,166,1284,1,0,0,0,168,1286,1,0,0,0,170,1300,1,0,0,0,172,174, + 3,2,1,0,173,172,1,0,0,0,174,177,1,0,0,0,175,173,1,0,0,0,175,176, + 1,0,0,0,176,178,1,0,0,0,177,175,1,0,0,0,178,179,5,0,0,1,179,1,1, + 0,0,0,180,183,3,6,3,0,181,183,3,10,5,0,182,180,1,0,0,0,182,181,1, + 0,0,0,183,3,1,0,0,0,184,185,3,116,58,0,185,5,1,0,0,0,186,187,5,52, + 0,0,187,191,3,156,78,0,188,189,5,115,0,0,189,190,5,122,0,0,190,192, + 3,4,2,0,191,188,1,0,0,0,191,192,1,0,0,0,192,7,1,0,0,0,193,198,3, + 156,78,0,194,195,5,116,0,0,195,197,3,156,78,0,196,194,1,0,0,0,197, + 200,1,0,0,0,198,196,1,0,0,0,198,199,1,0,0,0,199,202,1,0,0,0,200, + 198,1,0,0,0,201,203,5,116,0,0,202,201,1,0,0,0,202,203,1,0,0,0,203, + 9,1,0,0,0,204,217,3,12,6,0,205,217,3,14,7,0,206,217,3,18,9,0,207, + 217,3,20,10,0,208,217,3,22,11,0,209,217,3,26,13,0,210,217,3,24,12, + 0,211,217,3,28,14,0,212,217,3,30,15,0,213,217,3,36,18,0,214,217, + 3,32,16,0,215,217,3,34,17,0,216,204,1,0,0,0,216,205,1,0,0,0,216, + 206,1,0,0,0,216,207,1,0,0,0,216,208,1,0,0,0,216,209,1,0,0,0,216, + 210,1,0,0,0,216,211,1,0,0,0,216,212,1,0,0,0,216,213,1,0,0,0,216, + 214,1,0,0,0,216,215,1,0,0,0,217,11,1,0,0,0,218,220,5,72,0,0,219, + 221,3,4,2,0,220,219,1,0,0,0,220,221,1,0,0,0,221,223,1,0,0,0,222, + 224,5,150,0,0,223,222,1,0,0,0,223,224,1,0,0,0,224,13,1,0,0,0,225, + 227,5,84,0,0,226,228,3,4,2,0,227,226,1,0,0,0,227,228,1,0,0,0,228, + 230,1,0,0,0,229,231,5,150,0,0,230,229,1,0,0,0,230,231,1,0,0,0,231, + 15,1,0,0,0,232,241,5,14,0,0,233,234,5,130,0,0,234,237,3,156,78,0, + 235,236,5,115,0,0,236,238,3,156,78,0,237,235,1,0,0,0,237,238,1,0, + 0,0,238,239,1,0,0,0,239,240,5,149,0,0,240,242,1,0,0,0,241,233,1, + 0,0,0,241,242,1,0,0,0,242,243,1,0,0,0,243,244,3,36,18,0,244,17,1, + 0,0,0,245,246,5,93,0,0,246,250,3,36,18,0,247,249,3,16,8,0,248,247, + 1,0,0,0,249,252,1,0,0,0,250,248,1,0,0,0,250,251,1,0,0,0,251,255, + 1,0,0,0,252,250,1,0,0,0,253,254,5,29,0,0,254,256,3,36,18,0,255,253, + 1,0,0,0,255,256,1,0,0,0,256,19,1,0,0,0,257,258,5,40,0,0,258,259, + 5,130,0,0,259,260,3,4,2,0,260,261,5,149,0,0,261,264,3,10,5,0,262, + 263,5,25,0,0,263,265,3,10,5,0,264,262,1,0,0,0,264,265,1,0,0,0,265, + 21,1,0,0,0,266,267,5,100,0,0,267,268,5,130,0,0,268,269,3,4,2,0,269, + 270,5,149,0,0,270,272,3,10,5,0,271,273,5,150,0,0,272,271,1,0,0,0, + 272,273,1,0,0,0,273,23,1,0,0,0,274,275,5,33,0,0,275,279,5,130,0, + 0,276,280,3,6,3,0,277,280,3,30,15,0,278,280,3,4,2,0,279,276,1,0, + 0,0,279,277,1,0,0,0,279,278,1,0,0,0,279,280,1,0,0,0,280,281,1,0, + 0,0,281,283,5,150,0,0,282,284,3,4,2,0,283,282,1,0,0,0,283,284,1, + 0,0,0,284,285,1,0,0,0,285,289,5,150,0,0,286,290,3,6,3,0,287,290, + 3,30,15,0,288,290,3,4,2,0,289,286,1,0,0,0,289,287,1,0,0,0,289,288, + 1,0,0,0,289,290,1,0,0,0,290,291,1,0,0,0,291,292,5,149,0,0,292,294, + 3,10,5,0,293,295,5,150,0,0,294,293,1,0,0,0,294,295,1,0,0,0,295,25, + 1,0,0,0,296,297,5,33,0,0,297,298,5,130,0,0,298,299,5,52,0,0,299, + 302,3,156,78,0,300,301,5,116,0,0,301,303,3,156,78,0,302,300,1,0, + 0,0,302,303,1,0,0,0,303,304,1,0,0,0,304,305,5,42,0,0,305,306,3,4, + 2,0,306,307,5,149,0,0,307,309,3,10,5,0,308,310,5,150,0,0,309,308, + 1,0,0,0,309,310,1,0,0,0,310,27,1,0,0,0,311,312,5,31,0,0,312,313, + 3,156,78,0,313,315,5,130,0,0,314,316,3,8,4,0,315,314,1,0,0,0,315, + 316,1,0,0,0,316,317,1,0,0,0,317,318,5,149,0,0,318,319,3,36,18,0, + 319,29,1,0,0,0,320,321,3,4,2,0,321,322,5,115,0,0,322,323,5,122,0, + 0,323,324,3,4,2,0,324,31,1,0,0,0,325,327,3,4,2,0,326,328,5,150,0, + 0,327,326,1,0,0,0,327,328,1,0,0,0,328,33,1,0,0,0,329,330,5,150,0, + 0,330,35,1,0,0,0,331,335,5,128,0,0,332,334,3,2,1,0,333,332,1,0,0, + 0,334,337,1,0,0,0,335,333,1,0,0,0,335,336,1,0,0,0,336,338,1,0,0, + 0,337,335,1,0,0,0,338,339,5,147,0,0,339,37,1,0,0,0,340,341,3,4,2, + 0,341,342,5,115,0,0,342,343,3,4,2,0,343,39,1,0,0,0,344,349,3,38, + 19,0,345,346,5,116,0,0,346,348,3,38,19,0,347,345,1,0,0,0,348,351, + 1,0,0,0,349,347,1,0,0,0,349,350,1,0,0,0,350,353,1,0,0,0,351,349, + 1,0,0,0,352,354,5,116,0,0,353,352,1,0,0,0,353,354,1,0,0,0,354,41, + 1,0,0,0,355,359,3,44,22,0,356,359,3,48,24,0,357,359,3,120,60,0,358, + 355,1,0,0,0,358,356,1,0,0,0,358,357,1,0,0,0,359,360,1,0,0,0,360, + 361,5,0,0,1,361,43,1,0,0,0,362,368,3,46,23,0,363,364,5,95,0,0,364, + 365,5,1,0,0,365,367,3,46,23,0,366,363,1,0,0,0,367,370,1,0,0,0,368, + 366,1,0,0,0,368,369,1,0,0,0,369,45,1,0,0,0,370,368,1,0,0,0,371,378, + 3,48,24,0,372,373,5,130,0,0,373,374,3,44,22,0,374,375,5,149,0,0, + 375,378,1,0,0,0,376,378,3,160,80,0,377,371,1,0,0,0,377,372,1,0,0, + 0,377,376,1,0,0,0,378,47,1,0,0,0,379,381,3,50,25,0,380,379,1,0,0, + 0,380,381,1,0,0,0,381,382,1,0,0,0,382,384,5,79,0,0,383,385,5,24, + 0,0,384,383,1,0,0,0,384,385,1,0,0,0,385,387,1,0,0,0,386,388,3,52, + 26,0,387,386,1,0,0,0,387,388,1,0,0,0,388,389,1,0,0,0,389,391,3,114, + 57,0,390,392,3,54,27,0,391,390,1,0,0,0,391,392,1,0,0,0,392,394,1, + 0,0,0,393,395,3,56,28,0,394,393,1,0,0,0,394,395,1,0,0,0,395,397, + 1,0,0,0,396,398,3,60,30,0,397,396,1,0,0,0,397,398,1,0,0,0,398,400, + 1,0,0,0,399,401,3,62,31,0,400,399,1,0,0,0,400,401,1,0,0,0,401,403, + 1,0,0,0,402,404,3,64,32,0,403,402,1,0,0,0,403,404,1,0,0,0,404,407, + 1,0,0,0,405,406,5,102,0,0,406,408,7,0,0,0,407,405,1,0,0,0,407,408, + 1,0,0,0,408,411,1,0,0,0,409,410,5,102,0,0,410,412,5,89,0,0,411,409, + 1,0,0,0,411,412,1,0,0,0,412,414,1,0,0,0,413,415,3,66,33,0,414,413, + 1,0,0,0,414,415,1,0,0,0,415,417,1,0,0,0,416,418,3,58,29,0,417,416, + 1,0,0,0,417,418,1,0,0,0,418,420,1,0,0,0,419,421,3,68,34,0,420,419, + 1,0,0,0,420,421,1,0,0,0,421,424,1,0,0,0,422,425,3,72,36,0,423,425, + 3,74,37,0,424,422,1,0,0,0,424,423,1,0,0,0,424,425,1,0,0,0,425,427, + 1,0,0,0,426,428,3,76,38,0,427,426,1,0,0,0,427,428,1,0,0,0,428,49, + 1,0,0,0,429,430,5,102,0,0,430,431,3,124,62,0,431,51,1,0,0,0,432, + 433,5,88,0,0,433,436,5,108,0,0,434,435,5,102,0,0,435,437,5,85,0, + 0,436,434,1,0,0,0,436,437,1,0,0,0,437,53,1,0,0,0,438,439,5,34,0, + 0,439,440,3,78,39,0,440,55,1,0,0,0,441,443,7,1,0,0,442,441,1,0,0, + 0,442,443,1,0,0,0,443,444,1,0,0,0,444,445,5,5,0,0,445,446,5,47,0, + 0,446,447,3,114,57,0,447,57,1,0,0,0,448,449,5,101,0,0,449,450,3, + 156,78,0,450,451,5,6,0,0,451,452,5,130,0,0,452,453,3,98,49,0,453, + 463,5,149,0,0,454,455,5,116,0,0,455,456,3,156,78,0,456,457,5,6,0, + 0,457,458,5,130,0,0,458,459,3,98,49,0,459,460,5,149,0,0,460,462, + 1,0,0,0,461,454,1,0,0,0,462,465,1,0,0,0,463,461,1,0,0,0,463,464, + 1,0,0,0,464,59,1,0,0,0,465,463,1,0,0,0,466,467,5,69,0,0,467,468, + 3,116,58,0,468,61,1,0,0,0,469,470,5,99,0,0,470,471,3,116,58,0,471, + 63,1,0,0,0,472,473,5,36,0,0,473,480,5,11,0,0,474,475,7,0,0,0,475, + 476,5,130,0,0,476,477,3,114,57,0,477,478,5,149,0,0,478,481,1,0,0, + 0,479,481,3,114,57,0,480,474,1,0,0,0,480,479,1,0,0,0,481,65,1,0, + 0,0,482,483,5,37,0,0,483,484,3,116,58,0,484,67,1,0,0,0,485,486,5, + 64,0,0,486,487,5,11,0,0,487,488,3,88,44,0,488,69,1,0,0,0,489,490, + 5,64,0,0,490,491,5,11,0,0,491,492,3,114,57,0,492,71,1,0,0,0,493, + 494,5,54,0,0,494,497,3,116,58,0,495,496,5,116,0,0,496,498,3,116, + 58,0,497,495,1,0,0,0,497,498,1,0,0,0,498,503,1,0,0,0,499,500,5,102, + 0,0,500,504,5,85,0,0,501,502,5,11,0,0,502,504,3,114,57,0,503,499, + 1,0,0,0,503,501,1,0,0,0,503,504,1,0,0,0,504,523,1,0,0,0,505,506, + 5,54,0,0,506,509,3,116,58,0,507,508,5,102,0,0,508,510,5,85,0,0,509, + 507,1,0,0,0,509,510,1,0,0,0,510,511,1,0,0,0,511,512,5,61,0,0,512, + 513,3,116,58,0,513,523,1,0,0,0,514,515,5,54,0,0,515,516,3,116,58, + 0,516,517,5,61,0,0,517,520,3,116,58,0,518,519,5,11,0,0,519,521,3, + 114,57,0,520,518,1,0,0,0,520,521,1,0,0,0,521,523,1,0,0,0,522,493, + 1,0,0,0,522,505,1,0,0,0,522,514,1,0,0,0,523,73,1,0,0,0,524,525,5, + 61,0,0,525,526,3,116,58,0,526,75,1,0,0,0,527,528,5,81,0,0,528,529, + 3,94,47,0,529,77,1,0,0,0,530,531,6,39,-1,0,531,533,3,132,66,0,532, + 534,5,28,0,0,533,532,1,0,0,0,533,534,1,0,0,0,534,536,1,0,0,0,535, + 537,3,86,43,0,536,535,1,0,0,0,536,537,1,0,0,0,537,543,1,0,0,0,538, + 539,5,130,0,0,539,540,3,78,39,0,540,541,5,149,0,0,541,543,1,0,0, + 0,542,530,1,0,0,0,542,538,1,0,0,0,543,558,1,0,0,0,544,545,10,3,0, + 0,545,546,3,82,41,0,546,547,3,78,39,4,547,557,1,0,0,0,548,550,10, + 4,0,0,549,551,3,80,40,0,550,549,1,0,0,0,550,551,1,0,0,0,551,552, + 1,0,0,0,552,553,5,47,0,0,553,554,3,78,39,0,554,555,3,84,42,0,555, + 557,1,0,0,0,556,544,1,0,0,0,556,548,1,0,0,0,557,560,1,0,0,0,558, + 556,1,0,0,0,558,559,1,0,0,0,559,79,1,0,0,0,560,558,1,0,0,0,561,563, + 7,2,0,0,562,561,1,0,0,0,562,563,1,0,0,0,563,564,1,0,0,0,564,571, + 5,44,0,0,565,567,5,44,0,0,566,568,7,2,0,0,567,566,1,0,0,0,567,568, + 1,0,0,0,568,571,1,0,0,0,569,571,7,2,0,0,570,562,1,0,0,0,570,565, + 1,0,0,0,570,569,1,0,0,0,571,605,1,0,0,0,572,574,7,3,0,0,573,572, + 1,0,0,0,573,574,1,0,0,0,574,575,1,0,0,0,575,577,7,4,0,0,576,578, + 5,65,0,0,577,576,1,0,0,0,577,578,1,0,0,0,578,587,1,0,0,0,579,581, + 7,4,0,0,580,582,5,65,0,0,581,580,1,0,0,0,581,582,1,0,0,0,582,584, + 1,0,0,0,583,585,7,3,0,0,584,583,1,0,0,0,584,585,1,0,0,0,585,587, + 1,0,0,0,586,573,1,0,0,0,586,579,1,0,0,0,587,605,1,0,0,0,588,590, + 7,5,0,0,589,588,1,0,0,0,589,590,1,0,0,0,590,591,1,0,0,0,591,593, + 5,35,0,0,592,594,5,65,0,0,593,592,1,0,0,0,593,594,1,0,0,0,594,603, 1,0,0,0,595,597,5,35,0,0,596,598,5,65,0,0,597,596,1,0,0,0,597,598, - 1,0,0,0,598,607,1,0,0,0,599,601,5,35,0,0,600,602,5,65,0,0,601,600, - 1,0,0,0,601,602,1,0,0,0,602,604,1,0,0,0,603,605,7,5,0,0,604,603, - 1,0,0,0,604,605,1,0,0,0,605,607,1,0,0,0,606,593,1,0,0,0,606,599, - 1,0,0,0,607,609,1,0,0,0,608,574,1,0,0,0,608,590,1,0,0,0,608,606, - 1,0,0,0,609,81,1,0,0,0,610,611,5,17,0,0,611,614,5,47,0,0,612,614, - 5,116,0,0,613,610,1,0,0,0,613,612,1,0,0,0,614,83,1,0,0,0,615,616, - 5,62,0,0,616,625,3,114,57,0,617,618,5,96,0,0,618,619,5,130,0,0,619, - 620,3,114,57,0,620,621,5,149,0,0,621,625,1,0,0,0,622,623,5,96,0, - 0,623,625,3,114,57,0,624,615,1,0,0,0,624,617,1,0,0,0,624,622,1,0, - 0,0,625,85,1,0,0,0,626,627,5,77,0,0,627,630,3,92,46,0,628,629,5, - 61,0,0,629,631,3,92,46,0,630,628,1,0,0,0,630,631,1,0,0,0,631,87, - 1,0,0,0,632,637,3,90,45,0,633,634,5,116,0,0,634,636,3,90,45,0,635, - 633,1,0,0,0,636,639,1,0,0,0,637,635,1,0,0,0,637,638,1,0,0,0,638, - 89,1,0,0,0,639,637,1,0,0,0,640,642,3,116,58,0,641,643,7,6,0,0,642, - 641,1,0,0,0,642,643,1,0,0,0,643,646,1,0,0,0,644,645,5,60,0,0,645, - 647,7,7,0,0,646,644,1,0,0,0,646,647,1,0,0,0,647,650,1,0,0,0,648, - 649,5,16,0,0,649,651,5,110,0,0,650,648,1,0,0,0,650,651,1,0,0,0,651, - 91,1,0,0,0,652,659,3,164,82,0,653,656,3,148,74,0,654,655,5,151,0, - 0,655,657,3,148,74,0,656,654,1,0,0,0,656,657,1,0,0,0,657,659,1,0, - 0,0,658,652,1,0,0,0,658,653,1,0,0,0,659,93,1,0,0,0,660,665,3,96, - 48,0,661,662,5,116,0,0,662,664,3,96,48,0,663,661,1,0,0,0,664,667, - 1,0,0,0,665,663,1,0,0,0,665,666,1,0,0,0,666,95,1,0,0,0,667,665,1, - 0,0,0,668,669,3,160,80,0,669,670,5,122,0,0,670,671,3,150,75,0,671, - 97,1,0,0,0,672,674,3,100,50,0,673,672,1,0,0,0,673,674,1,0,0,0,674, - 676,1,0,0,0,675,677,3,102,51,0,676,675,1,0,0,0,676,677,1,0,0,0,677, - 679,1,0,0,0,678,680,3,104,52,0,679,678,1,0,0,0,679,680,1,0,0,0,680, - 99,1,0,0,0,681,682,5,67,0,0,682,683,5,11,0,0,683,684,3,114,57,0, - 684,101,1,0,0,0,685,686,5,64,0,0,686,687,5,11,0,0,687,688,3,88,44, - 0,688,103,1,0,0,0,689,690,7,8,0,0,690,691,3,106,53,0,691,105,1,0, - 0,0,692,699,3,108,54,0,693,694,5,9,0,0,694,695,3,108,54,0,695,696, - 5,2,0,0,696,697,3,108,54,0,697,699,1,0,0,0,698,692,1,0,0,0,698,693, - 1,0,0,0,699,107,1,0,0,0,700,701,5,19,0,0,701,713,5,75,0,0,702,703, - 5,94,0,0,703,713,5,68,0,0,704,705,5,94,0,0,705,713,5,32,0,0,706, - 707,3,148,74,0,707,708,5,68,0,0,708,713,1,0,0,0,709,710,3,148,74, - 0,710,711,5,32,0,0,711,713,1,0,0,0,712,700,1,0,0,0,712,702,1,0,0, - 0,712,704,1,0,0,0,712,706,1,0,0,0,712,709,1,0,0,0,713,109,1,0,0, - 0,714,715,3,116,58,0,715,716,5,0,0,1,716,111,1,0,0,0,717,774,3,160, - 80,0,718,719,3,160,80,0,719,720,5,130,0,0,720,721,3,160,80,0,721, - 728,3,112,56,0,722,723,5,116,0,0,723,724,3,160,80,0,724,725,3,112, - 56,0,725,727,1,0,0,0,726,722,1,0,0,0,727,730,1,0,0,0,728,726,1,0, - 0,0,728,729,1,0,0,0,729,732,1,0,0,0,730,728,1,0,0,0,731,733,5,116, - 0,0,732,731,1,0,0,0,732,733,1,0,0,0,733,734,1,0,0,0,734,735,5,149, - 0,0,735,774,1,0,0,0,736,737,3,160,80,0,737,738,5,130,0,0,738,743, - 3,162,81,0,739,740,5,116,0,0,740,742,3,162,81,0,741,739,1,0,0,0, - 742,745,1,0,0,0,743,741,1,0,0,0,743,744,1,0,0,0,744,747,1,0,0,0, - 745,743,1,0,0,0,746,748,5,116,0,0,747,746,1,0,0,0,747,748,1,0,0, - 0,748,749,1,0,0,0,749,750,5,149,0,0,750,774,1,0,0,0,751,752,3,160, - 80,0,752,753,5,130,0,0,753,758,3,112,56,0,754,755,5,116,0,0,755, - 757,3,112,56,0,756,754,1,0,0,0,757,760,1,0,0,0,758,756,1,0,0,0,758, - 759,1,0,0,0,759,762,1,0,0,0,760,758,1,0,0,0,761,763,5,116,0,0,762, - 761,1,0,0,0,762,763,1,0,0,0,763,764,1,0,0,0,764,765,5,149,0,0,765, - 774,1,0,0,0,766,767,3,160,80,0,767,769,5,130,0,0,768,770,3,114,57, - 0,769,768,1,0,0,0,769,770,1,0,0,0,770,771,1,0,0,0,771,772,5,149, - 0,0,772,774,1,0,0,0,773,717,1,0,0,0,773,718,1,0,0,0,773,736,1,0, - 0,0,773,751,1,0,0,0,773,766,1,0,0,0,774,113,1,0,0,0,775,780,3,116, - 58,0,776,777,5,116,0,0,777,779,3,116,58,0,778,776,1,0,0,0,779,782, - 1,0,0,0,780,778,1,0,0,0,780,781,1,0,0,0,781,784,1,0,0,0,782,780, - 1,0,0,0,783,785,5,116,0,0,784,783,1,0,0,0,784,785,1,0,0,0,785,115, - 1,0,0,0,786,787,6,58,-1,0,787,789,5,12,0,0,788,790,3,116,58,0,789, - 788,1,0,0,0,789,790,1,0,0,0,790,796,1,0,0,0,791,792,5,98,0,0,792, - 793,3,116,58,0,793,794,5,83,0,0,794,795,3,116,58,0,795,797,1,0,0, - 0,796,791,1,0,0,0,797,798,1,0,0,0,798,796,1,0,0,0,798,799,1,0,0, - 0,799,802,1,0,0,0,800,801,5,25,0,0,801,803,3,116,58,0,802,800,1, - 0,0,0,802,803,1,0,0,0,803,804,1,0,0,0,804,805,5,26,0,0,805,936,1, - 0,0,0,806,807,5,13,0,0,807,808,5,130,0,0,808,809,3,116,58,0,809, - 810,5,6,0,0,810,811,3,112,56,0,811,812,5,149,0,0,812,936,1,0,0,0, - 813,814,5,20,0,0,814,936,5,110,0,0,815,816,5,45,0,0,816,817,3,116, - 58,0,817,818,3,152,76,0,818,936,1,0,0,0,819,820,5,82,0,0,820,821, - 5,130,0,0,821,822,3,116,58,0,822,823,5,34,0,0,823,826,3,116,58,0, - 824,825,5,33,0,0,825,827,3,116,58,0,826,824,1,0,0,0,826,827,1,0, - 0,0,827,828,1,0,0,0,828,829,5,149,0,0,829,936,1,0,0,0,830,831,5, - 86,0,0,831,936,5,110,0,0,832,833,5,91,0,0,833,834,5,130,0,0,834, - 835,7,9,0,0,835,836,3,166,83,0,836,837,5,34,0,0,837,838,3,116,58, - 0,838,839,5,149,0,0,839,936,1,0,0,0,840,841,3,160,80,0,841,843,5, - 130,0,0,842,844,3,114,57,0,843,842,1,0,0,0,843,844,1,0,0,0,844,845, - 1,0,0,0,845,846,5,149,0,0,846,855,1,0,0,0,847,849,5,130,0,0,848, - 850,5,24,0,0,849,848,1,0,0,0,849,850,1,0,0,0,850,852,1,0,0,0,851, - 853,3,118,59,0,852,851,1,0,0,0,852,853,1,0,0,0,853,854,1,0,0,0,854, - 856,5,149,0,0,855,847,1,0,0,0,855,856,1,0,0,0,856,857,1,0,0,0,857, - 858,5,66,0,0,858,859,5,130,0,0,859,860,3,98,49,0,860,861,5,149,0, - 0,861,936,1,0,0,0,862,863,3,160,80,0,863,865,5,130,0,0,864,866,3, - 114,57,0,865,864,1,0,0,0,865,866,1,0,0,0,866,867,1,0,0,0,867,868, - 5,149,0,0,868,877,1,0,0,0,869,871,5,130,0,0,870,872,5,24,0,0,871, - 870,1,0,0,0,871,872,1,0,0,0,872,874,1,0,0,0,873,875,3,118,59,0,874, - 873,1,0,0,0,874,875,1,0,0,0,875,876,1,0,0,0,876,878,5,149,0,0,877, - 869,1,0,0,0,877,878,1,0,0,0,878,879,1,0,0,0,879,880,5,66,0,0,880, - 881,3,160,80,0,881,936,1,0,0,0,882,888,3,160,80,0,883,885,5,130, - 0,0,884,886,3,114,57,0,885,884,1,0,0,0,885,886,1,0,0,0,886,887,1, - 0,0,0,887,889,5,149,0,0,888,883,1,0,0,0,888,889,1,0,0,0,889,890, - 1,0,0,0,890,892,5,130,0,0,891,893,5,24,0,0,892,891,1,0,0,0,892,893, - 1,0,0,0,893,895,1,0,0,0,894,896,3,118,59,0,895,894,1,0,0,0,895,896, - 1,0,0,0,896,897,1,0,0,0,897,898,5,149,0,0,898,936,1,0,0,0,899,936, - 3,124,62,0,900,936,3,168,84,0,901,936,3,150,75,0,902,903,5,118,0, - 0,903,936,3,116,58,19,904,905,5,58,0,0,905,936,3,116,58,13,906,907, - 3,140,70,0,907,908,5,120,0,0,908,910,1,0,0,0,909,906,1,0,0,0,909, - 910,1,0,0,0,910,911,1,0,0,0,911,936,5,112,0,0,912,913,5,130,0,0, - 913,914,3,44,22,0,914,915,5,149,0,0,915,936,1,0,0,0,916,917,5,130, - 0,0,917,918,3,116,58,0,918,919,5,149,0,0,919,936,1,0,0,0,920,921, - 5,130,0,0,921,922,3,114,57,0,922,923,5,149,0,0,923,936,1,0,0,0,924, - 926,5,129,0,0,925,927,3,114,57,0,926,925,1,0,0,0,926,927,1,0,0,0, - 927,928,1,0,0,0,928,936,5,148,0,0,929,931,5,128,0,0,930,932,3,40, - 20,0,931,930,1,0,0,0,931,932,1,0,0,0,932,933,1,0,0,0,933,936,5,147, - 0,0,934,936,3,132,66,0,935,786,1,0,0,0,935,806,1,0,0,0,935,813,1, - 0,0,0,935,815,1,0,0,0,935,819,1,0,0,0,935,830,1,0,0,0,935,832,1, - 0,0,0,935,840,1,0,0,0,935,862,1,0,0,0,935,882,1,0,0,0,935,899,1, - 0,0,0,935,900,1,0,0,0,935,901,1,0,0,0,935,902,1,0,0,0,935,904,1, - 0,0,0,935,909,1,0,0,0,935,912,1,0,0,0,935,916,1,0,0,0,935,920,1, - 0,0,0,935,924,1,0,0,0,935,929,1,0,0,0,935,934,1,0,0,0,936,1041,1, - 0,0,0,937,941,10,18,0,0,938,942,5,112,0,0,939,942,5,151,0,0,940, - 942,5,138,0,0,941,938,1,0,0,0,941,939,1,0,0,0,941,940,1,0,0,0,942, - 943,1,0,0,0,943,1040,3,116,58,19,944,948,10,17,0,0,945,949,5,139, - 0,0,946,949,5,118,0,0,947,949,5,117,0,0,948,945,1,0,0,0,948,946, - 1,0,0,0,948,947,1,0,0,0,949,950,1,0,0,0,950,1040,3,116,58,18,951, - 976,10,16,0,0,952,977,5,121,0,0,953,977,5,122,0,0,954,977,5,133, - 0,0,955,977,5,131,0,0,956,977,5,132,0,0,957,977,5,123,0,0,958,977, - 5,124,0,0,959,961,5,58,0,0,960,959,1,0,0,0,960,961,1,0,0,0,961,962, - 1,0,0,0,962,964,5,42,0,0,963,965,5,15,0,0,964,963,1,0,0,0,964,965, - 1,0,0,0,965,977,1,0,0,0,966,968,5,58,0,0,967,966,1,0,0,0,967,968, - 1,0,0,0,968,969,1,0,0,0,969,977,7,10,0,0,970,977,5,145,0,0,971,977, - 5,146,0,0,972,977,5,135,0,0,973,977,5,126,0,0,974,977,5,127,0,0, - 975,977,5,134,0,0,976,952,1,0,0,0,976,953,1,0,0,0,976,954,1,0,0, - 0,976,955,1,0,0,0,976,956,1,0,0,0,976,957,1,0,0,0,976,958,1,0,0, - 0,976,960,1,0,0,0,976,967,1,0,0,0,976,970,1,0,0,0,976,971,1,0,0, - 0,976,972,1,0,0,0,976,973,1,0,0,0,976,974,1,0,0,0,976,975,1,0,0, - 0,977,978,1,0,0,0,978,1040,3,116,58,17,979,980,10,14,0,0,980,981, - 5,137,0,0,981,1040,3,116,58,15,982,983,10,12,0,0,983,984,5,2,0,0, - 984,1040,3,116,58,13,985,986,10,11,0,0,986,987,5,63,0,0,987,1040, - 3,116,58,12,988,990,10,10,0,0,989,991,5,58,0,0,990,989,1,0,0,0,990, - 991,1,0,0,0,991,992,1,0,0,0,992,993,5,9,0,0,993,994,3,116,58,0,994, - 995,5,2,0,0,995,996,3,116,58,11,996,1040,1,0,0,0,997,998,10,9,0, - 0,998,999,5,140,0,0,999,1000,3,116,58,0,1000,1001,5,115,0,0,1001, - 1002,3,116,58,9,1002,1040,1,0,0,0,1003,1004,10,25,0,0,1004,1005, - 5,129,0,0,1005,1006,3,116,58,0,1006,1007,5,148,0,0,1007,1040,1,0, - 0,0,1008,1009,10,24,0,0,1009,1010,5,120,0,0,1010,1040,5,108,0,0, - 1011,1012,10,23,0,0,1012,1013,5,120,0,0,1013,1040,3,160,80,0,1014, - 1015,10,22,0,0,1015,1016,5,136,0,0,1016,1017,5,129,0,0,1017,1018, - 3,116,58,0,1018,1019,5,148,0,0,1019,1040,1,0,0,0,1020,1021,10,21, - 0,0,1021,1022,5,136,0,0,1022,1040,5,108,0,0,1023,1024,10,20,0,0, - 1024,1025,5,136,0,0,1025,1040,3,160,80,0,1026,1027,10,15,0,0,1027, - 1029,5,46,0,0,1028,1030,5,58,0,0,1029,1028,1,0,0,0,1029,1030,1,0, - 0,0,1030,1031,1,0,0,0,1031,1040,5,59,0,0,1032,1037,10,8,0,0,1033, - 1034,5,6,0,0,1034,1038,3,160,80,0,1035,1036,5,6,0,0,1036,1038,5, - 110,0,0,1037,1033,1,0,0,0,1037,1035,1,0,0,0,1038,1040,1,0,0,0,1039, - 937,1,0,0,0,1039,944,1,0,0,0,1039,951,1,0,0,0,1039,979,1,0,0,0,1039, - 982,1,0,0,0,1039,985,1,0,0,0,1039,988,1,0,0,0,1039,997,1,0,0,0,1039, - 1003,1,0,0,0,1039,1008,1,0,0,0,1039,1011,1,0,0,0,1039,1014,1,0,0, - 0,1039,1020,1,0,0,0,1039,1023,1,0,0,0,1039,1026,1,0,0,0,1039,1032, - 1,0,0,0,1040,1043,1,0,0,0,1041,1039,1,0,0,0,1041,1042,1,0,0,0,1042, - 117,1,0,0,0,1043,1041,1,0,0,0,1044,1049,3,120,60,0,1045,1046,5,116, - 0,0,1046,1048,3,120,60,0,1047,1045,1,0,0,0,1048,1051,1,0,0,0,1049, - 1047,1,0,0,0,1049,1050,1,0,0,0,1050,1053,1,0,0,0,1051,1049,1,0,0, - 0,1052,1054,5,116,0,0,1053,1052,1,0,0,0,1053,1054,1,0,0,0,1054,119, - 1,0,0,0,1055,1058,3,122,61,0,1056,1058,3,116,58,0,1057,1055,1,0, - 0,0,1057,1056,1,0,0,0,1058,121,1,0,0,0,1059,1060,5,130,0,0,1060, - 1065,3,160,80,0,1061,1062,5,116,0,0,1062,1064,3,160,80,0,1063,1061, - 1,0,0,0,1064,1067,1,0,0,0,1065,1063,1,0,0,0,1065,1066,1,0,0,0,1066, - 1069,1,0,0,0,1067,1065,1,0,0,0,1068,1070,5,116,0,0,1069,1068,1,0, - 0,0,1069,1070,1,0,0,0,1070,1071,1,0,0,0,1071,1072,5,149,0,0,1072, - 1085,1,0,0,0,1073,1078,3,160,80,0,1074,1075,5,116,0,0,1075,1077, - 3,160,80,0,1076,1074,1,0,0,0,1077,1080,1,0,0,0,1078,1076,1,0,0,0, - 1078,1079,1,0,0,0,1079,1082,1,0,0,0,1080,1078,1,0,0,0,1081,1083, - 5,116,0,0,1082,1081,1,0,0,0,1082,1083,1,0,0,0,1083,1085,1,0,0,0, - 1084,1059,1,0,0,0,1084,1073,1,0,0,0,1085,1086,1,0,0,0,1086,1087, - 5,111,0,0,1087,1088,3,116,58,0,1088,123,1,0,0,0,1089,1090,5,132, - 0,0,1090,1094,3,160,80,0,1091,1093,3,126,63,0,1092,1091,1,0,0,0, - 1093,1096,1,0,0,0,1094,1092,1,0,0,0,1094,1095,1,0,0,0,1095,1097, - 1,0,0,0,1096,1094,1,0,0,0,1097,1098,5,151,0,0,1098,1099,5,124,0, - 0,1099,1122,1,0,0,0,1100,1101,5,132,0,0,1101,1105,3,160,80,0,1102, - 1104,3,126,63,0,1103,1102,1,0,0,0,1104,1107,1,0,0,0,1105,1103,1, - 0,0,0,1105,1106,1,0,0,0,1106,1108,1,0,0,0,1107,1105,1,0,0,0,1108, - 1114,5,124,0,0,1109,1115,3,124,62,0,1110,1111,5,128,0,0,1111,1112, - 3,116,58,0,1112,1113,5,147,0,0,1113,1115,1,0,0,0,1114,1109,1,0,0, - 0,1114,1110,1,0,0,0,1114,1115,1,0,0,0,1115,1116,1,0,0,0,1116,1117, - 5,132,0,0,1117,1118,5,151,0,0,1118,1119,3,160,80,0,1119,1120,5,124, - 0,0,1120,1122,1,0,0,0,1121,1089,1,0,0,0,1121,1100,1,0,0,0,1122,125, - 1,0,0,0,1123,1124,3,160,80,0,1124,1125,5,122,0,0,1125,1126,3,166, - 83,0,1126,1135,1,0,0,0,1127,1128,3,160,80,0,1128,1129,5,122,0,0, - 1129,1130,5,128,0,0,1130,1131,3,116,58,0,1131,1132,5,147,0,0,1132, - 1135,1,0,0,0,1133,1135,3,160,80,0,1134,1123,1,0,0,0,1134,1127,1, - 0,0,0,1134,1133,1,0,0,0,1135,127,1,0,0,0,1136,1141,3,130,65,0,1137, - 1138,5,116,0,0,1138,1140,3,130,65,0,1139,1137,1,0,0,0,1140,1143, - 1,0,0,0,1141,1139,1,0,0,0,1141,1142,1,0,0,0,1142,1145,1,0,0,0,1143, - 1141,1,0,0,0,1144,1146,5,116,0,0,1145,1144,1,0,0,0,1145,1146,1,0, - 0,0,1146,129,1,0,0,0,1147,1148,3,160,80,0,1148,1149,5,6,0,0,1149, - 1150,5,130,0,0,1150,1151,3,44,22,0,1151,1152,5,149,0,0,1152,1158, - 1,0,0,0,1153,1154,3,116,58,0,1154,1155,5,6,0,0,1155,1156,3,160,80, - 0,1156,1158,1,0,0,0,1157,1147,1,0,0,0,1157,1153,1,0,0,0,1158,131, - 1,0,0,0,1159,1167,3,164,82,0,1160,1161,3,140,70,0,1161,1162,5,120, - 0,0,1162,1164,1,0,0,0,1163,1160,1,0,0,0,1163,1164,1,0,0,0,1164,1165, - 1,0,0,0,1165,1167,3,134,67,0,1166,1159,1,0,0,0,1166,1163,1,0,0,0, - 1167,133,1,0,0,0,1168,1173,3,160,80,0,1169,1170,5,120,0,0,1170,1172, - 3,160,80,0,1171,1169,1,0,0,0,1172,1175,1,0,0,0,1173,1171,1,0,0,0, - 1173,1174,1,0,0,0,1174,135,1,0,0,0,1175,1173,1,0,0,0,1176,1177,6, - 68,-1,0,1177,1186,3,140,70,0,1178,1186,3,138,69,0,1179,1180,5,130, - 0,0,1180,1181,3,44,22,0,1181,1182,5,149,0,0,1182,1186,1,0,0,0,1183, - 1186,3,124,62,0,1184,1186,3,164,82,0,1185,1176,1,0,0,0,1185,1178, - 1,0,0,0,1185,1179,1,0,0,0,1185,1183,1,0,0,0,1185,1184,1,0,0,0,1186, - 1195,1,0,0,0,1187,1191,10,3,0,0,1188,1192,3,158,79,0,1189,1190,5, - 6,0,0,1190,1192,3,160,80,0,1191,1188,1,0,0,0,1191,1189,1,0,0,0,1192, - 1194,1,0,0,0,1193,1187,1,0,0,0,1194,1197,1,0,0,0,1195,1193,1,0,0, - 0,1195,1196,1,0,0,0,1196,137,1,0,0,0,1197,1195,1,0,0,0,1198,1199, - 3,160,80,0,1199,1201,5,130,0,0,1200,1202,3,142,71,0,1201,1200,1, - 0,0,0,1201,1202,1,0,0,0,1202,1203,1,0,0,0,1203,1204,5,149,0,0,1204, - 139,1,0,0,0,1205,1206,3,144,72,0,1206,1207,5,120,0,0,1207,1209,1, - 0,0,0,1208,1205,1,0,0,0,1208,1209,1,0,0,0,1209,1210,1,0,0,0,1210, - 1211,3,160,80,0,1211,141,1,0,0,0,1212,1217,3,116,58,0,1213,1214, - 5,116,0,0,1214,1216,3,116,58,0,1215,1213,1,0,0,0,1216,1219,1,0,0, - 0,1217,1215,1,0,0,0,1217,1218,1,0,0,0,1218,1221,1,0,0,0,1219,1217, - 1,0,0,0,1220,1222,5,116,0,0,1221,1220,1,0,0,0,1221,1222,1,0,0,0, - 1222,143,1,0,0,0,1223,1224,3,160,80,0,1224,145,1,0,0,0,1225,1234, - 5,106,0,0,1226,1227,5,120,0,0,1227,1234,7,11,0,0,1228,1229,5,108, - 0,0,1229,1231,5,120,0,0,1230,1232,7,11,0,0,1231,1230,1,0,0,0,1231, - 1232,1,0,0,0,1232,1234,1,0,0,0,1233,1225,1,0,0,0,1233,1226,1,0,0, - 0,1233,1228,1,0,0,0,1234,147,1,0,0,0,1235,1237,7,12,0,0,1236,1235, - 1,0,0,0,1236,1237,1,0,0,0,1237,1244,1,0,0,0,1238,1245,3,146,73,0, - 1239,1245,5,107,0,0,1240,1245,5,108,0,0,1241,1245,5,109,0,0,1242, - 1245,5,43,0,0,1243,1245,5,57,0,0,1244,1238,1,0,0,0,1244,1239,1,0, - 0,0,1244,1240,1,0,0,0,1244,1241,1,0,0,0,1244,1242,1,0,0,0,1244,1243, - 1,0,0,0,1245,149,1,0,0,0,1246,1250,3,148,74,0,1247,1250,5,110,0, - 0,1248,1250,5,59,0,0,1249,1246,1,0,0,0,1249,1247,1,0,0,0,1249,1248, - 1,0,0,0,1250,151,1,0,0,0,1251,1252,7,13,0,0,1252,153,1,0,0,0,1253, - 1254,7,14,0,0,1254,155,1,0,0,0,1255,1256,7,15,0,0,1256,157,1,0,0, - 0,1257,1260,5,105,0,0,1258,1260,3,156,78,0,1259,1257,1,0,0,0,1259, - 1258,1,0,0,0,1260,159,1,0,0,0,1261,1265,5,105,0,0,1262,1265,3,152, - 76,0,1263,1265,3,154,77,0,1264,1261,1,0,0,0,1264,1262,1,0,0,0,1264, - 1263,1,0,0,0,1265,161,1,0,0,0,1266,1267,3,166,83,0,1267,1268,5,122, - 0,0,1268,1269,3,148,74,0,1269,163,1,0,0,0,1270,1271,5,128,0,0,1271, - 1272,3,134,67,0,1272,1273,5,147,0,0,1273,165,1,0,0,0,1274,1277,5, - 110,0,0,1275,1277,3,168,84,0,1276,1274,1,0,0,0,1276,1275,1,0,0,0, - 1277,167,1,0,0,0,1278,1282,5,142,0,0,1279,1281,3,170,85,0,1280,1279, - 1,0,0,0,1281,1284,1,0,0,0,1282,1280,1,0,0,0,1282,1283,1,0,0,0,1283, - 1285,1,0,0,0,1284,1282,1,0,0,0,1285,1286,5,144,0,0,1286,169,1,0, - 0,0,1287,1288,5,157,0,0,1288,1289,3,116,58,0,1289,1290,5,147,0,0, - 1290,1293,1,0,0,0,1291,1293,5,156,0,0,1292,1287,1,0,0,0,1292,1291, - 1,0,0,0,1293,171,1,0,0,0,1294,1298,5,143,0,0,1295,1297,3,174,87, - 0,1296,1295,1,0,0,0,1297,1300,1,0,0,0,1298,1296,1,0,0,0,1298,1299, - 1,0,0,0,1299,1301,1,0,0,0,1300,1298,1,0,0,0,1301,1302,5,0,0,1,1302, - 173,1,0,0,0,1303,1304,5,159,0,0,1304,1305,3,116,58,0,1305,1306,5, - 147,0,0,1306,1309,1,0,0,0,1307,1309,5,158,0,0,1308,1303,1,0,0,0, - 1308,1307,1,0,0,0,1309,175,1,0,0,0,168,179,186,195,202,206,220,224, - 227,231,234,241,245,254,259,268,276,283,287,293,298,306,313,319, - 331,339,353,357,362,372,381,384,388,391,395,398,401,404,407,411, - 415,418,421,424,428,431,440,446,467,484,501,507,513,524,526,537, - 540,546,554,560,562,566,571,574,577,581,585,588,590,593,597,601, - 604,606,608,613,624,630,637,642,646,650,656,658,665,673,676,679, - 698,712,728,732,743,747,758,762,769,773,780,784,789,798,802,826, - 843,849,852,855,865,871,874,877,885,888,892,895,909,926,931,935, - 941,948,960,964,967,976,990,1029,1037,1039,1041,1049,1053,1057,1065, - 1069,1078,1082,1084,1094,1105,1114,1121,1134,1141,1145,1157,1163, - 1166,1173,1185,1191,1195,1201,1208,1217,1221,1231,1233,1236,1244, - 1249,1259,1264,1276,1282,1292,1298,1308 + 1,0,0,0,598,600,1,0,0,0,599,601,7,5,0,0,600,599,1,0,0,0,600,601, + 1,0,0,0,601,603,1,0,0,0,602,589,1,0,0,0,602,595,1,0,0,0,603,605, + 1,0,0,0,604,570,1,0,0,0,604,586,1,0,0,0,604,602,1,0,0,0,605,81,1, + 0,0,0,606,607,5,17,0,0,607,610,5,47,0,0,608,610,5,116,0,0,609,606, + 1,0,0,0,609,608,1,0,0,0,610,83,1,0,0,0,611,612,5,62,0,0,612,621, + 3,114,57,0,613,614,5,96,0,0,614,615,5,130,0,0,615,616,3,114,57,0, + 616,617,5,149,0,0,617,621,1,0,0,0,618,619,5,96,0,0,619,621,3,114, + 57,0,620,611,1,0,0,0,620,613,1,0,0,0,620,618,1,0,0,0,621,85,1,0, + 0,0,622,623,5,77,0,0,623,626,3,92,46,0,624,625,5,61,0,0,625,627, + 3,92,46,0,626,624,1,0,0,0,626,627,1,0,0,0,627,87,1,0,0,0,628,633, + 3,90,45,0,629,630,5,116,0,0,630,632,3,90,45,0,631,629,1,0,0,0,632, + 635,1,0,0,0,633,631,1,0,0,0,633,634,1,0,0,0,634,89,1,0,0,0,635,633, + 1,0,0,0,636,638,3,116,58,0,637,639,7,6,0,0,638,637,1,0,0,0,638,639, + 1,0,0,0,639,642,1,0,0,0,640,641,5,60,0,0,641,643,7,7,0,0,642,640, + 1,0,0,0,642,643,1,0,0,0,643,646,1,0,0,0,644,645,5,16,0,0,645,647, + 5,110,0,0,646,644,1,0,0,0,646,647,1,0,0,0,647,91,1,0,0,0,648,655, + 3,160,80,0,649,652,3,144,72,0,650,651,5,151,0,0,651,653,3,144,72, + 0,652,650,1,0,0,0,652,653,1,0,0,0,653,655,1,0,0,0,654,648,1,0,0, + 0,654,649,1,0,0,0,655,93,1,0,0,0,656,661,3,96,48,0,657,658,5,116, + 0,0,658,660,3,96,48,0,659,657,1,0,0,0,660,663,1,0,0,0,661,659,1, + 0,0,0,661,662,1,0,0,0,662,95,1,0,0,0,663,661,1,0,0,0,664,665,3,156, + 78,0,665,666,5,122,0,0,666,667,3,146,73,0,667,97,1,0,0,0,668,670, + 3,100,50,0,669,668,1,0,0,0,669,670,1,0,0,0,670,672,1,0,0,0,671,673, + 3,102,51,0,672,671,1,0,0,0,672,673,1,0,0,0,673,675,1,0,0,0,674,676, + 3,104,52,0,675,674,1,0,0,0,675,676,1,0,0,0,676,99,1,0,0,0,677,678, + 5,67,0,0,678,679,5,11,0,0,679,680,3,114,57,0,680,101,1,0,0,0,681, + 682,5,64,0,0,682,683,5,11,0,0,683,684,3,88,44,0,684,103,1,0,0,0, + 685,686,7,8,0,0,686,687,3,106,53,0,687,105,1,0,0,0,688,695,3,108, + 54,0,689,690,5,9,0,0,690,691,3,108,54,0,691,692,5,2,0,0,692,693, + 3,108,54,0,693,695,1,0,0,0,694,688,1,0,0,0,694,689,1,0,0,0,695,107, + 1,0,0,0,696,697,5,19,0,0,697,709,5,75,0,0,698,699,5,94,0,0,699,709, + 5,68,0,0,700,701,5,94,0,0,701,709,5,32,0,0,702,703,3,144,72,0,703, + 704,5,68,0,0,704,709,1,0,0,0,705,706,3,144,72,0,706,707,5,32,0,0, + 707,709,1,0,0,0,708,696,1,0,0,0,708,698,1,0,0,0,708,700,1,0,0,0, + 708,702,1,0,0,0,708,705,1,0,0,0,709,109,1,0,0,0,710,711,3,116,58, + 0,711,712,5,0,0,1,712,111,1,0,0,0,713,770,3,156,78,0,714,715,3,156, + 78,0,715,716,5,130,0,0,716,717,3,156,78,0,717,724,3,112,56,0,718, + 719,5,116,0,0,719,720,3,156,78,0,720,721,3,112,56,0,721,723,1,0, + 0,0,722,718,1,0,0,0,723,726,1,0,0,0,724,722,1,0,0,0,724,725,1,0, + 0,0,725,728,1,0,0,0,726,724,1,0,0,0,727,729,5,116,0,0,728,727,1, + 0,0,0,728,729,1,0,0,0,729,730,1,0,0,0,730,731,5,149,0,0,731,770, + 1,0,0,0,732,733,3,156,78,0,733,734,5,130,0,0,734,739,3,158,79,0, + 735,736,5,116,0,0,736,738,3,158,79,0,737,735,1,0,0,0,738,741,1,0, + 0,0,739,737,1,0,0,0,739,740,1,0,0,0,740,743,1,0,0,0,741,739,1,0, + 0,0,742,744,5,116,0,0,743,742,1,0,0,0,743,744,1,0,0,0,744,745,1, + 0,0,0,745,746,5,149,0,0,746,770,1,0,0,0,747,748,3,156,78,0,748,749, + 5,130,0,0,749,754,3,112,56,0,750,751,5,116,0,0,751,753,3,112,56, + 0,752,750,1,0,0,0,753,756,1,0,0,0,754,752,1,0,0,0,754,755,1,0,0, + 0,755,758,1,0,0,0,756,754,1,0,0,0,757,759,5,116,0,0,758,757,1,0, + 0,0,758,759,1,0,0,0,759,760,1,0,0,0,760,761,5,149,0,0,761,770,1, + 0,0,0,762,763,3,156,78,0,763,765,5,130,0,0,764,766,3,114,57,0,765, + 764,1,0,0,0,765,766,1,0,0,0,766,767,1,0,0,0,767,768,5,149,0,0,768, + 770,1,0,0,0,769,713,1,0,0,0,769,714,1,0,0,0,769,732,1,0,0,0,769, + 747,1,0,0,0,769,762,1,0,0,0,770,113,1,0,0,0,771,776,3,116,58,0,772, + 773,5,116,0,0,773,775,3,116,58,0,774,772,1,0,0,0,775,778,1,0,0,0, + 776,774,1,0,0,0,776,777,1,0,0,0,777,780,1,0,0,0,778,776,1,0,0,0, + 779,781,5,116,0,0,780,779,1,0,0,0,780,781,1,0,0,0,781,115,1,0,0, + 0,782,783,6,58,-1,0,783,785,5,12,0,0,784,786,3,116,58,0,785,784, + 1,0,0,0,785,786,1,0,0,0,786,792,1,0,0,0,787,788,5,98,0,0,788,789, + 3,116,58,0,789,790,5,83,0,0,790,791,3,116,58,0,791,793,1,0,0,0,792, + 787,1,0,0,0,793,794,1,0,0,0,794,792,1,0,0,0,794,795,1,0,0,0,795, + 798,1,0,0,0,796,797,5,25,0,0,797,799,3,116,58,0,798,796,1,0,0,0, + 798,799,1,0,0,0,799,800,1,0,0,0,800,801,5,26,0,0,801,933,1,0,0,0, + 802,803,5,13,0,0,803,804,5,130,0,0,804,805,3,116,58,0,805,806,5, + 6,0,0,806,807,3,112,56,0,807,808,5,149,0,0,808,933,1,0,0,0,809,810, + 5,20,0,0,810,933,5,110,0,0,811,812,5,45,0,0,812,813,3,116,58,0,813, + 814,3,148,74,0,814,933,1,0,0,0,815,816,5,82,0,0,816,817,5,130,0, + 0,817,818,3,116,58,0,818,819,5,34,0,0,819,822,3,116,58,0,820,821, + 5,33,0,0,821,823,3,116,58,0,822,820,1,0,0,0,822,823,1,0,0,0,823, + 824,1,0,0,0,824,825,5,149,0,0,825,933,1,0,0,0,826,827,5,86,0,0,827, + 933,5,110,0,0,828,829,5,91,0,0,829,830,5,130,0,0,830,831,7,9,0,0, + 831,832,3,162,81,0,832,833,5,34,0,0,833,834,3,116,58,0,834,835,5, + 149,0,0,835,933,1,0,0,0,836,837,3,156,78,0,837,839,5,130,0,0,838, + 840,3,114,57,0,839,838,1,0,0,0,839,840,1,0,0,0,840,841,1,0,0,0,841, + 842,5,149,0,0,842,851,1,0,0,0,843,845,5,130,0,0,844,846,5,24,0,0, + 845,844,1,0,0,0,845,846,1,0,0,0,846,848,1,0,0,0,847,849,3,114,57, + 0,848,847,1,0,0,0,848,849,1,0,0,0,849,850,1,0,0,0,850,852,5,149, + 0,0,851,843,1,0,0,0,851,852,1,0,0,0,852,853,1,0,0,0,853,854,5,66, + 0,0,854,855,5,130,0,0,855,856,3,98,49,0,856,857,5,149,0,0,857,933, + 1,0,0,0,858,859,3,156,78,0,859,861,5,130,0,0,860,862,3,114,57,0, + 861,860,1,0,0,0,861,862,1,0,0,0,862,863,1,0,0,0,863,864,5,149,0, + 0,864,873,1,0,0,0,865,867,5,130,0,0,866,868,5,24,0,0,867,866,1,0, + 0,0,867,868,1,0,0,0,868,870,1,0,0,0,869,871,3,114,57,0,870,869,1, + 0,0,0,870,871,1,0,0,0,871,872,1,0,0,0,872,874,5,149,0,0,873,865, + 1,0,0,0,873,874,1,0,0,0,874,875,1,0,0,0,875,876,5,66,0,0,876,877, + 3,156,78,0,877,933,1,0,0,0,878,884,3,156,78,0,879,881,5,130,0,0, + 880,882,3,114,57,0,881,880,1,0,0,0,881,882,1,0,0,0,882,883,1,0,0, + 0,883,885,5,149,0,0,884,879,1,0,0,0,884,885,1,0,0,0,885,886,1,0, + 0,0,886,888,5,130,0,0,887,889,5,24,0,0,888,887,1,0,0,0,888,889,1, + 0,0,0,889,891,1,0,0,0,890,892,3,114,57,0,891,890,1,0,0,0,891,892, + 1,0,0,0,892,893,1,0,0,0,893,894,5,149,0,0,894,933,1,0,0,0,895,933, + 3,120,60,0,896,933,3,164,82,0,897,933,3,146,73,0,898,899,5,118,0, + 0,899,933,3,116,58,20,900,901,5,58,0,0,901,933,3,116,58,14,902,903, + 3,136,68,0,903,904,5,120,0,0,904,906,1,0,0,0,905,902,1,0,0,0,905, + 906,1,0,0,0,906,907,1,0,0,0,907,933,5,112,0,0,908,909,5,130,0,0, + 909,910,3,44,22,0,910,911,5,149,0,0,911,933,1,0,0,0,912,913,5,130, + 0,0,913,914,3,116,58,0,914,915,5,149,0,0,915,933,1,0,0,0,916,917, + 5,130,0,0,917,918,3,114,57,0,918,919,5,149,0,0,919,933,1,0,0,0,920, + 922,5,129,0,0,921,923,3,114,57,0,922,921,1,0,0,0,922,923,1,0,0,0, + 923,924,1,0,0,0,924,933,5,148,0,0,925,927,5,128,0,0,926,928,3,40, + 20,0,927,926,1,0,0,0,927,928,1,0,0,0,928,929,1,0,0,0,929,933,5,147, + 0,0,930,933,3,118,59,0,931,933,3,128,64,0,932,782,1,0,0,0,932,802, + 1,0,0,0,932,809,1,0,0,0,932,811,1,0,0,0,932,815,1,0,0,0,932,826, + 1,0,0,0,932,828,1,0,0,0,932,836,1,0,0,0,932,858,1,0,0,0,932,878, + 1,0,0,0,932,895,1,0,0,0,932,896,1,0,0,0,932,897,1,0,0,0,932,898, + 1,0,0,0,932,900,1,0,0,0,932,905,1,0,0,0,932,908,1,0,0,0,932,912, + 1,0,0,0,932,916,1,0,0,0,932,920,1,0,0,0,932,925,1,0,0,0,932,930, + 1,0,0,0,932,931,1,0,0,0,933,1044,1,0,0,0,934,938,10,19,0,0,935,939, + 5,112,0,0,936,939,5,151,0,0,937,939,5,138,0,0,938,935,1,0,0,0,938, + 936,1,0,0,0,938,937,1,0,0,0,939,940,1,0,0,0,940,1043,3,116,58,20, + 941,945,10,18,0,0,942,946,5,139,0,0,943,946,5,118,0,0,944,946,5, + 117,0,0,945,942,1,0,0,0,945,943,1,0,0,0,945,944,1,0,0,0,946,947, + 1,0,0,0,947,1043,3,116,58,19,948,973,10,17,0,0,949,974,5,121,0,0, + 950,974,5,122,0,0,951,974,5,133,0,0,952,974,5,131,0,0,953,974,5, + 132,0,0,954,974,5,123,0,0,955,974,5,124,0,0,956,958,5,58,0,0,957, + 956,1,0,0,0,957,958,1,0,0,0,958,959,1,0,0,0,959,961,5,42,0,0,960, + 962,5,15,0,0,961,960,1,0,0,0,961,962,1,0,0,0,962,974,1,0,0,0,963, + 965,5,58,0,0,964,963,1,0,0,0,964,965,1,0,0,0,965,966,1,0,0,0,966, + 974,7,10,0,0,967,974,5,145,0,0,968,974,5,146,0,0,969,974,5,135,0, + 0,970,974,5,126,0,0,971,974,5,127,0,0,972,974,5,134,0,0,973,949, + 1,0,0,0,973,950,1,0,0,0,973,951,1,0,0,0,973,952,1,0,0,0,973,953, + 1,0,0,0,973,954,1,0,0,0,973,955,1,0,0,0,973,957,1,0,0,0,973,964, + 1,0,0,0,973,967,1,0,0,0,973,968,1,0,0,0,973,969,1,0,0,0,973,970, + 1,0,0,0,973,971,1,0,0,0,973,972,1,0,0,0,974,975,1,0,0,0,975,1043, + 3,116,58,18,976,977,10,15,0,0,977,978,5,137,0,0,978,1043,3,116,58, + 16,979,980,10,13,0,0,980,981,5,2,0,0,981,1043,3,116,58,14,982,983, + 10,12,0,0,983,984,5,63,0,0,984,1043,3,116,58,13,985,987,10,11,0, + 0,986,988,5,58,0,0,987,986,1,0,0,0,987,988,1,0,0,0,988,989,1,0,0, + 0,989,990,5,9,0,0,990,991,3,116,58,0,991,992,5,2,0,0,992,993,3,116, + 58,12,993,1043,1,0,0,0,994,995,10,10,0,0,995,996,5,140,0,0,996,997, + 3,116,58,0,997,998,5,115,0,0,998,999,3,116,58,10,999,1043,1,0,0, + 0,1000,1001,10,30,0,0,1001,1003,5,130,0,0,1002,1004,3,114,57,0,1003, + 1002,1,0,0,0,1003,1004,1,0,0,0,1004,1005,1,0,0,0,1005,1043,5,149, + 0,0,1006,1007,10,26,0,0,1007,1008,5,129,0,0,1008,1009,3,116,58,0, + 1009,1010,5,148,0,0,1010,1043,1,0,0,0,1011,1012,10,25,0,0,1012,1013, + 5,120,0,0,1013,1043,5,108,0,0,1014,1015,10,24,0,0,1015,1016,5,120, + 0,0,1016,1043,3,156,78,0,1017,1018,10,23,0,0,1018,1019,5,136,0,0, + 1019,1020,5,129,0,0,1020,1021,3,116,58,0,1021,1022,5,148,0,0,1022, + 1043,1,0,0,0,1023,1024,10,22,0,0,1024,1025,5,136,0,0,1025,1043,5, + 108,0,0,1026,1027,10,21,0,0,1027,1028,5,136,0,0,1028,1043,3,156, + 78,0,1029,1030,10,16,0,0,1030,1032,5,46,0,0,1031,1033,5,58,0,0,1032, + 1031,1,0,0,0,1032,1033,1,0,0,0,1033,1034,1,0,0,0,1034,1043,5,59, + 0,0,1035,1040,10,9,0,0,1036,1037,5,6,0,0,1037,1041,3,156,78,0,1038, + 1039,5,6,0,0,1039,1041,5,110,0,0,1040,1036,1,0,0,0,1040,1038,1,0, + 0,0,1041,1043,1,0,0,0,1042,934,1,0,0,0,1042,941,1,0,0,0,1042,948, + 1,0,0,0,1042,976,1,0,0,0,1042,979,1,0,0,0,1042,982,1,0,0,0,1042, + 985,1,0,0,0,1042,994,1,0,0,0,1042,1000,1,0,0,0,1042,1006,1,0,0,0, + 1042,1011,1,0,0,0,1042,1014,1,0,0,0,1042,1017,1,0,0,0,1042,1023, + 1,0,0,0,1042,1026,1,0,0,0,1042,1029,1,0,0,0,1042,1035,1,0,0,0,1043, + 1046,1,0,0,0,1044,1042,1,0,0,0,1044,1045,1,0,0,0,1045,117,1,0,0, + 0,1046,1044,1,0,0,0,1047,1048,5,130,0,0,1048,1053,3,156,78,0,1049, + 1050,5,116,0,0,1050,1052,3,156,78,0,1051,1049,1,0,0,0,1052,1055, + 1,0,0,0,1053,1051,1,0,0,0,1053,1054,1,0,0,0,1054,1057,1,0,0,0,1055, + 1053,1,0,0,0,1056,1058,5,116,0,0,1057,1056,1,0,0,0,1057,1058,1,0, + 0,0,1058,1059,1,0,0,0,1059,1060,5,149,0,0,1060,1075,1,0,0,0,1061, + 1066,3,156,78,0,1062,1063,5,116,0,0,1063,1065,3,156,78,0,1064,1062, + 1,0,0,0,1065,1068,1,0,0,0,1066,1064,1,0,0,0,1066,1067,1,0,0,0,1067, + 1070,1,0,0,0,1068,1066,1,0,0,0,1069,1071,5,116,0,0,1070,1069,1,0, + 0,0,1070,1071,1,0,0,0,1071,1075,1,0,0,0,1072,1073,5,130,0,0,1073, + 1075,5,149,0,0,1074,1047,1,0,0,0,1074,1061,1,0,0,0,1074,1072,1,0, + 0,0,1075,1076,1,0,0,0,1076,1079,5,111,0,0,1077,1080,3,116,58,0,1078, + 1080,3,36,18,0,1079,1077,1,0,0,0,1079,1078,1,0,0,0,1080,119,1,0, + 0,0,1081,1082,5,132,0,0,1082,1086,3,156,78,0,1083,1085,3,122,61, + 0,1084,1083,1,0,0,0,1085,1088,1,0,0,0,1086,1084,1,0,0,0,1086,1087, + 1,0,0,0,1087,1089,1,0,0,0,1088,1086,1,0,0,0,1089,1090,5,151,0,0, + 1090,1091,5,124,0,0,1091,1114,1,0,0,0,1092,1093,5,132,0,0,1093,1097, + 3,156,78,0,1094,1096,3,122,61,0,1095,1094,1,0,0,0,1096,1099,1,0, + 0,0,1097,1095,1,0,0,0,1097,1098,1,0,0,0,1098,1100,1,0,0,0,1099,1097, + 1,0,0,0,1100,1106,5,124,0,0,1101,1107,3,120,60,0,1102,1103,5,128, + 0,0,1103,1104,3,116,58,0,1104,1105,5,147,0,0,1105,1107,1,0,0,0,1106, + 1101,1,0,0,0,1106,1102,1,0,0,0,1106,1107,1,0,0,0,1107,1108,1,0,0, + 0,1108,1109,5,132,0,0,1109,1110,5,151,0,0,1110,1111,3,156,78,0,1111, + 1112,5,124,0,0,1112,1114,1,0,0,0,1113,1081,1,0,0,0,1113,1092,1,0, + 0,0,1114,121,1,0,0,0,1115,1116,3,156,78,0,1116,1117,5,122,0,0,1117, + 1118,3,162,81,0,1118,1127,1,0,0,0,1119,1120,3,156,78,0,1120,1121, + 5,122,0,0,1121,1122,5,128,0,0,1122,1123,3,116,58,0,1123,1124,5,147, + 0,0,1124,1127,1,0,0,0,1125,1127,3,156,78,0,1126,1115,1,0,0,0,1126, + 1119,1,0,0,0,1126,1125,1,0,0,0,1127,123,1,0,0,0,1128,1133,3,126, + 63,0,1129,1130,5,116,0,0,1130,1132,3,126,63,0,1131,1129,1,0,0,0, + 1132,1135,1,0,0,0,1133,1131,1,0,0,0,1133,1134,1,0,0,0,1134,1137, + 1,0,0,0,1135,1133,1,0,0,0,1136,1138,5,116,0,0,1137,1136,1,0,0,0, + 1137,1138,1,0,0,0,1138,125,1,0,0,0,1139,1140,3,156,78,0,1140,1141, + 5,6,0,0,1141,1142,5,130,0,0,1142,1143,3,44,22,0,1143,1144,5,149, + 0,0,1144,1150,1,0,0,0,1145,1146,3,116,58,0,1146,1147,5,6,0,0,1147, + 1148,3,156,78,0,1148,1150,1,0,0,0,1149,1139,1,0,0,0,1149,1145,1, + 0,0,0,1150,127,1,0,0,0,1151,1159,3,160,80,0,1152,1153,3,136,68,0, + 1153,1154,5,120,0,0,1154,1156,1,0,0,0,1155,1152,1,0,0,0,1155,1156, + 1,0,0,0,1156,1157,1,0,0,0,1157,1159,3,130,65,0,1158,1151,1,0,0,0, + 1158,1155,1,0,0,0,1159,129,1,0,0,0,1160,1165,3,156,78,0,1161,1162, + 5,120,0,0,1162,1164,3,156,78,0,1163,1161,1,0,0,0,1164,1167,1,0,0, + 0,1165,1163,1,0,0,0,1165,1166,1,0,0,0,1166,131,1,0,0,0,1167,1165, + 1,0,0,0,1168,1169,6,66,-1,0,1169,1178,3,136,68,0,1170,1178,3,134, + 67,0,1171,1172,5,130,0,0,1172,1173,3,44,22,0,1173,1174,5,149,0,0, + 1174,1178,1,0,0,0,1175,1178,3,120,60,0,1176,1178,3,160,80,0,1177, + 1168,1,0,0,0,1177,1170,1,0,0,0,1177,1171,1,0,0,0,1177,1175,1,0,0, + 0,1177,1176,1,0,0,0,1178,1187,1,0,0,0,1179,1183,10,3,0,0,1180,1184, + 3,154,77,0,1181,1182,5,6,0,0,1182,1184,3,156,78,0,1183,1180,1,0, + 0,0,1183,1181,1,0,0,0,1184,1186,1,0,0,0,1185,1179,1,0,0,0,1186,1189, + 1,0,0,0,1187,1185,1,0,0,0,1187,1188,1,0,0,0,1188,133,1,0,0,0,1189, + 1187,1,0,0,0,1190,1191,3,156,78,0,1191,1193,5,130,0,0,1192,1194, + 3,138,69,0,1193,1192,1,0,0,0,1193,1194,1,0,0,0,1194,1195,1,0,0,0, + 1195,1196,5,149,0,0,1196,135,1,0,0,0,1197,1198,3,140,70,0,1198,1199, + 5,120,0,0,1199,1201,1,0,0,0,1200,1197,1,0,0,0,1200,1201,1,0,0,0, + 1201,1202,1,0,0,0,1202,1203,3,156,78,0,1203,137,1,0,0,0,1204,1209, + 3,116,58,0,1205,1206,5,116,0,0,1206,1208,3,116,58,0,1207,1205,1, + 0,0,0,1208,1211,1,0,0,0,1209,1207,1,0,0,0,1209,1210,1,0,0,0,1210, + 1213,1,0,0,0,1211,1209,1,0,0,0,1212,1214,5,116,0,0,1213,1212,1,0, + 0,0,1213,1214,1,0,0,0,1214,139,1,0,0,0,1215,1216,3,156,78,0,1216, + 141,1,0,0,0,1217,1226,5,106,0,0,1218,1219,5,120,0,0,1219,1226,7, + 11,0,0,1220,1221,5,108,0,0,1221,1223,5,120,0,0,1222,1224,7,11,0, + 0,1223,1222,1,0,0,0,1223,1224,1,0,0,0,1224,1226,1,0,0,0,1225,1217, + 1,0,0,0,1225,1218,1,0,0,0,1225,1220,1,0,0,0,1226,143,1,0,0,0,1227, + 1229,7,12,0,0,1228,1227,1,0,0,0,1228,1229,1,0,0,0,1229,1236,1,0, + 0,0,1230,1237,3,142,71,0,1231,1237,5,107,0,0,1232,1237,5,108,0,0, + 1233,1237,5,109,0,0,1234,1237,5,43,0,0,1235,1237,5,57,0,0,1236,1230, + 1,0,0,0,1236,1231,1,0,0,0,1236,1232,1,0,0,0,1236,1233,1,0,0,0,1236, + 1234,1,0,0,0,1236,1235,1,0,0,0,1237,145,1,0,0,0,1238,1242,3,144, + 72,0,1239,1242,5,110,0,0,1240,1242,5,59,0,0,1241,1238,1,0,0,0,1241, + 1239,1,0,0,0,1241,1240,1,0,0,0,1242,147,1,0,0,0,1243,1244,7,13,0, + 0,1244,149,1,0,0,0,1245,1246,7,14,0,0,1246,151,1,0,0,0,1247,1248, + 7,15,0,0,1248,153,1,0,0,0,1249,1252,5,105,0,0,1250,1252,3,152,76, + 0,1251,1249,1,0,0,0,1251,1250,1,0,0,0,1252,155,1,0,0,0,1253,1257, + 5,105,0,0,1254,1257,3,148,74,0,1255,1257,3,150,75,0,1256,1253,1, + 0,0,0,1256,1254,1,0,0,0,1256,1255,1,0,0,0,1257,157,1,0,0,0,1258, + 1259,3,162,81,0,1259,1260,5,122,0,0,1260,1261,3,144,72,0,1261,159, + 1,0,0,0,1262,1263,5,128,0,0,1263,1264,3,130,65,0,1264,1265,5,147, + 0,0,1265,161,1,0,0,0,1266,1269,5,110,0,0,1267,1269,3,164,82,0,1268, + 1266,1,0,0,0,1268,1267,1,0,0,0,1269,163,1,0,0,0,1270,1274,5,142, + 0,0,1271,1273,3,166,83,0,1272,1271,1,0,0,0,1273,1276,1,0,0,0,1274, + 1272,1,0,0,0,1274,1275,1,0,0,0,1275,1277,1,0,0,0,1276,1274,1,0,0, + 0,1277,1278,5,144,0,0,1278,165,1,0,0,0,1279,1280,5,157,0,0,1280, + 1281,3,116,58,0,1281,1282,5,147,0,0,1282,1285,1,0,0,0,1283,1285, + 5,156,0,0,1284,1279,1,0,0,0,1284,1283,1,0,0,0,1285,167,1,0,0,0,1286, + 1290,5,143,0,0,1287,1289,3,170,85,0,1288,1287,1,0,0,0,1289,1292, + 1,0,0,0,1290,1288,1,0,0,0,1290,1291,1,0,0,0,1291,1293,1,0,0,0,1292, + 1290,1,0,0,0,1293,1294,5,0,0,1,1294,169,1,0,0,0,1295,1296,5,159, + 0,0,1296,1297,3,116,58,0,1297,1298,5,147,0,0,1298,1301,1,0,0,0,1299, + 1301,5,158,0,0,1300,1295,1,0,0,0,1300,1299,1,0,0,0,1301,171,1,0, + 0,0,167,175,182,191,198,202,216,220,223,227,230,237,241,250,255, + 264,272,279,283,289,294,302,309,315,327,335,349,353,358,368,377, + 380,384,387,391,394,397,400,403,407,411,414,417,420,424,427,436, + 442,463,480,497,503,509,520,522,533,536,542,550,556,558,562,567, + 570,573,577,581,584,586,589,593,597,600,602,604,609,620,626,633, + 638,642,646,652,654,661,669,672,675,694,708,724,728,739,743,754, + 758,765,769,776,780,785,794,798,822,839,845,848,851,861,867,870, + 873,881,884,888,891,905,922,927,932,938,945,957,961,964,973,987, + 1003,1032,1040,1042,1044,1053,1057,1066,1070,1074,1079,1086,1097, + 1106,1113,1126,1133,1137,1149,1155,1158,1165,1177,1183,1187,1193, + 1200,1209,1213,1223,1225,1228,1236,1241,1251,1256,1268,1274,1284, + 1290,1300 ] class HogQLParser ( Parser ): @@ -675,35 +674,33 @@ class HogQLParser ( Parser ): RULE_columnTypeExpr = 56 RULE_columnExprList = 57 RULE_columnExpr = 58 - RULE_columnArgList = 59 - RULE_columnArgExpr = 60 - RULE_columnLambdaExpr = 61 - RULE_hogqlxTagElement = 62 - RULE_hogqlxTagAttribute = 63 - RULE_withExprList = 64 - RULE_withExpr = 65 - RULE_columnIdentifier = 66 - RULE_nestedIdentifier = 67 - RULE_tableExpr = 68 - RULE_tableFunctionExpr = 69 - RULE_tableIdentifier = 70 - RULE_tableArgList = 71 - RULE_databaseIdentifier = 72 - RULE_floatingLiteral = 73 - RULE_numberLiteral = 74 - RULE_literal = 75 - RULE_interval = 76 - RULE_keyword = 77 - RULE_keywordForAlias = 78 - RULE_alias = 79 - RULE_identifier = 80 - RULE_enumValue = 81 - RULE_placeholder = 82 - RULE_string = 83 - RULE_templateString = 84 - RULE_stringContents = 85 - RULE_fullTemplateString = 86 - RULE_stringContentsFull = 87 + RULE_columnLambdaExpr = 59 + RULE_hogqlxTagElement = 60 + RULE_hogqlxTagAttribute = 61 + RULE_withExprList = 62 + RULE_withExpr = 63 + RULE_columnIdentifier = 64 + RULE_nestedIdentifier = 65 + RULE_tableExpr = 66 + RULE_tableFunctionExpr = 67 + RULE_tableIdentifier = 68 + RULE_tableArgList = 69 + RULE_databaseIdentifier = 70 + RULE_floatingLiteral = 71 + RULE_numberLiteral = 72 + RULE_literal = 73 + RULE_interval = 74 + RULE_keyword = 75 + RULE_keywordForAlias = 76 + RULE_alias = 77 + RULE_identifier = 78 + RULE_enumValue = 79 + RULE_placeholder = 80 + RULE_string = 81 + RULE_templateString = 82 + RULE_stringContents = 83 + RULE_fullTemplateString = 84 + RULE_stringContentsFull = 85 ruleNames = [ "program", "declaration", "expression", "varDecl", "identifierList", "statement", "returnStmt", "throwStmt", "catchBlock", @@ -720,12 +717,11 @@ class HogQLParser ( Parser ): "windowExpr", "winPartitionByClause", "winOrderByClause", "winFrameClause", "winFrameExtend", "winFrameBound", "expr", "columnTypeExpr", "columnExprList", "columnExpr", - "columnArgList", "columnArgExpr", "columnLambdaExpr", - "hogqlxTagElement", "hogqlxTagAttribute", "withExprList", - "withExpr", "columnIdentifier", "nestedIdentifier", "tableExpr", - "tableFunctionExpr", "tableIdentifier", "tableArgList", - "databaseIdentifier", "floatingLiteral", "numberLiteral", - "literal", "interval", "keyword", "keywordForAlias", + "columnLambdaExpr", "hogqlxTagElement", "hogqlxTagAttribute", + "withExprList", "withExpr", "columnIdentifier", "nestedIdentifier", + "tableExpr", "tableFunctionExpr", "tableIdentifier", + "tableArgList", "databaseIdentifier", "floatingLiteral", + "numberLiteral", "literal", "interval", "keyword", "keywordForAlias", "alias", "identifier", "enumValue", "placeholder", "string", "templateString", "stringContents", "fullTemplateString", "stringContentsFull" ] @@ -893,7 +889,7 @@ class HogQLParser ( Parser ): def __init__(self, input:TokenStream, output:TextIO = sys.stdout): super().__init__(input, output) - self.checkVersion("4.13.1") + self.checkVersion("4.13.2") self._interp = ParserATNSimulator(self, self.atn, self.decisionsToDFA, self.sharedContextCache) self._predicates = None @@ -936,17 +932,17 @@ def program(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 179 + self.state = 175 self._errHandler.sync(self) _la = self._input.LA(1) while (((_la) & ~0x3f) == 0 and ((1 << _la) & -536887298) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493105500848127) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 4212759) != 0): - self.state = 176 + self.state = 172 self.declaration() - self.state = 181 + self.state = 177 self._errHandler.sync(self) _la = self._input.LA(1) - self.state = 182 + self.state = 178 self.match(HogQLParser.EOF) except RecognitionException as re: localctx.exception = re @@ -989,17 +985,17 @@ def declaration(self): localctx = HogQLParser.DeclarationContext(self, self._ctx, self.state) self.enterRule(localctx, 2, self.RULE_declaration) try: - self.state = 186 + self.state = 182 self._errHandler.sync(self) token = self._input.LA(1) if token in [52]: self.enterOuterAlt(localctx, 1) - self.state = 184 + self.state = 180 self.varDecl() pass elif token in [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95, 96, 97, 98, 99, 100, 101, 102, 103, 105, 106, 107, 108, 109, 110, 112, 118, 120, 128, 129, 130, 132, 139, 142, 150]: self.enterOuterAlt(localctx, 2) - self.state = 185 + self.state = 181 self.statement() pass else: @@ -1043,7 +1039,7 @@ def expression(self): self.enterRule(localctx, 4, self.RULE_expression) try: self.enterOuterAlt(localctx, 1) - self.state = 188 + self.state = 184 self.columnExpr(0) except RecognitionException as re: localctx.exception = re @@ -1097,19 +1093,19 @@ def varDecl(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 190 + self.state = 186 self.match(HogQLParser.LET) - self.state = 191 + self.state = 187 self.identifier() - self.state = 195 + self.state = 191 self._errHandler.sync(self) _la = self._input.LA(1) if _la==115: - self.state = 192 + self.state = 188 self.match(HogQLParser.COLON) - self.state = 193 + self.state = 189 self.match(HogQLParser.EQ_SINGLE) - self.state = 194 + self.state = 190 self.expression() @@ -1161,26 +1157,26 @@ def identifierList(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 197 + self.state = 193 self.identifier() - self.state = 202 + self.state = 198 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,3,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 198 + self.state = 194 self.match(HogQLParser.COMMA) - self.state = 199 + self.state = 195 self.identifier() - self.state = 204 + self.state = 200 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,3,self._ctx) - self.state = 206 + self.state = 202 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 205 + self.state = 201 self.match(HogQLParser.COMMA) @@ -1265,78 +1261,78 @@ def statement(self): localctx = HogQLParser.StatementContext(self, self._ctx, self.state) self.enterRule(localctx, 10, self.RULE_statement) try: - self.state = 220 + self.state = 216 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,5,self._ctx) if la_ == 1: self.enterOuterAlt(localctx, 1) - self.state = 208 + self.state = 204 self.returnStmt() pass elif la_ == 2: self.enterOuterAlt(localctx, 2) - self.state = 209 + self.state = 205 self.throwStmt() pass elif la_ == 3: self.enterOuterAlt(localctx, 3) - self.state = 210 + self.state = 206 self.tryCatchStmt() pass elif la_ == 4: self.enterOuterAlt(localctx, 4) - self.state = 211 + self.state = 207 self.ifStmt() pass elif la_ == 5: self.enterOuterAlt(localctx, 5) - self.state = 212 + self.state = 208 self.whileStmt() pass elif la_ == 6: self.enterOuterAlt(localctx, 6) - self.state = 213 + self.state = 209 self.forInStmt() pass elif la_ == 7: self.enterOuterAlt(localctx, 7) - self.state = 214 + self.state = 210 self.forStmt() pass elif la_ == 8: self.enterOuterAlt(localctx, 8) - self.state = 215 + self.state = 211 self.funcStmt() pass elif la_ == 9: self.enterOuterAlt(localctx, 9) - self.state = 216 + self.state = 212 self.varAssignment() pass elif la_ == 10: self.enterOuterAlt(localctx, 10) - self.state = 217 + self.state = 213 self.block() pass elif la_ == 11: self.enterOuterAlt(localctx, 11) - self.state = 218 + self.state = 214 self.exprStmt() pass elif la_ == 12: self.enterOuterAlt(localctx, 12) - self.state = 219 + self.state = 215 self.emptyStmt() pass @@ -1385,21 +1381,21 @@ def returnStmt(self): self.enterRule(localctx, 12, self.RULE_returnStmt) try: self.enterOuterAlt(localctx, 1) - self.state = 222 + self.state = 218 self.match(HogQLParser.RETURN) - self.state = 224 + self.state = 220 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,6,self._ctx) if la_ == 1: - self.state = 223 + self.state = 219 self.expression() - self.state = 227 + self.state = 223 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,7,self._ctx) if la_ == 1: - self.state = 226 + self.state = 222 self.match(HogQLParser.SEMICOLON) @@ -1447,21 +1443,21 @@ def throwStmt(self): self.enterRule(localctx, 14, self.RULE_throwStmt) try: self.enterOuterAlt(localctx, 1) - self.state = 229 + self.state = 225 self.match(HogQLParser.THROW) - self.state = 231 + self.state = 227 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,8,self._ctx) if la_ == 1: - self.state = 230 + self.state = 226 self.expression() - self.state = 234 + self.state = 230 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,9,self._ctx) if la_ == 1: - self.state = 233 + self.state = 229 self.match(HogQLParser.SEMICOLON) @@ -1526,31 +1522,31 @@ def catchBlock(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 236 + self.state = 232 self.match(HogQLParser.CATCH) - self.state = 245 + self.state = 241 self._errHandler.sync(self) _la = self._input.LA(1) if _la==130: - self.state = 237 + self.state = 233 self.match(HogQLParser.LPAREN) - self.state = 238 + self.state = 234 localctx.catchVar = self.identifier() - self.state = 241 + self.state = 237 self._errHandler.sync(self) _la = self._input.LA(1) if _la==115: - self.state = 239 + self.state = 235 self.match(HogQLParser.COLON) - self.state = 240 + self.state = 236 localctx.catchType = self.identifier() - self.state = 243 + self.state = 239 self.match(HogQLParser.RPAREN) - self.state = 247 + self.state = 243 localctx.catchStmt = self.block() except RecognitionException as re: localctx.exception = re @@ -1609,27 +1605,27 @@ def tryCatchStmt(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 249 + self.state = 245 self.match(HogQLParser.TRY) - self.state = 250 + self.state = 246 localctx.tryStmt = self.block() - self.state = 254 + self.state = 250 self._errHandler.sync(self) _la = self._input.LA(1) while _la==14: - self.state = 251 + self.state = 247 self.catchBlock() - self.state = 256 + self.state = 252 self._errHandler.sync(self) _la = self._input.LA(1) - self.state = 259 + self.state = 255 self._errHandler.sync(self) _la = self._input.LA(1) if _la==29: - self.state = 257 + self.state = 253 self.match(HogQLParser.FINALLY) - self.state = 258 + self.state = 254 localctx.finallyStmt = self.block() @@ -1690,23 +1686,23 @@ def ifStmt(self): self.enterRule(localctx, 20, self.RULE_ifStmt) try: self.enterOuterAlt(localctx, 1) - self.state = 261 + self.state = 257 self.match(HogQLParser.IF) - self.state = 262 + self.state = 258 self.match(HogQLParser.LPAREN) - self.state = 263 + self.state = 259 self.expression() - self.state = 264 + self.state = 260 self.match(HogQLParser.RPAREN) - self.state = 265 + self.state = 261 self.statement() - self.state = 268 + self.state = 264 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,14,self._ctx) if la_ == 1: - self.state = 266 + self.state = 262 self.match(HogQLParser.ELSE) - self.state = 267 + self.state = 263 self.statement() @@ -1764,21 +1760,21 @@ def whileStmt(self): self.enterRule(localctx, 22, self.RULE_whileStmt) try: self.enterOuterAlt(localctx, 1) - self.state = 270 + self.state = 266 self.match(HogQLParser.WHILE) - self.state = 271 + self.state = 267 self.match(HogQLParser.LPAREN) - self.state = 272 + self.state = 268 self.expression() - self.state = 273 + self.state = 269 self.match(HogQLParser.RPAREN) - self.state = 274 + self.state = 270 self.statement() - self.state = 276 + self.state = 272 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,15,self._ctx) if la_ == 1: - self.state = 275 + self.state = 271 self.match(HogQLParser.SEMICOLON) @@ -1864,63 +1860,63 @@ def forStmt(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 278 + self.state = 274 self.match(HogQLParser.FOR) - self.state = 279 + self.state = 275 self.match(HogQLParser.LPAREN) - self.state = 283 + self.state = 279 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,16,self._ctx) if la_ == 1: - self.state = 280 + self.state = 276 localctx.initializerVarDeclr = self.varDecl() elif la_ == 2: - self.state = 281 + self.state = 277 localctx.initializerVarAssignment = self.varAssignment() elif la_ == 3: - self.state = 282 + self.state = 278 localctx.initializerExpression = self.expression() - self.state = 285 + self.state = 281 self.match(HogQLParser.SEMICOLON) - self.state = 287 + self.state = 283 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 286 + self.state = 282 localctx.condition = self.expression() - self.state = 289 + self.state = 285 self.match(HogQLParser.SEMICOLON) - self.state = 293 + self.state = 289 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,18,self._ctx) if la_ == 1: - self.state = 290 + self.state = 286 localctx.incrementVarDeclr = self.varDecl() elif la_ == 2: - self.state = 291 + self.state = 287 localctx.incrementVarAssignment = self.varAssignment() elif la_ == 3: - self.state = 292 + self.state = 288 localctx.incrementExpression = self.expression() - self.state = 295 + self.state = 291 self.match(HogQLParser.RPAREN) - self.state = 296 + self.state = 292 self.statement() - self.state = 298 + self.state = 294 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,19,self._ctx) if la_ == 1: - self.state = 297 + self.state = 293 self.match(HogQLParser.SEMICOLON) @@ -1995,37 +1991,37 @@ def forInStmt(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 300 + self.state = 296 self.match(HogQLParser.FOR) - self.state = 301 + self.state = 297 self.match(HogQLParser.LPAREN) - self.state = 302 + self.state = 298 self.match(HogQLParser.LET) - self.state = 303 + self.state = 299 self.identifier() - self.state = 306 + self.state = 302 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 304 + self.state = 300 self.match(HogQLParser.COMMA) - self.state = 305 + self.state = 301 self.identifier() - self.state = 308 + self.state = 304 self.match(HogQLParser.IN) - self.state = 309 + self.state = 305 self.expression() - self.state = 310 + self.state = 306 self.match(HogQLParser.RPAREN) - self.state = 311 + self.state = 307 self.statement() - self.state = 313 + self.state = 309 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,21,self._ctx) if la_ == 1: - self.state = 312 + self.state = 308 self.match(HogQLParser.SEMICOLON) @@ -2085,23 +2081,23 @@ def funcStmt(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 315 + self.state = 311 self.match(HogQLParser.FN) - self.state = 316 + self.state = 312 self.identifier() - self.state = 317 + self.state = 313 self.match(HogQLParser.LPAREN) - self.state = 319 + self.state = 315 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -725088338784043010) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 3229277487103) != 0): - self.state = 318 + self.state = 314 self.identifierList() - self.state = 321 + self.state = 317 self.match(HogQLParser.RPAREN) - self.state = 322 + self.state = 318 self.block() except RecognitionException as re: localctx.exception = re @@ -2150,13 +2146,13 @@ def varAssignment(self): self.enterRule(localctx, 30, self.RULE_varAssignment) try: self.enterOuterAlt(localctx, 1) - self.state = 324 + self.state = 320 self.expression() - self.state = 325 + self.state = 321 self.match(HogQLParser.COLON) - self.state = 326 + self.state = 322 self.match(HogQLParser.EQ_SINGLE) - self.state = 327 + self.state = 323 self.expression() except RecognitionException as re: localctx.exception = re @@ -2199,13 +2195,13 @@ def exprStmt(self): self.enterRule(localctx, 32, self.RULE_exprStmt) try: self.enterOuterAlt(localctx, 1) - self.state = 329 + self.state = 325 self.expression() - self.state = 331 + self.state = 327 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,23,self._ctx) if la_ == 1: - self.state = 330 + self.state = 326 self.match(HogQLParser.SEMICOLON) @@ -2246,7 +2242,7 @@ def emptyStmt(self): self.enterRule(localctx, 34, self.RULE_emptyStmt) try: self.enterOuterAlt(localctx, 1) - self.state = 333 + self.state = 329 self.match(HogQLParser.SEMICOLON) except RecognitionException as re: localctx.exception = re @@ -2296,19 +2292,19 @@ def block(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 335 + self.state = 331 self.match(HogQLParser.LBRACE) - self.state = 339 + self.state = 335 self._errHandler.sync(self) _la = self._input.LA(1) while (((_la) & ~0x3f) == 0 and ((1 << _la) & -536887298) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493105500848127) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 4212759) != 0): - self.state = 336 + self.state = 332 self.declaration() - self.state = 341 + self.state = 337 self._errHandler.sync(self) _la = self._input.LA(1) - self.state = 342 + self.state = 338 self.match(HogQLParser.RBRACE) except RecognitionException as re: localctx.exception = re @@ -2354,11 +2350,11 @@ def kvPair(self): self.enterRule(localctx, 38, self.RULE_kvPair) try: self.enterOuterAlt(localctx, 1) - self.state = 344 + self.state = 340 self.expression() - self.state = 345 + self.state = 341 self.match(HogQLParser.COLON) - self.state = 346 + self.state = 342 self.expression() except RecognitionException as re: localctx.exception = re @@ -2408,26 +2404,26 @@ def kvPairList(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 348 + self.state = 344 self.kvPair() - self.state = 353 + self.state = 349 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,25,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 349 + self.state = 345 self.match(HogQLParser.COMMA) - self.state = 350 + self.state = 346 self.kvPair() - self.state = 355 + self.state = 351 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,25,self._ctx) - self.state = 357 + self.state = 353 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 356 + self.state = 352 self.match(HogQLParser.COMMA) @@ -2480,26 +2476,26 @@ def select(self): self.enterRule(localctx, 42, self.RULE_select) try: self.enterOuterAlt(localctx, 1) - self.state = 362 + self.state = 358 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,27,self._ctx) if la_ == 1: - self.state = 359 + self.state = 355 self.selectUnionStmt() pass elif la_ == 2: - self.state = 360 + self.state = 356 self.selectStmt() pass elif la_ == 3: - self.state = 361 + self.state = 357 self.hogqlxTagElement() pass - self.state = 364 + self.state = 360 self.match(HogQLParser.EOF) except RecognitionException as re: localctx.exception = re @@ -2555,19 +2551,19 @@ def selectUnionStmt(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 366 + self.state = 362 self.selectStmtWithParens() - self.state = 372 + self.state = 368 self._errHandler.sync(self) _la = self._input.LA(1) while _la==95: - self.state = 367 + self.state = 363 self.match(HogQLParser.UNION) - self.state = 368 + self.state = 364 self.match(HogQLParser.ALL) - self.state = 369 + self.state = 365 self.selectStmtWithParens() - self.state = 374 + self.state = 370 self._errHandler.sync(self) _la = self._input.LA(1) @@ -2622,26 +2618,26 @@ def selectStmtWithParens(self): localctx = HogQLParser.SelectStmtWithParensContext(self, self._ctx, self.state) self.enterRule(localctx, 46, self.RULE_selectStmtWithParens) try: - self.state = 381 + self.state = 377 self._errHandler.sync(self) token = self._input.LA(1) if token in [79, 102]: self.enterOuterAlt(localctx, 1) - self.state = 375 + self.state = 371 self.selectStmt() pass elif token in [130]: self.enterOuterAlt(localctx, 2) - self.state = 376 + self.state = 372 self.match(HogQLParser.LPAREN) - self.state = 377 + self.state = 373 self.selectUnionStmt() - self.state = 378 + self.state = 374 self.match(HogQLParser.RPAREN) pass elif token in [128]: self.enterOuterAlt(localctx, 3) - self.state = 380 + self.state = 376 self.placeholder() pass else: @@ -2763,81 +2759,81 @@ def selectStmt(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 384 + self.state = 380 self._errHandler.sync(self) _la = self._input.LA(1) if _la==102: - self.state = 383 + self.state = 379 localctx.with_ = self.withClause() - self.state = 386 + self.state = 382 self.match(HogQLParser.SELECT) - self.state = 388 + self.state = 384 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,31,self._ctx) if la_ == 1: - self.state = 387 + self.state = 383 self.match(HogQLParser.DISTINCT) - self.state = 391 + self.state = 387 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,32,self._ctx) if la_ == 1: - self.state = 390 + self.state = 386 self.topClause() - self.state = 393 + self.state = 389 localctx.columns = self.columnExprList() - self.state = 395 + self.state = 391 self._errHandler.sync(self) _la = self._input.LA(1) if _la==34: - self.state = 394 + self.state = 390 localctx.from_ = self.fromClause() - self.state = 398 + self.state = 394 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & 2269391999729696) != 0): - self.state = 397 + self.state = 393 self.arrayJoinClause() - self.state = 401 + self.state = 397 self._errHandler.sync(self) _la = self._input.LA(1) if _la==69: - self.state = 400 + self.state = 396 self.prewhereClause() - self.state = 404 + self.state = 400 self._errHandler.sync(self) _la = self._input.LA(1) if _la==99: - self.state = 403 + self.state = 399 localctx.where = self.whereClause() - self.state = 407 + self.state = 403 self._errHandler.sync(self) _la = self._input.LA(1) if _la==36: - self.state = 406 + self.state = 402 self.groupByClause() - self.state = 411 + self.state = 407 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,38,self._ctx) if la_ == 1: - self.state = 409 + self.state = 405 self.match(HogQLParser.WITH) - self.state = 410 + self.state = 406 _la = self._input.LA(1) if not(_la==18 or _la==74): self._errHandler.recoverInline(self) @@ -2846,60 +2842,60 @@ def selectStmt(self): self.consume() - self.state = 415 + self.state = 411 self._errHandler.sync(self) _la = self._input.LA(1) if _la==102: - self.state = 413 + self.state = 409 self.match(HogQLParser.WITH) - self.state = 414 + self.state = 410 self.match(HogQLParser.TOTALS) - self.state = 418 + self.state = 414 self._errHandler.sync(self) _la = self._input.LA(1) if _la==37: - self.state = 417 + self.state = 413 self.havingClause() - self.state = 421 + self.state = 417 self._errHandler.sync(self) _la = self._input.LA(1) if _la==101: - self.state = 420 + self.state = 416 self.windowClause() - self.state = 424 + self.state = 420 self._errHandler.sync(self) _la = self._input.LA(1) if _la==64: - self.state = 423 + self.state = 419 self.orderByClause() - self.state = 428 + self.state = 424 self._errHandler.sync(self) token = self._input.LA(1) if token in [54]: - self.state = 426 + self.state = 422 self.limitAndOffsetClause() pass elif token in [61]: - self.state = 427 + self.state = 423 self.offsetOnlyClause() pass elif token in [-1, 81, 95, 149]: pass else: pass - self.state = 431 + self.state = 427 self._errHandler.sync(self) _la = self._input.LA(1) if _la==81: - self.state = 430 + self.state = 426 self.settingsClause() @@ -2944,9 +2940,9 @@ def withClause(self): self.enterRule(localctx, 50, self.RULE_withClause) try: self.enterOuterAlt(localctx, 1) - self.state = 433 + self.state = 429 self.match(HogQLParser.WITH) - self.state = 434 + self.state = 430 self.withExprList() except RecognitionException as re: localctx.exception = re @@ -2994,17 +2990,17 @@ def topClause(self): self.enterRule(localctx, 52, self.RULE_topClause) try: self.enterOuterAlt(localctx, 1) - self.state = 436 + self.state = 432 self.match(HogQLParser.TOP) - self.state = 437 + self.state = 433 self.match(HogQLParser.DECIMAL_LITERAL) - self.state = 440 + self.state = 436 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,45,self._ctx) if la_ == 1: - self.state = 438 + self.state = 434 self.match(HogQLParser.WITH) - self.state = 439 + self.state = 435 self.match(HogQLParser.TIES) @@ -3049,9 +3045,9 @@ def fromClause(self): self.enterRule(localctx, 54, self.RULE_fromClause) try: self.enterOuterAlt(localctx, 1) - self.state = 442 + self.state = 438 self.match(HogQLParser.FROM) - self.state = 443 + self.state = 439 self.joinExpr(0) except RecognitionException as re: localctx.exception = re @@ -3104,11 +3100,11 @@ def arrayJoinClause(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 446 + self.state = 442 self._errHandler.sync(self) _la = self._input.LA(1) if _la==44 or _la==51: - self.state = 445 + self.state = 441 _la = self._input.LA(1) if not(_la==44 or _la==51): self._errHandler.recoverInline(self) @@ -3117,11 +3113,11 @@ def arrayJoinClause(self): self.consume() - self.state = 448 + self.state = 444 self.match(HogQLParser.ARRAY) - self.state = 449 + self.state = 445 self.match(HogQLParser.JOIN) - self.state = 450 + self.state = 446 self.columnExprList() except RecognitionException as re: localctx.exception = re @@ -3199,35 +3195,35 @@ def windowClause(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 452 + self.state = 448 self.match(HogQLParser.WINDOW) - self.state = 453 + self.state = 449 self.identifier() - self.state = 454 + self.state = 450 self.match(HogQLParser.AS) - self.state = 455 + self.state = 451 self.match(HogQLParser.LPAREN) - self.state = 456 + self.state = 452 self.windowExpr() - self.state = 457 + self.state = 453 self.match(HogQLParser.RPAREN) - self.state = 467 + self.state = 463 self._errHandler.sync(self) _la = self._input.LA(1) while _la==116: - self.state = 458 + self.state = 454 self.match(HogQLParser.COMMA) - self.state = 459 + self.state = 455 self.identifier() - self.state = 460 + self.state = 456 self.match(HogQLParser.AS) - self.state = 461 + self.state = 457 self.match(HogQLParser.LPAREN) - self.state = 462 + self.state = 458 self.windowExpr() - self.state = 463 + self.state = 459 self.match(HogQLParser.RPAREN) - self.state = 469 + self.state = 465 self._errHandler.sync(self) _la = self._input.LA(1) @@ -3272,9 +3268,9 @@ def prewhereClause(self): self.enterRule(localctx, 60, self.RULE_prewhereClause) try: self.enterOuterAlt(localctx, 1) - self.state = 470 + self.state = 466 self.match(HogQLParser.PREWHERE) - self.state = 471 + self.state = 467 self.columnExpr(0) except RecognitionException as re: localctx.exception = re @@ -3317,9 +3313,9 @@ def whereClause(self): self.enterRule(localctx, 62, self.RULE_whereClause) try: self.enterOuterAlt(localctx, 1) - self.state = 473 + self.state = 469 self.match(HogQLParser.WHERE) - self.state = 474 + self.state = 470 self.columnExpr(0) except RecognitionException as re: localctx.exception = re @@ -3378,31 +3374,31 @@ def groupByClause(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 476 + self.state = 472 self.match(HogQLParser.GROUP) - self.state = 477 + self.state = 473 self.match(HogQLParser.BY) - self.state = 484 + self.state = 480 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,48,self._ctx) if la_ == 1: - self.state = 478 + self.state = 474 _la = self._input.LA(1) if not(_la==18 or _la==74): self._errHandler.recoverInline(self) else: self._errHandler.reportMatch(self) self.consume() - self.state = 479 + self.state = 475 self.match(HogQLParser.LPAREN) - self.state = 480 + self.state = 476 self.columnExprList() - self.state = 481 + self.state = 477 self.match(HogQLParser.RPAREN) pass elif la_ == 2: - self.state = 483 + self.state = 479 self.columnExprList() pass @@ -3448,9 +3444,9 @@ def havingClause(self): self.enterRule(localctx, 66, self.RULE_havingClause) try: self.enterOuterAlt(localctx, 1) - self.state = 486 + self.state = 482 self.match(HogQLParser.HAVING) - self.state = 487 + self.state = 483 self.columnExpr(0) except RecognitionException as re: localctx.exception = re @@ -3496,11 +3492,11 @@ def orderByClause(self): self.enterRule(localctx, 68, self.RULE_orderByClause) try: self.enterOuterAlt(localctx, 1) - self.state = 489 + self.state = 485 self.match(HogQLParser.ORDER) - self.state = 490 + self.state = 486 self.match(HogQLParser.BY) - self.state = 491 + self.state = 487 self.orderExprList() except RecognitionException as re: localctx.exception = re @@ -3546,11 +3542,11 @@ def projectionOrderByClause(self): self.enterRule(localctx, 70, self.RULE_projectionOrderByClause) try: self.enterOuterAlt(localctx, 1) - self.state = 493 + self.state = 489 self.match(HogQLParser.ORDER) - self.state = 494 + self.state = 490 self.match(HogQLParser.BY) - self.state = 495 + self.state = 491 self.columnExprList() except RecognitionException as re: localctx.exception = re @@ -3615,38 +3611,38 @@ def limitAndOffsetClause(self): self.enterRule(localctx, 72, self.RULE_limitAndOffsetClause) self._la = 0 # Token type try: - self.state = 526 + self.state = 522 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,53,self._ctx) if la_ == 1: self.enterOuterAlt(localctx, 1) - self.state = 497 + self.state = 493 self.match(HogQLParser.LIMIT) - self.state = 498 + self.state = 494 self.columnExpr(0) - self.state = 501 + self.state = 497 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 499 + self.state = 495 self.match(HogQLParser.COMMA) - self.state = 500 + self.state = 496 self.columnExpr(0) - self.state = 507 + self.state = 503 self._errHandler.sync(self) token = self._input.LA(1) if token in [102]: - self.state = 503 + self.state = 499 self.match(HogQLParser.WITH) - self.state = 504 + self.state = 500 self.match(HogQLParser.TIES) pass elif token in [11]: - self.state = 505 + self.state = 501 self.match(HogQLParser.BY) - self.state = 506 + self.state = 502 self.columnExprList() pass elif token in [-1, 81, 95, 149]: @@ -3657,43 +3653,43 @@ def limitAndOffsetClause(self): elif la_ == 2: self.enterOuterAlt(localctx, 2) - self.state = 509 + self.state = 505 self.match(HogQLParser.LIMIT) - self.state = 510 + self.state = 506 self.columnExpr(0) - self.state = 513 + self.state = 509 self._errHandler.sync(self) _la = self._input.LA(1) if _la==102: - self.state = 511 + self.state = 507 self.match(HogQLParser.WITH) - self.state = 512 + self.state = 508 self.match(HogQLParser.TIES) - self.state = 515 + self.state = 511 self.match(HogQLParser.OFFSET) - self.state = 516 + self.state = 512 self.columnExpr(0) pass elif la_ == 3: self.enterOuterAlt(localctx, 3) - self.state = 518 + self.state = 514 self.match(HogQLParser.LIMIT) - self.state = 519 + self.state = 515 self.columnExpr(0) - self.state = 520 + self.state = 516 self.match(HogQLParser.OFFSET) - self.state = 521 + self.state = 517 self.columnExpr(0) - self.state = 524 + self.state = 520 self._errHandler.sync(self) _la = self._input.LA(1) if _la==11: - self.state = 522 + self.state = 518 self.match(HogQLParser.BY) - self.state = 523 + self.state = 519 self.columnExprList() @@ -3741,9 +3737,9 @@ def offsetOnlyClause(self): self.enterRule(localctx, 74, self.RULE_offsetOnlyClause) try: self.enterOuterAlt(localctx, 1) - self.state = 528 + self.state = 524 self.match(HogQLParser.OFFSET) - self.state = 529 + self.state = 525 self.columnExpr(0) except RecognitionException as re: localctx.exception = re @@ -3786,9 +3782,9 @@ def settingsClause(self): self.enterRule(localctx, 76, self.RULE_settingsClause) try: self.enterOuterAlt(localctx, 1) - self.state = 531 + self.state = 527 self.match(HogQLParser.SETTINGS) - self.state = 532 + self.state = 528 self.settingExprList() except RecognitionException as re: localctx.exception = re @@ -3920,7 +3916,7 @@ def joinExpr(self, _p:int=0): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 546 + self.state = 542 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,56,self._ctx) if la_ == 1: @@ -3928,21 +3924,21 @@ def joinExpr(self, _p:int=0): self._ctx = localctx _prevctx = localctx - self.state = 535 + self.state = 531 self.tableExpr(0) - self.state = 537 + self.state = 533 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,54,self._ctx) if la_ == 1: - self.state = 536 + self.state = 532 self.match(HogQLParser.FINAL) - self.state = 540 + self.state = 536 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,55,self._ctx) if la_ == 1: - self.state = 539 + self.state = 535 self.sampleClause() @@ -3952,17 +3948,17 @@ def joinExpr(self, _p:int=0): localctx = HogQLParser.JoinExprParensContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 542 + self.state = 538 self.match(HogQLParser.LPAREN) - self.state = 543 + self.state = 539 self.joinExpr(0) - self.state = 544 + self.state = 540 self.match(HogQLParser.RPAREN) pass self._ctx.stop = self._input.LT(-1) - self.state = 562 + self.state = 558 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,59,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: @@ -3970,47 +3966,47 @@ def joinExpr(self, _p:int=0): if self._parseListeners is not None: self.triggerExitRuleEvent() _prevctx = localctx - self.state = 560 + self.state = 556 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,58,self._ctx) if la_ == 1: localctx = HogQLParser.JoinExprCrossOpContext(self, HogQLParser.JoinExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_joinExpr) - self.state = 548 + self.state = 544 if not self.precpred(self._ctx, 3): from antlr4.error.Errors import FailedPredicateException raise FailedPredicateException(self, "self.precpred(self._ctx, 3)") - self.state = 549 + self.state = 545 self.joinOpCross() - self.state = 550 + self.state = 546 self.joinExpr(4) pass elif la_ == 2: localctx = HogQLParser.JoinExprOpContext(self, HogQLParser.JoinExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_joinExpr) - self.state = 552 + self.state = 548 if not self.precpred(self._ctx, 4): from antlr4.error.Errors import FailedPredicateException raise FailedPredicateException(self, "self.precpred(self._ctx, 4)") - self.state = 554 + self.state = 550 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & 2269426359468314) != 0) or _la==73 or _la==80: - self.state = 553 + self.state = 549 self.joinOp() - self.state = 556 + self.state = 552 self.match(HogQLParser.JOIN) - self.state = 557 + self.state = 553 self.joinExpr(0) - self.state = 558 + self.state = 554 self.joinConstraintClause() pass - self.state = 564 + self.state = 560 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,59,self._ctx) @@ -4121,21 +4117,21 @@ def joinOp(self): self.enterRule(localctx, 80, self.RULE_joinOp) self._la = 0 # Token type try: - self.state = 608 + self.state = 604 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,73,self._ctx) if la_ == 1: localctx = HogQLParser.JoinOpInnerContext(self, localctx) self.enterOuterAlt(localctx, 1) - self.state = 574 + self.state = 570 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,62,self._ctx) if la_ == 1: - self.state = 566 + self.state = 562 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & 274) != 0): - self.state = 565 + self.state = 561 _la = self._input.LA(1) if not((((_la) & ~0x3f) == 0 and ((1 << _la) & 274) != 0)): self._errHandler.recoverInline(self) @@ -4144,18 +4140,18 @@ def joinOp(self): self.consume() - self.state = 568 + self.state = 564 self.match(HogQLParser.INNER) pass elif la_ == 2: - self.state = 569 + self.state = 565 self.match(HogQLParser.INNER) - self.state = 571 + self.state = 567 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & 274) != 0): - self.state = 570 + self.state = 566 _la = self._input.LA(1) if not((((_la) & ~0x3f) == 0 and ((1 << _la) & 274) != 0)): self._errHandler.recoverInline(self) @@ -4167,7 +4163,7 @@ def joinOp(self): pass elif la_ == 3: - self.state = 573 + self.state = 569 _la = self._input.LA(1) if not((((_la) & ~0x3f) == 0 and ((1 << _la) & 274) != 0)): self._errHandler.recoverInline(self) @@ -4182,15 +4178,15 @@ def joinOp(self): elif la_ == 2: localctx = HogQLParser.JoinOpLeftRightContext(self, localctx) self.enterOuterAlt(localctx, 2) - self.state = 590 + self.state = 586 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,67,self._ctx) if la_ == 1: - self.state = 577 + self.state = 573 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & 282) != 0) or _la==80: - self.state = 576 + self.state = 572 _la = self._input.LA(1) if not((((_la) & ~0x3f) == 0 and ((1 << _la) & 282) != 0) or _la==80): self._errHandler.recoverInline(self) @@ -4199,44 +4195,44 @@ def joinOp(self): self.consume() - self.state = 579 + self.state = 575 _la = self._input.LA(1) if not(_la==51 or _la==73): self._errHandler.recoverInline(self) else: self._errHandler.reportMatch(self) self.consume() - self.state = 581 + self.state = 577 self._errHandler.sync(self) _la = self._input.LA(1) if _la==65: - self.state = 580 + self.state = 576 self.match(HogQLParser.OUTER) pass elif la_ == 2: - self.state = 583 + self.state = 579 _la = self._input.LA(1) if not(_la==51 or _la==73): self._errHandler.recoverInline(self) else: self._errHandler.reportMatch(self) self.consume() - self.state = 585 + self.state = 581 self._errHandler.sync(self) _la = self._input.LA(1) if _la==65: - self.state = 584 + self.state = 580 self.match(HogQLParser.OUTER) - self.state = 588 + self.state = 584 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & 282) != 0) or _la==80: - self.state = 587 + self.state = 583 _la = self._input.LA(1) if not((((_la) & ~0x3f) == 0 and ((1 << _la) & 282) != 0) or _la==80): self._errHandler.recoverInline(self) @@ -4253,15 +4249,15 @@ def joinOp(self): elif la_ == 3: localctx = HogQLParser.JoinOpFullContext(self, localctx) self.enterOuterAlt(localctx, 3) - self.state = 606 + self.state = 602 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,72,self._ctx) if la_ == 1: - self.state = 593 + self.state = 589 self._errHandler.sync(self) _la = self._input.LA(1) if _la==1 or _la==4: - self.state = 592 + self.state = 588 _la = self._input.LA(1) if not(_la==1 or _la==4): self._errHandler.recoverInline(self) @@ -4270,34 +4266,34 @@ def joinOp(self): self.consume() - self.state = 595 + self.state = 591 self.match(HogQLParser.FULL) - self.state = 597 + self.state = 593 self._errHandler.sync(self) _la = self._input.LA(1) if _la==65: - self.state = 596 + self.state = 592 self.match(HogQLParser.OUTER) pass elif la_ == 2: - self.state = 599 + self.state = 595 self.match(HogQLParser.FULL) - self.state = 601 + self.state = 597 self._errHandler.sync(self) _la = self._input.LA(1) if _la==65: - self.state = 600 + self.state = 596 self.match(HogQLParser.OUTER) - self.state = 604 + self.state = 600 self._errHandler.sync(self) _la = self._input.LA(1) if _la==1 or _la==4: - self.state = 603 + self.state = 599 _la = self._input.LA(1) if not(_la==1 or _la==4): self._errHandler.recoverInline(self) @@ -4354,19 +4350,19 @@ def joinOpCross(self): localctx = HogQLParser.JoinOpCrossContext(self, self._ctx, self.state) self.enterRule(localctx, 82, self.RULE_joinOpCross) try: - self.state = 613 + self.state = 609 self._errHandler.sync(self) token = self._input.LA(1) if token in [17]: self.enterOuterAlt(localctx, 1) - self.state = 610 + self.state = 606 self.match(HogQLParser.CROSS) - self.state = 611 + self.state = 607 self.match(HogQLParser.JOIN) pass elif token in [116]: self.enterOuterAlt(localctx, 2) - self.state = 612 + self.state = 608 self.match(HogQLParser.COMMA) pass else: @@ -4421,34 +4417,34 @@ def joinConstraintClause(self): localctx = HogQLParser.JoinConstraintClauseContext(self, self._ctx, self.state) self.enterRule(localctx, 84, self.RULE_joinConstraintClause) try: - self.state = 624 + self.state = 620 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,75,self._ctx) if la_ == 1: self.enterOuterAlt(localctx, 1) - self.state = 615 + self.state = 611 self.match(HogQLParser.ON) - self.state = 616 + self.state = 612 self.columnExprList() pass elif la_ == 2: self.enterOuterAlt(localctx, 2) - self.state = 617 + self.state = 613 self.match(HogQLParser.USING) - self.state = 618 + self.state = 614 self.match(HogQLParser.LPAREN) - self.state = 619 + self.state = 615 self.columnExprList() - self.state = 620 + self.state = 616 self.match(HogQLParser.RPAREN) pass elif la_ == 3: self.enterOuterAlt(localctx, 3) - self.state = 622 + self.state = 618 self.match(HogQLParser.USING) - self.state = 623 + self.state = 619 self.columnExprList() pass @@ -4500,17 +4496,17 @@ def sampleClause(self): self.enterRule(localctx, 86, self.RULE_sampleClause) try: self.enterOuterAlt(localctx, 1) - self.state = 626 + self.state = 622 self.match(HogQLParser.SAMPLE) - self.state = 627 + self.state = 623 self.ratioExpr() - self.state = 630 + self.state = 626 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,76,self._ctx) if la_ == 1: - self.state = 628 + self.state = 624 self.match(HogQLParser.OFFSET) - self.state = 629 + self.state = 625 self.ratioExpr() @@ -4562,17 +4558,17 @@ def orderExprList(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 632 + self.state = 628 self.orderExpr() - self.state = 637 + self.state = 633 self._errHandler.sync(self) _la = self._input.LA(1) while _la==116: - self.state = 633 + self.state = 629 self.match(HogQLParser.COMMA) - self.state = 634 + self.state = 630 self.orderExpr() - self.state = 639 + self.state = 635 self._errHandler.sync(self) _la = self._input.LA(1) @@ -4639,13 +4635,13 @@ def orderExpr(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 640 + self.state = 636 self.columnExpr(0) - self.state = 642 + self.state = 638 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & 12583040) != 0): - self.state = 641 + self.state = 637 _la = self._input.LA(1) if not((((_la) & ~0x3f) == 0 and ((1 << _la) & 12583040) != 0)): self._errHandler.recoverInline(self) @@ -4654,13 +4650,13 @@ def orderExpr(self): self.consume() - self.state = 646 + self.state = 642 self._errHandler.sync(self) _la = self._input.LA(1) if _la==60: - self.state = 644 + self.state = 640 self.match(HogQLParser.NULLS) - self.state = 645 + self.state = 641 _la = self._input.LA(1) if not(_la==30 or _la==49): self._errHandler.recoverInline(self) @@ -4669,13 +4665,13 @@ def orderExpr(self): self.consume() - self.state = 650 + self.state = 646 self._errHandler.sync(self) _la = self._input.LA(1) if _la==16: - self.state = 648 + self.state = 644 self.match(HogQLParser.COLLATE) - self.state = 649 + self.state = 645 self.match(HogQLParser.STRING_LITERAL) @@ -4726,25 +4722,25 @@ def ratioExpr(self): localctx = HogQLParser.RatioExprContext(self, self._ctx, self.state) self.enterRule(localctx, 92, self.RULE_ratioExpr) try: - self.state = 658 + self.state = 654 self._errHandler.sync(self) token = self._input.LA(1) if token in [128]: self.enterOuterAlt(localctx, 1) - self.state = 652 + self.state = 648 self.placeholder() pass elif token in [43, 57, 106, 107, 108, 109, 118, 120, 139]: self.enterOuterAlt(localctx, 2) - self.state = 653 + self.state = 649 self.numberLiteral() - self.state = 656 + self.state = 652 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,81,self._ctx) if la_ == 1: - self.state = 654 + self.state = 650 self.match(HogQLParser.SLASH) - self.state = 655 + self.state = 651 self.numberLiteral() @@ -4800,17 +4796,17 @@ def settingExprList(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 660 + self.state = 656 self.settingExpr() - self.state = 665 + self.state = 661 self._errHandler.sync(self) _la = self._input.LA(1) while _la==116: - self.state = 661 + self.state = 657 self.match(HogQLParser.COMMA) - self.state = 662 + self.state = 658 self.settingExpr() - self.state = 667 + self.state = 663 self._errHandler.sync(self) _la = self._input.LA(1) @@ -4859,11 +4855,11 @@ def settingExpr(self): self.enterRule(localctx, 96, self.RULE_settingExpr) try: self.enterOuterAlt(localctx, 1) - self.state = 668 + self.state = 664 self.identifier() - self.state = 669 + self.state = 665 self.match(HogQLParser.EQ_SINGLE) - self.state = 670 + self.state = 666 self.literal() except RecognitionException as re: localctx.exception = re @@ -4912,27 +4908,27 @@ def windowExpr(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 673 + self.state = 669 self._errHandler.sync(self) _la = self._input.LA(1) if _la==67: - self.state = 672 + self.state = 668 self.winPartitionByClause() - self.state = 676 + self.state = 672 self._errHandler.sync(self) _la = self._input.LA(1) if _la==64: - self.state = 675 + self.state = 671 self.winOrderByClause() - self.state = 679 + self.state = 675 self._errHandler.sync(self) _la = self._input.LA(1) if _la==71 or _la==76: - self.state = 678 + self.state = 674 self.winFrameClause() @@ -4980,11 +4976,11 @@ def winPartitionByClause(self): self.enterRule(localctx, 100, self.RULE_winPartitionByClause) try: self.enterOuterAlt(localctx, 1) - self.state = 681 + self.state = 677 self.match(HogQLParser.PARTITION) - self.state = 682 + self.state = 678 self.match(HogQLParser.BY) - self.state = 683 + self.state = 679 self.columnExprList() except RecognitionException as re: localctx.exception = re @@ -5030,11 +5026,11 @@ def winOrderByClause(self): self.enterRule(localctx, 102, self.RULE_winOrderByClause) try: self.enterOuterAlt(localctx, 1) - self.state = 685 + self.state = 681 self.match(HogQLParser.ORDER) - self.state = 686 + self.state = 682 self.match(HogQLParser.BY) - self.state = 687 + self.state = 683 self.orderExprList() except RecognitionException as re: localctx.exception = re @@ -5081,14 +5077,14 @@ def winFrameClause(self): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 689 + self.state = 685 _la = self._input.LA(1) if not(_la==71 or _la==76): self._errHandler.recoverInline(self) else: self._errHandler.reportMatch(self) self.consume() - self.state = 690 + self.state = 686 self.winFrameExtend() except RecognitionException as re: localctx.exception = re @@ -5163,25 +5159,25 @@ def winFrameExtend(self): localctx = HogQLParser.WinFrameExtendContext(self, self._ctx, self.state) self.enterRule(localctx, 106, self.RULE_winFrameExtend) try: - self.state = 698 + self.state = 694 self._errHandler.sync(self) token = self._input.LA(1) if token in [19, 43, 57, 94, 106, 107, 108, 109, 118, 120, 139]: localctx = HogQLParser.FrameStartContext(self, localctx) self.enterOuterAlt(localctx, 1) - self.state = 692 + self.state = 688 self.winFrameBound() pass elif token in [9]: localctx = HogQLParser.FrameBetweenContext(self, localctx) self.enterOuterAlt(localctx, 2) - self.state = 693 + self.state = 689 self.match(HogQLParser.BETWEEN) - self.state = 694 + self.state = 690 self.winFrameBound() - self.state = 695 + self.state = 691 self.match(HogQLParser.AND) - self.state = 696 + self.state = 692 self.winFrameBound() pass else: @@ -5240,41 +5236,41 @@ def winFrameBound(self): self.enterRule(localctx, 108, self.RULE_winFrameBound) try: self.enterOuterAlt(localctx, 1) - self.state = 712 + self.state = 708 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,88,self._ctx) if la_ == 1: - self.state = 700 + self.state = 696 self.match(HogQLParser.CURRENT) - self.state = 701 + self.state = 697 self.match(HogQLParser.ROW) pass elif la_ == 2: - self.state = 702 + self.state = 698 self.match(HogQLParser.UNBOUNDED) - self.state = 703 + self.state = 699 self.match(HogQLParser.PRECEDING) pass elif la_ == 3: - self.state = 704 + self.state = 700 self.match(HogQLParser.UNBOUNDED) - self.state = 705 + self.state = 701 self.match(HogQLParser.FOLLOWING) pass elif la_ == 4: - self.state = 706 + self.state = 702 self.numberLiteral() - self.state = 707 + self.state = 703 self.match(HogQLParser.PRECEDING) pass elif la_ == 5: - self.state = 709 + self.state = 705 self.numberLiteral() - self.state = 710 + self.state = 706 self.match(HogQLParser.FOLLOWING) pass @@ -5320,9 +5316,9 @@ def expr(self): self.enterRule(localctx, 110, self.RULE_expr) try: self.enterOuterAlt(localctx, 1) - self.state = 714 + self.state = 710 self.columnExpr(0) - self.state = 715 + self.state = 711 self.match(HogQLParser.EOF) except RecognitionException as re: localctx.exception = re @@ -5497,138 +5493,138 @@ def columnTypeExpr(self): self.enterRule(localctx, 112, self.RULE_columnTypeExpr) self._la = 0 # Token type try: - self.state = 773 + self.state = 769 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,96,self._ctx) if la_ == 1: localctx = HogQLParser.ColumnTypeExprSimpleContext(self, localctx) self.enterOuterAlt(localctx, 1) - self.state = 717 + self.state = 713 self.identifier() pass elif la_ == 2: localctx = HogQLParser.ColumnTypeExprNestedContext(self, localctx) self.enterOuterAlt(localctx, 2) - self.state = 718 + self.state = 714 self.identifier() - self.state = 719 + self.state = 715 self.match(HogQLParser.LPAREN) - self.state = 720 + self.state = 716 self.identifier() - self.state = 721 + self.state = 717 self.columnTypeExpr() - self.state = 728 + self.state = 724 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,89,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 722 + self.state = 718 self.match(HogQLParser.COMMA) - self.state = 723 + self.state = 719 self.identifier() - self.state = 724 + self.state = 720 self.columnTypeExpr() - self.state = 730 + self.state = 726 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,89,self._ctx) - self.state = 732 + self.state = 728 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 731 + self.state = 727 self.match(HogQLParser.COMMA) - self.state = 734 + self.state = 730 self.match(HogQLParser.RPAREN) pass elif la_ == 3: localctx = HogQLParser.ColumnTypeExprEnumContext(self, localctx) self.enterOuterAlt(localctx, 3) - self.state = 736 + self.state = 732 self.identifier() - self.state = 737 + self.state = 733 self.match(HogQLParser.LPAREN) - self.state = 738 + self.state = 734 self.enumValue() - self.state = 743 + self.state = 739 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,91,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 739 + self.state = 735 self.match(HogQLParser.COMMA) - self.state = 740 + self.state = 736 self.enumValue() - self.state = 745 + self.state = 741 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,91,self._ctx) - self.state = 747 + self.state = 743 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 746 + self.state = 742 self.match(HogQLParser.COMMA) - self.state = 749 + self.state = 745 self.match(HogQLParser.RPAREN) pass elif la_ == 4: localctx = HogQLParser.ColumnTypeExprComplexContext(self, localctx) self.enterOuterAlt(localctx, 4) - self.state = 751 + self.state = 747 self.identifier() - self.state = 752 + self.state = 748 self.match(HogQLParser.LPAREN) - self.state = 753 + self.state = 749 self.columnTypeExpr() - self.state = 758 + self.state = 754 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,93,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 754 + self.state = 750 self.match(HogQLParser.COMMA) - self.state = 755 + self.state = 751 self.columnTypeExpr() - self.state = 760 + self.state = 756 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,93,self._ctx) - self.state = 762 + self.state = 758 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 761 + self.state = 757 self.match(HogQLParser.COMMA) - self.state = 764 + self.state = 760 self.match(HogQLParser.RPAREN) pass elif la_ == 5: localctx = HogQLParser.ColumnTypeExprParamContext(self, localctx) self.enterOuterAlt(localctx, 5) - self.state = 766 + self.state = 762 self.identifier() - self.state = 767 + self.state = 763 self.match(HogQLParser.LPAREN) - self.state = 769 + self.state = 765 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 768 + self.state = 764 self.columnExprList() - self.state = 771 + self.state = 767 self.match(HogQLParser.RPAREN) pass @@ -5680,26 +5676,26 @@ def columnExprList(self): self.enterRule(localctx, 114, self.RULE_columnExprList) try: self.enterOuterAlt(localctx, 1) - self.state = 775 + self.state = 771 self.columnExpr(0) - self.state = 780 + self.state = 776 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,97,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 776 + self.state = 772 self.match(HogQLParser.COMMA) - self.state = 777 + self.state = 773 self.columnExpr(0) - self.state = 782 + self.state = 778 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,97,self._ctx) - self.state = 784 + self.state = 780 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,98,self._ctx) if la_ == 1: - self.state = 783 + self.state = 779 self.match(HogQLParser.COMMA) @@ -6142,6 +6138,8 @@ class ColumnExprWinFunctionTargetContext(ColumnExprContext): def __init__(self, parser, ctx:ParserRuleContext): # actually a HogQLParser.ColumnExprContext super().__init__(parser) + self.columnExprs = None # ColumnExprListContext + self.columnArgList = None # ColumnExprListContext self.copyFrom(ctx) def identifier(self, i:int=None): @@ -6162,14 +6160,14 @@ def RPAREN(self, i:int=None): return self.getTokens(HogQLParser.RPAREN) else: return self.getToken(HogQLParser.RPAREN, i) - def columnExprList(self): - return self.getTypedRuleContext(HogQLParser.ColumnExprListContext,0) + def columnExprList(self, i:int=None): + if i is None: + return self.getTypedRuleContexts(HogQLParser.ColumnExprListContext) + else: + return self.getTypedRuleContext(HogQLParser.ColumnExprListContext,i) def DISTINCT(self): return self.getToken(HogQLParser.DISTINCT, 0) - def columnArgList(self): - return self.getTypedRuleContext(HogQLParser.ColumnArgListContext,0) - def accept(self, visitor:ParseTreeVisitor): if hasattr( visitor, "visitColumnExprWinFunctionTarget" ): @@ -6289,6 +6287,30 @@ def accept(self, visitor:ParseTreeVisitor): return visitor.visitChildren(self) + class ColumnExprCallContext(ColumnExprContext): + + def __init__(self, parser, ctx:ParserRuleContext): # actually a HogQLParser.ColumnExprContext + super().__init__(parser) + self.copyFrom(ctx) + + def columnExpr(self): + return self.getTypedRuleContext(HogQLParser.ColumnExprContext,0) + + def LPAREN(self): + return self.getToken(HogQLParser.LPAREN, 0) + def RPAREN(self): + return self.getToken(HogQLParser.RPAREN, 0) + def columnExprList(self): + return self.getTypedRuleContext(HogQLParser.ColumnExprListContext,0) + + + def accept(self, visitor:ParseTreeVisitor): + if hasattr( visitor, "visitColumnExprCall" ): + return visitor.visitColumnExprCall(self) + else: + return visitor.visitChildren(self) + + class ColumnExprArrayAccessContext(ColumnExprContext): def __init__(self, parser, ctx:ParserRuleContext): # actually a HogQLParser.ColumnExprContext @@ -6572,6 +6594,8 @@ class ColumnExprWinFunctionContext(ColumnExprContext): def __init__(self, parser, ctx:ParserRuleContext): # actually a HogQLParser.ColumnExprContext super().__init__(parser) + self.columnExprs = None # ColumnExprListContext + self.columnArgList = None # ColumnExprListContext self.copyFrom(ctx) def identifier(self): @@ -6592,14 +6616,14 @@ def RPAREN(self, i:int=None): return self.getTokens(HogQLParser.RPAREN) else: return self.getToken(HogQLParser.RPAREN, i) - def columnExprList(self): - return self.getTypedRuleContext(HogQLParser.ColumnExprListContext,0) + def columnExprList(self, i:int=None): + if i is None: + return self.getTypedRuleContexts(HogQLParser.ColumnExprListContext) + else: + return self.getTypedRuleContext(HogQLParser.ColumnExprListContext,i) def DISTINCT(self): return self.getToken(HogQLParser.DISTINCT, 0) - def columnArgList(self): - return self.getTypedRuleContext(HogQLParser.ColumnArgListContext,0) - def accept(self, visitor:ParseTreeVisitor): if hasattr( visitor, "visitColumnExprWinFunction" ): @@ -6608,6 +6632,23 @@ def accept(self, visitor:ParseTreeVisitor): return visitor.visitChildren(self) + class ColumnExprLambdaContext(ColumnExprContext): + + def __init__(self, parser, ctx:ParserRuleContext): # actually a HogQLParser.ColumnExprContext + super().__init__(parser) + self.copyFrom(ctx) + + def columnLambdaExpr(self): + return self.getTypedRuleContext(HogQLParser.ColumnLambdaExprContext,0) + + + def accept(self, visitor:ParseTreeVisitor): + if hasattr( visitor, "visitColumnExprLambda" ): + return visitor.visitColumnExprLambda(self) + else: + return visitor.visitChildren(self) + + class ColumnExprIdentifierContext(ColumnExprContext): def __init__(self, parser, ctx:ParserRuleContext): # actually a HogQLParser.ColumnExprContext @@ -6629,6 +6670,8 @@ class ColumnExprFunctionContext(ColumnExprContext): def __init__(self, parser, ctx:ParserRuleContext): # actually a HogQLParser.ColumnExprContext super().__init__(parser) + self.columnExprs = None # ColumnExprListContext + self.columnArgList = None # ColumnExprListContext self.copyFrom(ctx) def identifier(self): @@ -6646,11 +6689,11 @@ def RPAREN(self, i:int=None): return self.getToken(HogQLParser.RPAREN, i) def DISTINCT(self): return self.getToken(HogQLParser.DISTINCT, 0) - def columnArgList(self): - return self.getTypedRuleContext(HogQLParser.ColumnArgListContext,0) - - def columnExprList(self): - return self.getTypedRuleContext(HogQLParser.ColumnExprListContext,0) + def columnExprList(self, i:int=None): + if i is None: + return self.getTypedRuleContexts(HogQLParser.ColumnExprListContext) + else: + return self.getTypedRuleContext(HogQLParser.ColumnExprListContext,i) def accept(self, visitor:ParseTreeVisitor): @@ -6692,7 +6735,7 @@ def columnExpr(self, _p:int=0): self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 935 + self.state = 932 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,118,self._ctx) if la_ == 1: @@ -6700,45 +6743,45 @@ def columnExpr(self, _p:int=0): self._ctx = localctx _prevctx = localctx - self.state = 787 + self.state = 783 self.match(HogQLParser.CASE) - self.state = 789 + self.state = 785 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,99,self._ctx) if la_ == 1: - self.state = 788 + self.state = 784 localctx.caseExpr = self.columnExpr(0) - self.state = 796 + self.state = 792 self._errHandler.sync(self) _la = self._input.LA(1) while True: - self.state = 791 + self.state = 787 self.match(HogQLParser.WHEN) - self.state = 792 + self.state = 788 localctx.whenExpr = self.columnExpr(0) - self.state = 793 + self.state = 789 self.match(HogQLParser.THEN) - self.state = 794 + self.state = 790 localctx.thenExpr = self.columnExpr(0) - self.state = 798 + self.state = 794 self._errHandler.sync(self) _la = self._input.LA(1) if not (_la==98): break - self.state = 802 + self.state = 798 self._errHandler.sync(self) _la = self._input.LA(1) if _la==25: - self.state = 800 + self.state = 796 self.match(HogQLParser.ELSE) - self.state = 801 + self.state = 797 localctx.elseExpr = self.columnExpr(0) - self.state = 804 + self.state = 800 self.match(HogQLParser.END) pass @@ -6746,17 +6789,17 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprCastContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 806 + self.state = 802 self.match(HogQLParser.CAST) - self.state = 807 + self.state = 803 self.match(HogQLParser.LPAREN) - self.state = 808 + self.state = 804 self.columnExpr(0) - self.state = 809 + self.state = 805 self.match(HogQLParser.AS) - self.state = 810 + self.state = 806 self.columnTypeExpr() - self.state = 811 + self.state = 807 self.match(HogQLParser.RPAREN) pass @@ -6764,9 +6807,9 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprDateContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 813 + self.state = 809 self.match(HogQLParser.DATE) - self.state = 814 + self.state = 810 self.match(HogQLParser.STRING_LITERAL) pass @@ -6774,11 +6817,11 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprIntervalContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 815 + self.state = 811 self.match(HogQLParser.INTERVAL) - self.state = 816 + self.state = 812 self.columnExpr(0) - self.state = 817 + self.state = 813 self.interval() pass @@ -6786,27 +6829,27 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprSubstringContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 819 + self.state = 815 self.match(HogQLParser.SUBSTRING) - self.state = 820 + self.state = 816 self.match(HogQLParser.LPAREN) - self.state = 821 + self.state = 817 self.columnExpr(0) - self.state = 822 + self.state = 818 self.match(HogQLParser.FROM) - self.state = 823 + self.state = 819 self.columnExpr(0) - self.state = 826 + self.state = 822 self._errHandler.sync(self) _la = self._input.LA(1) if _la==33: - self.state = 824 + self.state = 820 self.match(HogQLParser.FOR) - self.state = 825 + self.state = 821 self.columnExpr(0) - self.state = 828 + self.state = 824 self.match(HogQLParser.RPAREN) pass @@ -6814,9 +6857,9 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprTimestampContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 830 + self.state = 826 self.match(HogQLParser.TIMESTAMP) - self.state = 831 + self.state = 827 self.match(HogQLParser.STRING_LITERAL) pass @@ -6824,24 +6867,24 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprTrimContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 832 + self.state = 828 self.match(HogQLParser.TRIM) - self.state = 833 + self.state = 829 self.match(HogQLParser.LPAREN) - self.state = 834 + self.state = 830 _la = self._input.LA(1) if not(_la==10 or _la==50 or _la==90): self._errHandler.recoverInline(self) else: self._errHandler.reportMatch(self) self.consume() - self.state = 835 + self.state = 831 self.string() - self.state = 836 + self.state = 832 self.match(HogQLParser.FROM) - self.state = 837 + self.state = 833 self.columnExpr(0) - self.state = 838 + self.state = 834 self.match(HogQLParser.RPAREN) pass @@ -6849,54 +6892,54 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprWinFunctionContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 840 + self.state = 836 self.identifier() - self.state = 841 + self.state = 837 self.match(HogQLParser.LPAREN) - self.state = 843 + self.state = 839 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 842 - self.columnExprList() + self.state = 838 + localctx.columnExprs = self.columnExprList() - self.state = 845 + self.state = 841 self.match(HogQLParser.RPAREN) - self.state = 855 + self.state = 851 self._errHandler.sync(self) _la = self._input.LA(1) if _la==130: - self.state = 847 + self.state = 843 self.match(HogQLParser.LPAREN) - self.state = 849 + self.state = 845 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,104,self._ctx) if la_ == 1: - self.state = 848 + self.state = 844 self.match(HogQLParser.DISTINCT) - self.state = 852 + self.state = 848 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 851 - self.columnArgList() + self.state = 847 + localctx.columnArgList = self.columnExprList() - self.state = 854 + self.state = 850 self.match(HogQLParser.RPAREN) - self.state = 857 + self.state = 853 self.match(HogQLParser.OVER) - self.state = 858 + self.state = 854 self.match(HogQLParser.LPAREN) - self.state = 859 + self.state = 855 self.windowExpr() - self.state = 860 + self.state = 856 self.match(HogQLParser.RPAREN) pass @@ -6904,50 +6947,50 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprWinFunctionTargetContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 862 + self.state = 858 self.identifier() - self.state = 863 + self.state = 859 self.match(HogQLParser.LPAREN) - self.state = 865 + self.state = 861 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 864 - self.columnExprList() + self.state = 860 + localctx.columnExprs = self.columnExprList() - self.state = 867 + self.state = 863 self.match(HogQLParser.RPAREN) - self.state = 877 + self.state = 873 self._errHandler.sync(self) _la = self._input.LA(1) if _la==130: - self.state = 869 + self.state = 865 self.match(HogQLParser.LPAREN) - self.state = 871 + self.state = 867 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,108,self._ctx) if la_ == 1: - self.state = 870 + self.state = 866 self.match(HogQLParser.DISTINCT) - self.state = 874 + self.state = 870 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 873 - self.columnArgList() + self.state = 869 + localctx.columnArgList = self.columnExprList() - self.state = 876 + self.state = 872 self.match(HogQLParser.RPAREN) - self.state = 879 + self.state = 875 self.match(HogQLParser.OVER) - self.state = 880 + self.state = 876 self.identifier() pass @@ -6955,45 +6998,45 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprFunctionContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 882 + self.state = 878 self.identifier() - self.state = 888 + self.state = 884 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,112,self._ctx) if la_ == 1: - self.state = 883 + self.state = 879 self.match(HogQLParser.LPAREN) - self.state = 885 + self.state = 881 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 884 - self.columnExprList() + self.state = 880 + localctx.columnExprs = self.columnExprList() - self.state = 887 + self.state = 883 self.match(HogQLParser.RPAREN) - self.state = 890 + self.state = 886 self.match(HogQLParser.LPAREN) - self.state = 892 + self.state = 888 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,113,self._ctx) if la_ == 1: - self.state = 891 + self.state = 887 self.match(HogQLParser.DISTINCT) - self.state = 895 + self.state = 891 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 894 - self.columnArgList() + self.state = 890 + localctx.columnArgList = self.columnExprList() - self.state = 897 + self.state = 893 self.match(HogQLParser.RPAREN) pass @@ -7001,7 +7044,7 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprTagElementContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 899 + self.state = 895 self.hogqlxTagElement() pass @@ -7009,7 +7052,7 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprTemplateStringContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 900 + self.state = 896 self.templateString() pass @@ -7017,7 +7060,7 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprLiteralContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 901 + self.state = 897 self.literal() pass @@ -7025,37 +7068,37 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprNegateContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 902 + self.state = 898 self.match(HogQLParser.DASH) - self.state = 903 - self.columnExpr(19) + self.state = 899 + self.columnExpr(20) pass elif la_ == 15: localctx = HogQLParser.ColumnExprNotContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 904 + self.state = 900 self.match(HogQLParser.NOT) - self.state = 905 - self.columnExpr(13) + self.state = 901 + self.columnExpr(14) pass elif la_ == 16: localctx = HogQLParser.ColumnExprAsteriskContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 909 + self.state = 905 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -725088338784043010) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 3229277487103) != 0): - self.state = 906 + self.state = 902 self.tableIdentifier() - self.state = 907 + self.state = 903 self.match(HogQLParser.DOT) - self.state = 911 + self.state = 907 self.match(HogQLParser.ASTERISK) pass @@ -7063,11 +7106,11 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprSubqueryContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 912 + self.state = 908 self.match(HogQLParser.LPAREN) - self.state = 913 + self.state = 909 self.selectUnionStmt() - self.state = 914 + self.state = 910 self.match(HogQLParser.RPAREN) pass @@ -7075,11 +7118,11 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprParensContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 916 + self.state = 912 self.match(HogQLParser.LPAREN) - self.state = 917 + self.state = 913 self.columnExpr(0) - self.state = 918 + self.state = 914 self.match(HogQLParser.RPAREN) pass @@ -7087,11 +7130,11 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprTupleContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 920 + self.state = 916 self.match(HogQLParser.LPAREN) - self.state = 921 + self.state = 917 self.columnExprList() - self.state = 922 + self.state = 918 self.match(HogQLParser.RPAREN) pass @@ -7099,17 +7142,17 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprArrayContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 924 + self.state = 920 self.match(HogQLParser.LBRACKET) - self.state = 926 + self.state = 922 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 925 + self.state = 921 self.columnExprList() - self.state = 928 + self.state = 924 self.match(HogQLParser.RBRACKET) pass @@ -7117,178 +7160,186 @@ def columnExpr(self, _p:int=0): localctx = HogQLParser.ColumnExprDictContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 929 + self.state = 925 self.match(HogQLParser.LBRACE) - self.state = 931 + self.state = 927 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 930 + self.state = 926 self.kvPairList() - self.state = 933 + self.state = 929 self.match(HogQLParser.RBRACE) pass elif la_ == 22: + localctx = HogQLParser.ColumnExprLambdaContext(self, localctx) + self._ctx = localctx + _prevctx = localctx + self.state = 930 + self.columnLambdaExpr() + pass + + elif la_ == 23: localctx = HogQLParser.ColumnExprIdentifierContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 934 + self.state = 931 self.columnIdentifier() pass self._ctx.stop = self._input.LT(-1) - self.state = 1041 + self.state = 1044 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,129,self._ctx) + _alt = self._interp.adaptivePredict(self._input,130,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: if self._parseListeners is not None: self.triggerExitRuleEvent() _prevctx = localctx - self.state = 1039 + self.state = 1042 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,128,self._ctx) + la_ = self._interp.adaptivePredict(self._input,129,self._ctx) if la_ == 1: localctx = HogQLParser.ColumnExprPrecedence1Context(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) localctx.left = _prevctx self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 937 - if not self.precpred(self._ctx, 18): + self.state = 934 + if not self.precpred(self._ctx, 19): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 18)") - self.state = 941 + raise FailedPredicateException(self, "self.precpred(self._ctx, 19)") + self.state = 938 self._errHandler.sync(self) token = self._input.LA(1) if token in [112]: - self.state = 938 + self.state = 935 localctx.operator = self.match(HogQLParser.ASTERISK) pass elif token in [151]: - self.state = 939 + self.state = 936 localctx.operator = self.match(HogQLParser.SLASH) pass elif token in [138]: - self.state = 940 + self.state = 937 localctx.operator = self.match(HogQLParser.PERCENT) pass else: raise NoViableAltException(self) - self.state = 943 - localctx.right = self.columnExpr(19) + self.state = 940 + localctx.right = self.columnExpr(20) pass elif la_ == 2: localctx = HogQLParser.ColumnExprPrecedence2Context(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) localctx.left = _prevctx self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 944 - if not self.precpred(self._ctx, 17): + self.state = 941 + if not self.precpred(self._ctx, 18): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 17)") - self.state = 948 + raise FailedPredicateException(self, "self.precpred(self._ctx, 18)") + self.state = 945 self._errHandler.sync(self) token = self._input.LA(1) if token in [139]: - self.state = 945 + self.state = 942 localctx.operator = self.match(HogQLParser.PLUS) pass elif token in [118]: - self.state = 946 + self.state = 943 localctx.operator = self.match(HogQLParser.DASH) pass elif token in [117]: - self.state = 947 + self.state = 944 localctx.operator = self.match(HogQLParser.CONCAT) pass else: raise NoViableAltException(self) - self.state = 950 - localctx.right = self.columnExpr(18) + self.state = 947 + localctx.right = self.columnExpr(19) pass elif la_ == 3: localctx = HogQLParser.ColumnExprPrecedence3Context(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) localctx.left = _prevctx self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 951 - if not self.precpred(self._ctx, 16): + self.state = 948 + if not self.precpred(self._ctx, 17): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 16)") - self.state = 976 + raise FailedPredicateException(self, "self.precpred(self._ctx, 17)") + self.state = 973 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,124,self._ctx) if la_ == 1: - self.state = 952 + self.state = 949 localctx.operator = self.match(HogQLParser.EQ_DOUBLE) pass elif la_ == 2: - self.state = 953 + self.state = 950 localctx.operator = self.match(HogQLParser.EQ_SINGLE) pass elif la_ == 3: - self.state = 954 + self.state = 951 localctx.operator = self.match(HogQLParser.NOT_EQ) pass elif la_ == 4: - self.state = 955 + self.state = 952 localctx.operator = self.match(HogQLParser.LT_EQ) pass elif la_ == 5: - self.state = 956 + self.state = 953 localctx.operator = self.match(HogQLParser.LT) pass elif la_ == 6: - self.state = 957 + self.state = 954 localctx.operator = self.match(HogQLParser.GT_EQ) pass elif la_ == 7: - self.state = 958 + self.state = 955 localctx.operator = self.match(HogQLParser.GT) pass elif la_ == 8: - self.state = 960 + self.state = 957 self._errHandler.sync(self) _la = self._input.LA(1) if _la==58: - self.state = 959 + self.state = 956 localctx.operator = self.match(HogQLParser.NOT) - self.state = 962 + self.state = 959 self.match(HogQLParser.IN) - self.state = 964 + self.state = 961 self._errHandler.sync(self) la_ = self._interp.adaptivePredict(self._input,122,self._ctx) if la_ == 1: - self.state = 963 + self.state = 960 self.match(HogQLParser.COHORT) pass elif la_ == 9: - self.state = 967 + self.state = 964 self._errHandler.sync(self) _la = self._input.LA(1) if _la==58: - self.state = 966 + self.state = 963 localctx.operator = self.match(HogQLParser.NOT) - self.state = 969 + self.state = 966 _la = self._input.LA(1) if not(_la==41 or _la==53): self._errHandler.recoverInline(self) @@ -7298,247 +7349,268 @@ def columnExpr(self, _p:int=0): pass elif la_ == 10: - self.state = 970 + self.state = 967 localctx.operator = self.match(HogQLParser.REGEX_SINGLE) pass elif la_ == 11: - self.state = 971 + self.state = 968 localctx.operator = self.match(HogQLParser.REGEX_DOUBLE) pass elif la_ == 12: - self.state = 972 + self.state = 969 localctx.operator = self.match(HogQLParser.NOT_REGEX) pass elif la_ == 13: - self.state = 973 + self.state = 970 localctx.operator = self.match(HogQLParser.IREGEX_SINGLE) pass elif la_ == 14: - self.state = 974 + self.state = 971 localctx.operator = self.match(HogQLParser.IREGEX_DOUBLE) pass elif la_ == 15: - self.state = 975 + self.state = 972 localctx.operator = self.match(HogQLParser.NOT_IREGEX) pass - self.state = 978 - localctx.right = self.columnExpr(17) + self.state = 975 + localctx.right = self.columnExpr(18) pass elif la_ == 4: localctx = HogQLParser.ColumnExprNullishContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 979 - if not self.precpred(self._ctx, 14): + self.state = 976 + if not self.precpred(self._ctx, 15): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 14)") - self.state = 980 + raise FailedPredicateException(self, "self.precpred(self._ctx, 15)") + self.state = 977 self.match(HogQLParser.NULLISH) - self.state = 981 - self.columnExpr(15) + self.state = 978 + self.columnExpr(16) pass elif la_ == 5: localctx = HogQLParser.ColumnExprAndContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 982 - if not self.precpred(self._ctx, 12): + self.state = 979 + if not self.precpred(self._ctx, 13): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 12)") - self.state = 983 + raise FailedPredicateException(self, "self.precpred(self._ctx, 13)") + self.state = 980 self.match(HogQLParser.AND) - self.state = 984 - self.columnExpr(13) + self.state = 981 + self.columnExpr(14) pass elif la_ == 6: localctx = HogQLParser.ColumnExprOrContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 985 - if not self.precpred(self._ctx, 11): + self.state = 982 + if not self.precpred(self._ctx, 12): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 11)") - self.state = 986 + raise FailedPredicateException(self, "self.precpred(self._ctx, 12)") + self.state = 983 self.match(HogQLParser.OR) - self.state = 987 - self.columnExpr(12) + self.state = 984 + self.columnExpr(13) pass elif la_ == 7: localctx = HogQLParser.ColumnExprBetweenContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 988 - if not self.precpred(self._ctx, 10): + self.state = 985 + if not self.precpred(self._ctx, 11): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 10)") - self.state = 990 + raise FailedPredicateException(self, "self.precpred(self._ctx, 11)") + self.state = 987 self._errHandler.sync(self) _la = self._input.LA(1) if _la==58: - self.state = 989 + self.state = 986 self.match(HogQLParser.NOT) - self.state = 992 + self.state = 989 self.match(HogQLParser.BETWEEN) - self.state = 993 + self.state = 990 self.columnExpr(0) - self.state = 994 + self.state = 991 self.match(HogQLParser.AND) - self.state = 995 - self.columnExpr(11) + self.state = 992 + self.columnExpr(12) pass elif la_ == 8: localctx = HogQLParser.ColumnExprTernaryOpContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 997 - if not self.precpred(self._ctx, 9): + self.state = 994 + if not self.precpred(self._ctx, 10): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 9)") - self.state = 998 + raise FailedPredicateException(self, "self.precpred(self._ctx, 10)") + self.state = 995 self.match(HogQLParser.QUERY) - self.state = 999 + self.state = 996 self.columnExpr(0) - self.state = 1000 + self.state = 997 self.match(HogQLParser.COLON) - self.state = 1001 - self.columnExpr(9) + self.state = 998 + self.columnExpr(10) pass elif la_ == 9: - localctx = HogQLParser.ColumnExprArrayAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) + localctx = HogQLParser.ColumnExprCallContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 1003 - if not self.precpred(self._ctx, 25): + self.state = 1000 + if not self.precpred(self._ctx, 30): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 25)") - self.state = 1004 - self.match(HogQLParser.LBRACKET) + raise FailedPredicateException(self, "self.precpred(self._ctx, 30)") + self.state = 1001 + self.match(HogQLParser.LPAREN) + self.state = 1003 + self._errHandler.sync(self) + _la = self._input.LA(1) + if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): + self.state = 1002 + self.columnExprList() + + self.state = 1005 - self.columnExpr(0) - self.state = 1006 - self.match(HogQLParser.RBRACKET) + self.match(HogQLParser.RPAREN) pass elif la_ == 10: - localctx = HogQLParser.ColumnExprTupleAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) + localctx = HogQLParser.ColumnExprArrayAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 1008 - if not self.precpred(self._ctx, 24): + self.state = 1006 + if not self.precpred(self._ctx, 26): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 24)") + raise FailedPredicateException(self, "self.precpred(self._ctx, 26)") + self.state = 1007 + self.match(HogQLParser.LBRACKET) + self.state = 1008 + self.columnExpr(0) self.state = 1009 - self.match(HogQLParser.DOT) - self.state = 1010 - self.match(HogQLParser.DECIMAL_LITERAL) + self.match(HogQLParser.RBRACKET) pass elif la_ == 11: - localctx = HogQLParser.ColumnExprPropertyAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) + localctx = HogQLParser.ColumnExprTupleAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) self.state = 1011 - if not self.precpred(self._ctx, 23): + if not self.precpred(self._ctx, 25): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 23)") + raise FailedPredicateException(self, "self.precpred(self._ctx, 25)") self.state = 1012 self.match(HogQLParser.DOT) self.state = 1013 - self.identifier() + self.match(HogQLParser.DECIMAL_LITERAL) pass elif la_ == 12: - localctx = HogQLParser.ColumnExprNullArrayAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) + localctx = HogQLParser.ColumnExprPropertyAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) self.state = 1014 - if not self.precpred(self._ctx, 22): + if not self.precpred(self._ctx, 24): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 22)") + raise FailedPredicateException(self, "self.precpred(self._ctx, 24)") self.state = 1015 - self.match(HogQLParser.NULL_PROPERTY) + self.match(HogQLParser.DOT) self.state = 1016 - self.match(HogQLParser.LBRACKET) - self.state = 1017 - self.columnExpr(0) - self.state = 1018 - self.match(HogQLParser.RBRACKET) + self.identifier() pass elif la_ == 13: - localctx = HogQLParser.ColumnExprNullTupleAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) + localctx = HogQLParser.ColumnExprNullArrayAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 1020 - if not self.precpred(self._ctx, 21): + self.state = 1017 + if not self.precpred(self._ctx, 23): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 21)") - self.state = 1021 + raise FailedPredicateException(self, "self.precpred(self._ctx, 23)") + self.state = 1018 self.match(HogQLParser.NULL_PROPERTY) - self.state = 1022 - self.match(HogQLParser.DECIMAL_LITERAL) + self.state = 1019 + self.match(HogQLParser.LBRACKET) + self.state = 1020 + self.columnExpr(0) + self.state = 1021 + self.match(HogQLParser.RBRACKET) pass elif la_ == 14: - localctx = HogQLParser.ColumnExprNullPropertyAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) + localctx = HogQLParser.ColumnExprNullTupleAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) self.state = 1023 - if not self.precpred(self._ctx, 20): + if not self.precpred(self._ctx, 22): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 20)") + raise FailedPredicateException(self, "self.precpred(self._ctx, 22)") self.state = 1024 self.match(HogQLParser.NULL_PROPERTY) self.state = 1025 - self.identifier() + self.match(HogQLParser.DECIMAL_LITERAL) pass elif la_ == 15: - localctx = HogQLParser.ColumnExprIsNullContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) + localctx = HogQLParser.ColumnExprNullPropertyAccessContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) self.state = 1026 - if not self.precpred(self._ctx, 15): + if not self.precpred(self._ctx, 21): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 15)") + raise FailedPredicateException(self, "self.precpred(self._ctx, 21)") self.state = 1027 - self.match(HogQLParser.IS) + self.match(HogQLParser.NULL_PROPERTY) + self.state = 1028 + self.identifier() + pass + + elif la_ == 16: + localctx = HogQLParser.ColumnExprIsNullContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) + self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) self.state = 1029 + if not self.precpred(self._ctx, 16): + from antlr4.error.Errors import FailedPredicateException + raise FailedPredicateException(self, "self.precpred(self._ctx, 16)") + self.state = 1030 + self.match(HogQLParser.IS) + self.state = 1032 self._errHandler.sync(self) _la = self._input.LA(1) if _la==58: - self.state = 1028 + self.state = 1031 self.match(HogQLParser.NOT) - self.state = 1031 + self.state = 1034 self.match(HogQLParser.NULL_SQL) pass - elif la_ == 16: + elif la_ == 17: localctx = HogQLParser.ColumnExprAliasContext(self, HogQLParser.ColumnExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_columnExpr) - self.state = 1032 - if not self.precpred(self._ctx, 8): + self.state = 1035 + if not self.precpred(self._ctx, 9): from antlr4.error.Errors import FailedPredicateException - raise FailedPredicateException(self, "self.precpred(self._ctx, 8)") - self.state = 1037 + raise FailedPredicateException(self, "self.precpred(self._ctx, 9)") + self.state = 1040 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,127,self._ctx) + la_ = self._interp.adaptivePredict(self._input,128,self._ctx) if la_ == 1: - self.state = 1033 + self.state = 1036 self.match(HogQLParser.AS) - self.state = 1034 + self.state = 1037 self.identifier() pass elif la_ == 2: - self.state = 1035 + self.state = 1038 self.match(HogQLParser.AS) - self.state = 1036 + self.state = 1039 self.match(HogQLParser.STRING_LITERAL) pass @@ -7546,144 +7618,16 @@ def columnExpr(self, _p:int=0): pass - self.state = 1043 - self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,129,self._ctx) - - except RecognitionException as re: - localctx.exception = re - self._errHandler.reportError(self, re) - self._errHandler.recover(self, re) - finally: - self.unrollRecursionContexts(_parentctx) - return localctx - - - class ColumnArgListContext(ParserRuleContext): - __slots__ = 'parser' - - def __init__(self, parser, parent:ParserRuleContext=None, invokingState:int=-1): - super().__init__(parent, invokingState) - self.parser = parser - - def columnArgExpr(self, i:int=None): - if i is None: - return self.getTypedRuleContexts(HogQLParser.ColumnArgExprContext) - else: - return self.getTypedRuleContext(HogQLParser.ColumnArgExprContext,i) - - - def COMMA(self, i:int=None): - if i is None: - return self.getTokens(HogQLParser.COMMA) - else: - return self.getToken(HogQLParser.COMMA, i) - - def getRuleIndex(self): - return HogQLParser.RULE_columnArgList - - def accept(self, visitor:ParseTreeVisitor): - if hasattr( visitor, "visitColumnArgList" ): - return visitor.visitColumnArgList(self) - else: - return visitor.visitChildren(self) - - - - - def columnArgList(self): - - localctx = HogQLParser.ColumnArgListContext(self, self._ctx, self.state) - self.enterRule(localctx, 118, self.RULE_columnArgList) - self._la = 0 # Token type - try: - self.enterOuterAlt(localctx, 1) - self.state = 1044 - self.columnArgExpr() - self.state = 1049 - self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,130,self._ctx) - while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: - if _alt==1: - self.state = 1045 - self.match(HogQLParser.COMMA) - self.state = 1046 - self.columnArgExpr() - self.state = 1051 + self.state = 1046 self._errHandler.sync(self) _alt = self._interp.adaptivePredict(self._input,130,self._ctx) - self.state = 1053 - self._errHandler.sync(self) - _la = self._input.LA(1) - if _la==116: - self.state = 1052 - self.match(HogQLParser.COMMA) - - - except RecognitionException as re: - localctx.exception = re - self._errHandler.reportError(self, re) - self._errHandler.recover(self, re) - finally: - self.exitRule() - return localctx - - - class ColumnArgExprContext(ParserRuleContext): - __slots__ = 'parser' - - def __init__(self, parser, parent:ParserRuleContext=None, invokingState:int=-1): - super().__init__(parent, invokingState) - self.parser = parser - - def columnLambdaExpr(self): - return self.getTypedRuleContext(HogQLParser.ColumnLambdaExprContext,0) - - - def columnExpr(self): - return self.getTypedRuleContext(HogQLParser.ColumnExprContext,0) - - - def getRuleIndex(self): - return HogQLParser.RULE_columnArgExpr - - def accept(self, visitor:ParseTreeVisitor): - if hasattr( visitor, "visitColumnArgExpr" ): - return visitor.visitColumnArgExpr(self) - else: - return visitor.visitChildren(self) - - - - - def columnArgExpr(self): - - localctx = HogQLParser.ColumnArgExprContext(self, self._ctx, self.state) - self.enterRule(localctx, 120, self.RULE_columnArgExpr) - try: - self.state = 1057 - self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,132,self._ctx) - if la_ == 1: - self.enterOuterAlt(localctx, 1) - self.state = 1055 - self.columnLambdaExpr() - pass - - elif la_ == 2: - self.enterOuterAlt(localctx, 2) - self.state = 1056 - self.columnExpr(0) - pass - - except RecognitionException as re: localctx.exception = re self._errHandler.reportError(self, re) self._errHandler.recover(self, re) finally: - self.exitRule() + self.unrollRecursionContexts(_parentctx) return localctx @@ -7697,10 +7641,6 @@ def __init__(self, parser, parent:ParserRuleContext=None, invokingState:int=-1): def ARROW(self): return self.getToken(HogQLParser.ARROW, 0) - def columnExpr(self): - return self.getTypedRuleContext(HogQLParser.ColumnExprContext,0) - - def LPAREN(self): return self.getToken(HogQLParser.LPAREN, 0) @@ -7714,6 +7654,14 @@ def identifier(self, i:int=None): def RPAREN(self): return self.getToken(HogQLParser.RPAREN, 0) + def columnExpr(self): + return self.getTypedRuleContext(HogQLParser.ColumnExprContext,0) + + + def block(self): + return self.getTypedRuleContext(HogQLParser.BlockContext,0) + + def COMMA(self, i:int=None): if i is None: return self.getTokens(HogQLParser.COMMA) @@ -7735,74 +7683,93 @@ def accept(self, visitor:ParseTreeVisitor): def columnLambdaExpr(self): localctx = HogQLParser.ColumnLambdaExprContext(self, self._ctx, self.state) - self.enterRule(localctx, 122, self.RULE_columnLambdaExpr) + self.enterRule(localctx, 118, self.RULE_columnLambdaExpr) self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 1084 + self.state = 1074 self._errHandler.sync(self) - token = self._input.LA(1) - if token in [130]: - self.state = 1059 + la_ = self._interp.adaptivePredict(self._input,135,self._ctx) + if la_ == 1: + self.state = 1047 self.match(HogQLParser.LPAREN) - self.state = 1060 + self.state = 1048 self.identifier() - self.state = 1065 + self.state = 1053 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,133,self._ctx) + _alt = self._interp.adaptivePredict(self._input,131,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 1061 + self.state = 1049 self.match(HogQLParser.COMMA) - self.state = 1062 + self.state = 1050 self.identifier() - self.state = 1067 + self.state = 1055 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,133,self._ctx) + _alt = self._interp.adaptivePredict(self._input,131,self._ctx) - self.state = 1069 + self.state = 1057 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 1068 + self.state = 1056 self.match(HogQLParser.COMMA) - self.state = 1071 + self.state = 1059 self.match(HogQLParser.RPAREN) pass - elif token in [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 30, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 44, 45, 46, 47, 48, 49, 50, 51, 53, 54, 55, 56, 58, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 85, 86, 87, 88, 89, 90, 91, 92, 94, 95, 96, 97, 98, 99, 101, 102, 103, 105]: - self.state = 1073 + + elif la_ == 2: + self.state = 1061 self.identifier() - self.state = 1078 + self.state = 1066 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,135,self._ctx) + _alt = self._interp.adaptivePredict(self._input,133,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 1074 + self.state = 1062 self.match(HogQLParser.COMMA) - self.state = 1075 + self.state = 1063 self.identifier() - self.state = 1080 + self.state = 1068 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,135,self._ctx) + _alt = self._interp.adaptivePredict(self._input,133,self._ctx) - self.state = 1082 + self.state = 1070 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 1081 + self.state = 1069 self.match(HogQLParser.COMMA) pass - else: - raise NoViableAltException(self) - self.state = 1086 + elif la_ == 3: + self.state = 1072 + self.match(HogQLParser.LPAREN) + self.state = 1073 + self.match(HogQLParser.RPAREN) + pass + + + self.state = 1076 self.match(HogQLParser.ARROW) - self.state = 1087 - self.columnExpr(0) + self.state = 1079 + self._errHandler.sync(self) + la_ = self._interp.adaptivePredict(self._input,136,self._ctx) + if la_ == 1: + self.state = 1077 + self.columnExpr(0) + pass + + elif la_ == 2: + self.state = 1078 + self.block() + pass + + except RecognitionException as re: localctx.exception = re self._errHandler.reportError(self, re) @@ -7910,77 +7877,77 @@ def accept(self, visitor:ParseTreeVisitor): def hogqlxTagElement(self): localctx = HogQLParser.HogqlxTagElementContext(self, self._ctx, self.state) - self.enterRule(localctx, 124, self.RULE_hogqlxTagElement) + self.enterRule(localctx, 120, self.RULE_hogqlxTagElement) self._la = 0 # Token type try: - self.state = 1121 + self.state = 1113 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,141,self._ctx) + la_ = self._interp.adaptivePredict(self._input,140,self._ctx) if la_ == 1: localctx = HogQLParser.HogqlxTagElementClosedContext(self, localctx) self.enterOuterAlt(localctx, 1) - self.state = 1089 + self.state = 1081 self.match(HogQLParser.LT) - self.state = 1090 + self.state = 1082 self.identifier() - self.state = 1094 + self.state = 1086 self._errHandler.sync(self) _la = self._input.LA(1) while (((_la) & ~0x3f) == 0 and ((1 << _la) & -725088338784043010) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 3229277487103) != 0): - self.state = 1091 + self.state = 1083 self.hogqlxTagAttribute() - self.state = 1096 + self.state = 1088 self._errHandler.sync(self) _la = self._input.LA(1) - self.state = 1097 + self.state = 1089 self.match(HogQLParser.SLASH) - self.state = 1098 + self.state = 1090 self.match(HogQLParser.GT) pass elif la_ == 2: localctx = HogQLParser.HogqlxTagElementNestedContext(self, localctx) self.enterOuterAlt(localctx, 2) - self.state = 1100 + self.state = 1092 self.match(HogQLParser.LT) - self.state = 1101 + self.state = 1093 self.identifier() - self.state = 1105 + self.state = 1097 self._errHandler.sync(self) _la = self._input.LA(1) while (((_la) & ~0x3f) == 0 and ((1 << _la) & -725088338784043010) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 3229277487103) != 0): - self.state = 1102 + self.state = 1094 self.hogqlxTagAttribute() - self.state = 1107 + self.state = 1099 self._errHandler.sync(self) _la = self._input.LA(1) - self.state = 1108 + self.state = 1100 self.match(HogQLParser.GT) - self.state = 1114 + self.state = 1106 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,140,self._ctx) + la_ = self._interp.adaptivePredict(self._input,139,self._ctx) if la_ == 1: - self.state = 1109 + self.state = 1101 self.hogqlxTagElement() elif la_ == 2: - self.state = 1110 + self.state = 1102 self.match(HogQLParser.LBRACE) - self.state = 1111 + self.state = 1103 self.columnExpr(0) - self.state = 1112 + self.state = 1104 self.match(HogQLParser.RBRACE) - self.state = 1116 + self.state = 1108 self.match(HogQLParser.LT) - self.state = 1117 + self.state = 1109 self.match(HogQLParser.SLASH) - self.state = 1118 + self.state = 1110 self.identifier() - self.state = 1119 + self.state = 1111 self.match(HogQLParser.GT) pass @@ -8037,38 +8004,38 @@ def accept(self, visitor:ParseTreeVisitor): def hogqlxTagAttribute(self): localctx = HogQLParser.HogqlxTagAttributeContext(self, self._ctx, self.state) - self.enterRule(localctx, 126, self.RULE_hogqlxTagAttribute) + self.enterRule(localctx, 122, self.RULE_hogqlxTagAttribute) try: - self.state = 1134 + self.state = 1126 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,142,self._ctx) + la_ = self._interp.adaptivePredict(self._input,141,self._ctx) if la_ == 1: self.enterOuterAlt(localctx, 1) - self.state = 1123 + self.state = 1115 self.identifier() - self.state = 1124 + self.state = 1116 self.match(HogQLParser.EQ_SINGLE) - self.state = 1125 + self.state = 1117 self.string() pass elif la_ == 2: self.enterOuterAlt(localctx, 2) - self.state = 1127 + self.state = 1119 self.identifier() - self.state = 1128 + self.state = 1120 self.match(HogQLParser.EQ_SINGLE) - self.state = 1129 + self.state = 1121 self.match(HogQLParser.LBRACE) - self.state = 1130 + self.state = 1122 self.columnExpr(0) - self.state = 1131 + self.state = 1123 self.match(HogQLParser.RBRACE) pass elif la_ == 3: self.enterOuterAlt(localctx, 3) - self.state = 1133 + self.state = 1125 self.identifier() pass @@ -8117,30 +8084,30 @@ def accept(self, visitor:ParseTreeVisitor): def withExprList(self): localctx = HogQLParser.WithExprListContext(self, self._ctx, self.state) - self.enterRule(localctx, 128, self.RULE_withExprList) + self.enterRule(localctx, 124, self.RULE_withExprList) self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 1136 + self.state = 1128 self.withExpr() - self.state = 1141 + self.state = 1133 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,143,self._ctx) + _alt = self._interp.adaptivePredict(self._input,142,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 1137 + self.state = 1129 self.match(HogQLParser.COMMA) - self.state = 1138 + self.state = 1130 self.withExpr() - self.state = 1143 + self.state = 1135 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,143,self._ctx) + _alt = self._interp.adaptivePredict(self._input,142,self._ctx) - self.state = 1145 + self.state = 1137 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 1144 + self.state = 1136 self.match(HogQLParser.COMMA) @@ -8222,34 +8189,34 @@ def accept(self, visitor:ParseTreeVisitor): def withExpr(self): localctx = HogQLParser.WithExprContext(self, self._ctx, self.state) - self.enterRule(localctx, 130, self.RULE_withExpr) + self.enterRule(localctx, 126, self.RULE_withExpr) try: - self.state = 1157 + self.state = 1149 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,145,self._ctx) + la_ = self._interp.adaptivePredict(self._input,144,self._ctx) if la_ == 1: localctx = HogQLParser.WithExprSubqueryContext(self, localctx) self.enterOuterAlt(localctx, 1) - self.state = 1147 + self.state = 1139 self.identifier() - self.state = 1148 + self.state = 1140 self.match(HogQLParser.AS) - self.state = 1149 + self.state = 1141 self.match(HogQLParser.LPAREN) - self.state = 1150 + self.state = 1142 self.selectUnionStmt() - self.state = 1151 + self.state = 1143 self.match(HogQLParser.RPAREN) pass elif la_ == 2: localctx = HogQLParser.WithExprColumnContext(self, localctx) self.enterOuterAlt(localctx, 2) - self.state = 1153 + self.state = 1145 self.columnExpr(0) - self.state = 1154 + self.state = 1146 self.match(HogQLParser.AS) - self.state = 1155 + self.state = 1147 self.identifier() pass @@ -8300,29 +8267,29 @@ def accept(self, visitor:ParseTreeVisitor): def columnIdentifier(self): localctx = HogQLParser.ColumnIdentifierContext(self, self._ctx, self.state) - self.enterRule(localctx, 132, self.RULE_columnIdentifier) + self.enterRule(localctx, 128, self.RULE_columnIdentifier) try: - self.state = 1166 + self.state = 1158 self._errHandler.sync(self) token = self._input.LA(1) if token in [128]: self.enterOuterAlt(localctx, 1) - self.state = 1159 + self.state = 1151 self.placeholder() pass elif token in [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 30, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 44, 45, 46, 47, 48, 49, 50, 51, 53, 54, 55, 56, 58, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 85, 86, 87, 88, 89, 90, 91, 92, 94, 95, 96, 97, 98, 99, 101, 102, 103, 105]: self.enterOuterAlt(localctx, 2) - self.state = 1163 + self.state = 1155 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,146,self._ctx) + la_ = self._interp.adaptivePredict(self._input,145,self._ctx) if la_ == 1: - self.state = 1160 + self.state = 1152 self.tableIdentifier() - self.state = 1161 + self.state = 1153 self.match(HogQLParser.DOT) - self.state = 1165 + self.state = 1157 self.nestedIdentifier() pass else: @@ -8372,23 +8339,23 @@ def accept(self, visitor:ParseTreeVisitor): def nestedIdentifier(self): localctx = HogQLParser.NestedIdentifierContext(self, self._ctx, self.state) - self.enterRule(localctx, 134, self.RULE_nestedIdentifier) + self.enterRule(localctx, 130, self.RULE_nestedIdentifier) try: self.enterOuterAlt(localctx, 1) - self.state = 1168 + self.state = 1160 self.identifier() - self.state = 1173 + self.state = 1165 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,148,self._ctx) + _alt = self._interp.adaptivePredict(self._input,147,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 1169 + self.state = 1161 self.match(HogQLParser.DOT) - self.state = 1170 + self.state = 1162 self.identifier() - self.state = 1175 + self.state = 1167 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,148,self._ctx) + _alt = self._interp.adaptivePredict(self._input,147,self._ctx) except RecognitionException as re: localctx.exception = re @@ -8535,19 +8502,19 @@ def tableExpr(self, _p:int=0): _parentState = self.state localctx = HogQLParser.TableExprContext(self, self._ctx, _parentState) _prevctx = localctx - _startState = 136 - self.enterRecursionRule(localctx, 136, self.RULE_tableExpr, _p) + _startState = 132 + self.enterRecursionRule(localctx, 132, self.RULE_tableExpr, _p) try: self.enterOuterAlt(localctx, 1) - self.state = 1185 + self.state = 1177 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,149,self._ctx) + la_ = self._interp.adaptivePredict(self._input,148,self._ctx) if la_ == 1: localctx = HogQLParser.TableExprIdentifierContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 1177 + self.state = 1169 self.tableIdentifier() pass @@ -8555,7 +8522,7 @@ def tableExpr(self, _p:int=0): localctx = HogQLParser.TableExprFunctionContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 1178 + self.state = 1170 self.tableFunctionExpr() pass @@ -8563,11 +8530,11 @@ def tableExpr(self, _p:int=0): localctx = HogQLParser.TableExprSubqueryContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 1179 + self.state = 1171 self.match(HogQLParser.LPAREN) - self.state = 1180 + self.state = 1172 self.selectUnionStmt() - self.state = 1181 + self.state = 1173 self.match(HogQLParser.RPAREN) pass @@ -8575,7 +8542,7 @@ def tableExpr(self, _p:int=0): localctx = HogQLParser.TableExprTagContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 1183 + self.state = 1175 self.hogqlxTagElement() pass @@ -8583,15 +8550,15 @@ def tableExpr(self, _p:int=0): localctx = HogQLParser.TableExprPlaceholderContext(self, localctx) self._ctx = localctx _prevctx = localctx - self.state = 1184 + self.state = 1176 self.placeholder() pass self._ctx.stop = self._input.LT(-1) - self.state = 1195 + self.state = 1187 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,151,self._ctx) + _alt = self._interp.adaptivePredict(self._input,150,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: if self._parseListeners is not None: @@ -8599,29 +8566,29 @@ def tableExpr(self, _p:int=0): _prevctx = localctx localctx = HogQLParser.TableExprAliasContext(self, HogQLParser.TableExprContext(self, _parentctx, _parentState)) self.pushNewRecursionContext(localctx, _startState, self.RULE_tableExpr) - self.state = 1187 + self.state = 1179 if not self.precpred(self._ctx, 3): from antlr4.error.Errors import FailedPredicateException raise FailedPredicateException(self, "self.precpred(self._ctx, 3)") - self.state = 1191 + self.state = 1183 self._errHandler.sync(self) token = self._input.LA(1) if token in [20, 30, 39, 48, 105]: - self.state = 1188 + self.state = 1180 self.alias() pass elif token in [6]: - self.state = 1189 + self.state = 1181 self.match(HogQLParser.AS) - self.state = 1190 + self.state = 1182 self.identifier() pass else: raise NoViableAltException(self) - self.state = 1197 + self.state = 1189 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,151,self._ctx) + _alt = self._interp.adaptivePredict(self._input,150,self._ctx) except RecognitionException as re: localctx.exception = re @@ -8668,23 +8635,23 @@ def accept(self, visitor:ParseTreeVisitor): def tableFunctionExpr(self): localctx = HogQLParser.TableFunctionExprContext(self, self._ctx, self.state) - self.enterRule(localctx, 138, self.RULE_tableFunctionExpr) + self.enterRule(localctx, 134, self.RULE_tableFunctionExpr) self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 1198 + self.state = 1190 self.identifier() - self.state = 1199 + self.state = 1191 self.match(HogQLParser.LPAREN) - self.state = 1201 + self.state = 1193 self._errHandler.sync(self) _la = self._input.LA(1) if (((_la) & ~0x3f) == 0 and ((1 << _la) & -4503602311741442) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 90493036243451903) != 0) or ((((_la - 128)) & ~0x3f) == 0 and ((1 << (_la - 128)) & 18455) != 0): - self.state = 1200 + self.state = 1192 self.tableArgList() - self.state = 1203 + self.state = 1195 self.match(HogQLParser.RPAREN) except RecognitionException as re: localctx.exception = re @@ -8728,20 +8695,20 @@ def accept(self, visitor:ParseTreeVisitor): def tableIdentifier(self): localctx = HogQLParser.TableIdentifierContext(self, self._ctx, self.state) - self.enterRule(localctx, 140, self.RULE_tableIdentifier) + self.enterRule(localctx, 136, self.RULE_tableIdentifier) try: self.enterOuterAlt(localctx, 1) - self.state = 1208 + self.state = 1200 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,153,self._ctx) + la_ = self._interp.adaptivePredict(self._input,152,self._ctx) if la_ == 1: - self.state = 1205 + self.state = 1197 self.databaseIdentifier() - self.state = 1206 + self.state = 1198 self.match(HogQLParser.DOT) - self.state = 1210 + self.state = 1202 self.identifier() except RecognitionException as re: localctx.exception = re @@ -8787,30 +8754,30 @@ def accept(self, visitor:ParseTreeVisitor): def tableArgList(self): localctx = HogQLParser.TableArgListContext(self, self._ctx, self.state) - self.enterRule(localctx, 142, self.RULE_tableArgList) + self.enterRule(localctx, 138, self.RULE_tableArgList) self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 1212 + self.state = 1204 self.columnExpr(0) - self.state = 1217 + self.state = 1209 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,154,self._ctx) + _alt = self._interp.adaptivePredict(self._input,153,self._ctx) while _alt!=2 and _alt!=ATN.INVALID_ALT_NUMBER: if _alt==1: - self.state = 1213 + self.state = 1205 self.match(HogQLParser.COMMA) - self.state = 1214 + self.state = 1206 self.columnExpr(0) - self.state = 1219 + self.state = 1211 self._errHandler.sync(self) - _alt = self._interp.adaptivePredict(self._input,154,self._ctx) + _alt = self._interp.adaptivePredict(self._input,153,self._ctx) - self.state = 1221 + self.state = 1213 self._errHandler.sync(self) _la = self._input.LA(1) if _la==116: - self.state = 1220 + self.state = 1212 self.match(HogQLParser.COMMA) @@ -8849,10 +8816,10 @@ def accept(self, visitor:ParseTreeVisitor): def databaseIdentifier(self): localctx = HogQLParser.DatabaseIdentifierContext(self, self._ctx, self.state) - self.enterRule(localctx, 144, self.RULE_databaseIdentifier) + self.enterRule(localctx, 140, self.RULE_databaseIdentifier) try: self.enterOuterAlt(localctx, 1) - self.state = 1223 + self.state = 1215 self.identifier() except RecognitionException as re: localctx.exception = re @@ -8900,22 +8867,22 @@ def accept(self, visitor:ParseTreeVisitor): def floatingLiteral(self): localctx = HogQLParser.FloatingLiteralContext(self, self._ctx, self.state) - self.enterRule(localctx, 146, self.RULE_floatingLiteral) + self.enterRule(localctx, 142, self.RULE_floatingLiteral) self._la = 0 # Token type try: - self.state = 1233 + self.state = 1225 self._errHandler.sync(self) token = self._input.LA(1) if token in [106]: self.enterOuterAlt(localctx, 1) - self.state = 1225 + self.state = 1217 self.match(HogQLParser.FLOATING_LITERAL) pass elif token in [120]: self.enterOuterAlt(localctx, 2) - self.state = 1226 + self.state = 1218 self.match(HogQLParser.DOT) - self.state = 1227 + self.state = 1219 _la = self._input.LA(1) if not(_la==107 or _la==108): self._errHandler.recoverInline(self) @@ -8925,15 +8892,15 @@ def floatingLiteral(self): pass elif token in [108]: self.enterOuterAlt(localctx, 3) - self.state = 1228 + self.state = 1220 self.match(HogQLParser.DECIMAL_LITERAL) - self.state = 1229 + self.state = 1221 self.match(HogQLParser.DOT) - self.state = 1231 + self.state = 1223 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,156,self._ctx) + la_ = self._interp.adaptivePredict(self._input,155,self._ctx) if la_ == 1: - self.state = 1230 + self.state = 1222 _la = self._input.LA(1) if not(_la==107 or _la==108): self._errHandler.recoverInline(self) @@ -9002,15 +8969,15 @@ def accept(self, visitor:ParseTreeVisitor): def numberLiteral(self): localctx = HogQLParser.NumberLiteralContext(self, self._ctx, self.state) - self.enterRule(localctx, 148, self.RULE_numberLiteral) + self.enterRule(localctx, 144, self.RULE_numberLiteral) self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 1236 + self.state = 1228 self._errHandler.sync(self) _la = self._input.LA(1) if _la==118 or _la==139: - self.state = 1235 + self.state = 1227 _la = self._input.LA(1) if not(_la==118 or _la==139): self._errHandler.recoverInline(self) @@ -9019,36 +8986,36 @@ def numberLiteral(self): self.consume() - self.state = 1244 + self.state = 1236 self._errHandler.sync(self) - la_ = self._interp.adaptivePredict(self._input,159,self._ctx) + la_ = self._interp.adaptivePredict(self._input,158,self._ctx) if la_ == 1: - self.state = 1238 + self.state = 1230 self.floatingLiteral() pass elif la_ == 2: - self.state = 1239 + self.state = 1231 self.match(HogQLParser.OCTAL_LITERAL) pass elif la_ == 3: - self.state = 1240 + self.state = 1232 self.match(HogQLParser.DECIMAL_LITERAL) pass elif la_ == 4: - self.state = 1241 + self.state = 1233 self.match(HogQLParser.HEXADECIMAL_LITERAL) pass elif la_ == 5: - self.state = 1242 + self.state = 1234 self.match(HogQLParser.INF) pass elif la_ == 6: - self.state = 1243 + self.state = 1235 self.match(HogQLParser.NAN_SQL) pass @@ -9094,24 +9061,24 @@ def accept(self, visitor:ParseTreeVisitor): def literal(self): localctx = HogQLParser.LiteralContext(self, self._ctx, self.state) - self.enterRule(localctx, 150, self.RULE_literal) + self.enterRule(localctx, 146, self.RULE_literal) try: - self.state = 1249 + self.state = 1241 self._errHandler.sync(self) token = self._input.LA(1) if token in [43, 57, 106, 107, 108, 109, 118, 120, 139]: self.enterOuterAlt(localctx, 1) - self.state = 1246 + self.state = 1238 self.numberLiteral() pass elif token in [110]: self.enterOuterAlt(localctx, 2) - self.state = 1247 + self.state = 1239 self.match(HogQLParser.STRING_LITERAL) pass elif token in [59]: self.enterOuterAlt(localctx, 3) - self.state = 1248 + self.state = 1240 self.match(HogQLParser.NULL_SQL) pass else: @@ -9172,11 +9139,11 @@ def accept(self, visitor:ParseTreeVisitor): def interval(self): localctx = HogQLParser.IntervalContext(self, self._ctx, self.state) - self.enterRule(localctx, 152, self.RULE_interval) + self.enterRule(localctx, 148, self.RULE_interval) self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 1251 + self.state = 1243 _la = self._input.LA(1) if not((((_la) & ~0x3f) == 0 and ((1 << _la) & 108086665936896000) != 0) or ((((_la - 70)) & ~0x3f) == 0 and ((1 << (_la - 70)) & 8724152577) != 0)): self._errHandler.recoverInline(self) @@ -9469,11 +9436,11 @@ def accept(self, visitor:ParseTreeVisitor): def keyword(self): localctx = HogQLParser.KeywordContext(self, self._ctx, self.state) - self.enterRule(localctx, 154, self.RULE_keyword) + self.enterRule(localctx, 150, self.RULE_keyword) self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 1253 + self.state = 1245 _la = self._input.LA(1) if not((((_la) & ~0x3f) == 0 and ((1 << _la) & -833175004720939010) != 0) or ((((_la - 64)) & ~0x3f) == 0 and ((1 << (_la - 64)) & 471908466623) != 0)): self._errHandler.recoverInline(self) @@ -9523,11 +9490,11 @@ def accept(self, visitor:ParseTreeVisitor): def keywordForAlias(self): localctx = HogQLParser.KeywordForAliasContext(self, self._ctx, self.state) - self.enterRule(localctx, 156, self.RULE_keywordForAlias) + self.enterRule(localctx, 152, self.RULE_keywordForAlias) self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 1255 + self.state = 1247 _la = self._input.LA(1) if not((((_la) & ~0x3f) == 0 and ((1 << _la) & 282025807314944) != 0)): self._errHandler.recoverInline(self) @@ -9572,19 +9539,19 @@ def accept(self, visitor:ParseTreeVisitor): def alias(self): localctx = HogQLParser.AliasContext(self, self._ctx, self.state) - self.enterRule(localctx, 158, self.RULE_alias) + self.enterRule(localctx, 154, self.RULE_alias) try: - self.state = 1259 + self.state = 1251 self._errHandler.sync(self) token = self._input.LA(1) if token in [105]: self.enterOuterAlt(localctx, 1) - self.state = 1257 + self.state = 1249 self.match(HogQLParser.IDENTIFIER) pass elif token in [20, 30, 39, 48]: self.enterOuterAlt(localctx, 2) - self.state = 1258 + self.state = 1250 self.keywordForAlias() pass else: @@ -9632,24 +9599,24 @@ def accept(self, visitor:ParseTreeVisitor): def identifier(self): localctx = HogQLParser.IdentifierContext(self, self._ctx, self.state) - self.enterRule(localctx, 160, self.RULE_identifier) + self.enterRule(localctx, 156, self.RULE_identifier) try: - self.state = 1264 + self.state = 1256 self._errHandler.sync(self) token = self._input.LA(1) if token in [105]: self.enterOuterAlt(localctx, 1) - self.state = 1261 + self.state = 1253 self.match(HogQLParser.IDENTIFIER) pass elif token in [21, 38, 55, 56, 70, 78, 97, 103]: self.enterOuterAlt(localctx, 2) - self.state = 1262 + self.state = 1254 self.interval() pass elif token in [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 15, 16, 17, 18, 19, 20, 22, 23, 24, 25, 26, 27, 28, 30, 32, 33, 34, 35, 36, 37, 39, 40, 41, 42, 44, 45, 46, 47, 48, 49, 50, 51, 53, 54, 58, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 71, 72, 73, 74, 75, 76, 77, 79, 80, 81, 82, 83, 85, 86, 87, 88, 89, 90, 91, 92, 94, 95, 96, 98, 99, 101, 102]: self.enterOuterAlt(localctx, 3) - self.state = 1263 + self.state = 1255 self.keyword() pass else: @@ -9697,14 +9664,14 @@ def accept(self, visitor:ParseTreeVisitor): def enumValue(self): localctx = HogQLParser.EnumValueContext(self, self._ctx, self.state) - self.enterRule(localctx, 162, self.RULE_enumValue) + self.enterRule(localctx, 158, self.RULE_enumValue) try: self.enterOuterAlt(localctx, 1) - self.state = 1266 + self.state = 1258 self.string() - self.state = 1267 + self.state = 1259 self.match(HogQLParser.EQ_SINGLE) - self.state = 1268 + self.state = 1260 self.numberLiteral() except RecognitionException as re: localctx.exception = re @@ -9747,14 +9714,14 @@ def accept(self, visitor:ParseTreeVisitor): def placeholder(self): localctx = HogQLParser.PlaceholderContext(self, self._ctx, self.state) - self.enterRule(localctx, 164, self.RULE_placeholder) + self.enterRule(localctx, 160, self.RULE_placeholder) try: self.enterOuterAlt(localctx, 1) - self.state = 1270 + self.state = 1262 self.match(HogQLParser.LBRACE) - self.state = 1271 + self.state = 1263 self.nestedIdentifier() - self.state = 1272 + self.state = 1264 self.match(HogQLParser.RBRACE) except RecognitionException as re: localctx.exception = re @@ -9794,19 +9761,19 @@ def accept(self, visitor:ParseTreeVisitor): def string(self): localctx = HogQLParser.StringContext(self, self._ctx, self.state) - self.enterRule(localctx, 166, self.RULE_string) + self.enterRule(localctx, 162, self.RULE_string) try: - self.state = 1276 + self.state = 1268 self._errHandler.sync(self) token = self._input.LA(1) if token in [110]: self.enterOuterAlt(localctx, 1) - self.state = 1274 + self.state = 1266 self.match(HogQLParser.STRING_LITERAL) pass elif token in [142]: self.enterOuterAlt(localctx, 2) - self.state = 1275 + self.state = 1267 self.templateString() pass else: @@ -9856,23 +9823,23 @@ def accept(self, visitor:ParseTreeVisitor): def templateString(self): localctx = HogQLParser.TemplateStringContext(self, self._ctx, self.state) - self.enterRule(localctx, 168, self.RULE_templateString) + self.enterRule(localctx, 164, self.RULE_templateString) self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 1278 + self.state = 1270 self.match(HogQLParser.QUOTE_SINGLE_TEMPLATE) - self.state = 1282 + self.state = 1274 self._errHandler.sync(self) _la = self._input.LA(1) while _la==156 or _la==157: - self.state = 1279 + self.state = 1271 self.stringContents() - self.state = 1284 + self.state = 1276 self._errHandler.sync(self) _la = self._input.LA(1) - self.state = 1285 + self.state = 1277 self.match(HogQLParser.QUOTE_SINGLE) except RecognitionException as re: localctx.exception = re @@ -9918,23 +9885,23 @@ def accept(self, visitor:ParseTreeVisitor): def stringContents(self): localctx = HogQLParser.StringContentsContext(self, self._ctx, self.state) - self.enterRule(localctx, 170, self.RULE_stringContents) + self.enterRule(localctx, 166, self.RULE_stringContents) try: - self.state = 1292 + self.state = 1284 self._errHandler.sync(self) token = self._input.LA(1) if token in [157]: self.enterOuterAlt(localctx, 1) - self.state = 1287 + self.state = 1279 self.match(HogQLParser.STRING_ESCAPE_TRIGGER) - self.state = 1288 + self.state = 1280 self.columnExpr(0) - self.state = 1289 + self.state = 1281 self.match(HogQLParser.RBRACE) pass elif token in [156]: self.enterOuterAlt(localctx, 2) - self.state = 1291 + self.state = 1283 self.match(HogQLParser.STRING_TEXT) pass else: @@ -9984,23 +9951,23 @@ def accept(self, visitor:ParseTreeVisitor): def fullTemplateString(self): localctx = HogQLParser.FullTemplateStringContext(self, self._ctx, self.state) - self.enterRule(localctx, 172, self.RULE_fullTemplateString) + self.enterRule(localctx, 168, self.RULE_fullTemplateString) self._la = 0 # Token type try: self.enterOuterAlt(localctx, 1) - self.state = 1294 + self.state = 1286 self.match(HogQLParser.QUOTE_SINGLE_TEMPLATE_FULL) - self.state = 1298 + self.state = 1290 self._errHandler.sync(self) _la = self._input.LA(1) while _la==158 or _la==159: - self.state = 1295 + self.state = 1287 self.stringContentsFull() - self.state = 1300 + self.state = 1292 self._errHandler.sync(self) _la = self._input.LA(1) - self.state = 1301 + self.state = 1293 self.match(HogQLParser.EOF) except RecognitionException as re: localctx.exception = re @@ -10046,23 +10013,23 @@ def accept(self, visitor:ParseTreeVisitor): def stringContentsFull(self): localctx = HogQLParser.StringContentsFullContext(self, self._ctx, self.state) - self.enterRule(localctx, 174, self.RULE_stringContentsFull) + self.enterRule(localctx, 170, self.RULE_stringContentsFull) try: - self.state = 1308 + self.state = 1300 self._errHandler.sync(self) token = self._input.LA(1) if token in [159]: self.enterOuterAlt(localctx, 1) - self.state = 1303 + self.state = 1295 self.match(HogQLParser.FULL_STRING_ESCAPE_TRIGGER) - self.state = 1304 + self.state = 1296 self.columnExpr(0) - self.state = 1305 + self.state = 1297 self.match(HogQLParser.RBRACE) pass elif token in [158]: self.enterOuterAlt(localctx, 2) - self.state = 1307 + self.state = 1299 self.match(HogQLParser.FULL_STRING_TEXT) pass else: @@ -10083,7 +10050,7 @@ def sempred(self, localctx:RuleContext, ruleIndex:int, predIndex:int): self._predicates = dict() self._predicates[39] = self.joinExpr_sempred self._predicates[58] = self.columnExpr_sempred - self._predicates[68] = self.tableExpr_sempred + self._predicates[66] = self.tableExpr_sempred pred = self._predicates.get(ruleIndex, None) if pred is None: raise Exception("No predicate with index:" + str(ruleIndex)) @@ -10101,71 +10068,75 @@ def joinExpr_sempred(self, localctx:JoinExprContext, predIndex:int): def columnExpr_sempred(self, localctx:ColumnExprContext, predIndex:int): if predIndex == 2: - return self.precpred(self._ctx, 18) + return self.precpred(self._ctx, 19) if predIndex == 3: - return self.precpred(self._ctx, 17) + return self.precpred(self._ctx, 18) if predIndex == 4: - return self.precpred(self._ctx, 16) + return self.precpred(self._ctx, 17) if predIndex == 5: - return self.precpred(self._ctx, 14) + return self.precpred(self._ctx, 15) if predIndex == 6: - return self.precpred(self._ctx, 12) + return self.precpred(self._ctx, 13) if predIndex == 7: - return self.precpred(self._ctx, 11) + return self.precpred(self._ctx, 12) if predIndex == 8: - return self.precpred(self._ctx, 10) + return self.precpred(self._ctx, 11) if predIndex == 9: - return self.precpred(self._ctx, 9) + return self.precpred(self._ctx, 10) if predIndex == 10: - return self.precpred(self._ctx, 25) + return self.precpred(self._ctx, 30) if predIndex == 11: - return self.precpred(self._ctx, 24) + return self.precpred(self._ctx, 26) if predIndex == 12: - return self.precpred(self._ctx, 23) + return self.precpred(self._ctx, 25) if predIndex == 13: - return self.precpred(self._ctx, 22) + return self.precpred(self._ctx, 24) if predIndex == 14: - return self.precpred(self._ctx, 21) + return self.precpred(self._ctx, 23) if predIndex == 15: - return self.precpred(self._ctx, 20) + return self.precpred(self._ctx, 22) if predIndex == 16: - return self.precpred(self._ctx, 15) + return self.precpred(self._ctx, 21) if predIndex == 17: - return self.precpred(self._ctx, 8) + return self.precpred(self._ctx, 16) - def tableExpr_sempred(self, localctx:TableExprContext, predIndex:int): if predIndex == 18: + return self.precpred(self._ctx, 9) + + + def tableExpr_sempred(self, localctx:TableExprContext, predIndex:int): + if predIndex == 19: return self.precpred(self._ctx, 3) diff --git a/posthog/hogql/grammar/HogQLParserVisitor.py b/posthog/hogql/grammar/HogQLParserVisitor.py index bcd327d6abc68..be0ba9f0e29da 100644 --- a/posthog/hogql/grammar/HogQLParserVisitor.py +++ b/posthog/hogql/grammar/HogQLParserVisitor.py @@ -1,4 +1,4 @@ -# Generated from HogQLParser.g4 by ANTLR 4.13.1 +# Generated from HogQLParser.g4 by ANTLR 4.13.2 from antlr4 import * if "." in __name__: from .HogQLParser import HogQLParser @@ -459,6 +459,11 @@ def visitColumnExprTuple(self, ctx:HogQLParser.ColumnExprTupleContext): return self.visitChildren(ctx) + # Visit a parse tree produced by HogQLParser#ColumnExprCall. + def visitColumnExprCall(self, ctx:HogQLParser.ColumnExprCallContext): + return self.visitChildren(ctx) + + # Visit a parse tree produced by HogQLParser#ColumnExprArrayAccess. def visitColumnExprArrayAccess(self, ctx:HogQLParser.ColumnExprArrayAccessContext): return self.visitChildren(ctx) @@ -524,6 +529,11 @@ def visitColumnExprWinFunction(self, ctx:HogQLParser.ColumnExprWinFunctionContex return self.visitChildren(ctx) + # Visit a parse tree produced by HogQLParser#ColumnExprLambda. + def visitColumnExprLambda(self, ctx:HogQLParser.ColumnExprLambdaContext): + return self.visitChildren(ctx) + + # Visit a parse tree produced by HogQLParser#ColumnExprIdentifier. def visitColumnExprIdentifier(self, ctx:HogQLParser.ColumnExprIdentifierContext): return self.visitChildren(ctx) @@ -539,16 +549,6 @@ def visitColumnExprAsterisk(self, ctx:HogQLParser.ColumnExprAsteriskContext): return self.visitChildren(ctx) - # Visit a parse tree produced by HogQLParser#columnArgList. - def visitColumnArgList(self, ctx:HogQLParser.ColumnArgListContext): - return self.visitChildren(ctx) - - - # Visit a parse tree produced by HogQLParser#columnArgExpr. - def visitColumnArgExpr(self, ctx:HogQLParser.ColumnArgExprContext): - return self.visitChildren(ctx) - - # Visit a parse tree produced by HogQLParser#columnLambdaExpr. def visitColumnLambdaExpr(self, ctx:HogQLParser.ColumnLambdaExprContext): return self.visitChildren(ctx) diff --git a/posthog/hogql/grammar/README.md b/posthog/hogql/grammar/README.md index 4022ba71825a7..1b9837a2077ac 100644 --- a/posthog/hogql/grammar/README.md +++ b/posthog/hogql/grammar/README.md @@ -2,16 +2,18 @@ Grammar is located inside `HogQLLexer.g4` and `HogQLParser.g4` files. -To generate source code you need to install locally the `antlr` binary: +To generate source code you need to install locally the `antlr` binary. Run this on macOS: ```bash brew install antlr ``` -or this piece of art if you're using bash on ubuntu for quick access: +In case this installs a newer version than 4.13.2, update [ci-hog.yml](https://github.com/PostHog/posthog/blob/master/.github/workflows/ci-hog.yml) to reflect the changes. + +Run this if you're using bash on ubuntu: ```bash -export ANTLR_VERSION=4.11.1 +export ANTLR_VERSION=4.13.2 sudo apt-get install default-jre mkdir antlr diff --git a/posthog/hogql/parser.py b/posthog/hogql/parser.py index 96afe5ae15bc5..71708d1987032 100644 --- a/posthog/hogql/parser.py +++ b/posthog/hogql/parser.py @@ -907,16 +907,16 @@ def visitColumnExprNot(self, ctx: HogQLParser.ColumnExprNotContext): def visitColumnExprWinFunctionTarget(self, ctx: HogQLParser.ColumnExprWinFunctionTargetContext): return ast.WindowFunction( name=self.visit(ctx.identifier(0)), - exprs=self.visit(ctx.columnExprList()) if ctx.columnExprList() else [], - args=self.visit(ctx.columnArgList()) if ctx.columnArgList() else [], + exprs=self.visit(ctx.columnExprs) if ctx.columnExprs else [], + args=self.visit(ctx.columnArgList) if ctx.columnArgList else [], over_identifier=self.visit(ctx.identifier(1)), ) def visitColumnExprWinFunction(self, ctx: HogQLParser.ColumnExprWinFunctionContext): return ast.WindowFunction( name=self.visit(ctx.identifier()), - exprs=self.visit(ctx.columnExprList()) if ctx.columnExprList() else [], - args=self.visit(ctx.columnArgList()) if ctx.columnArgList() else [], + exprs=self.visit(ctx.columnExprs) if ctx.columnExprs else [], + args=self.visit(ctx.columnArgList) if ctx.columnArgList else [], over_expr=self.visit(ctx.windowExpr()) if ctx.windowExpr() else None, ) @@ -925,10 +925,13 @@ def visitColumnExprIdentifier(self, ctx: HogQLParser.ColumnExprIdentifierContext def visitColumnExprFunction(self, ctx: HogQLParser.ColumnExprFunctionContext): name = self.visit(ctx.identifier()) - column_expr_list = ctx.columnExprList() - parameters = self.visit(column_expr_list) if column_expr_list is not None else None - column_arg_list = ctx.columnArgList() - args = self.visit(column_arg_list) if column_arg_list is not None else [] + + parameters: list[ast.Expr] | None = self.visit(ctx.columnExprs) if ctx.columnExprs is not None else None + # two sets of parameters fn()(), return an empty list for the first even if no parameters + if ctx.LPAREN(1) and parameters is None: + parameters = [] + + args: list[ast.Expr] = self.visit(ctx.columnArgList) if ctx.columnArgList is not None else [] distinct = True if ctx.DISTINCT() else False return ast.Call(name=name, params=parameters, args=args, distinct=distinct) @@ -941,16 +944,10 @@ def visitColumnExprAsterisk(self, ctx: HogQLParser.ColumnExprAsteriskContext): def visitColumnExprTagElement(self, ctx: HogQLParser.ColumnExprTagElementContext): return self.visit(ctx.hogqlxTagElement()) - def visitColumnArgList(self, ctx: HogQLParser.ColumnArgListContext): - return [self.visit(arg) for arg in ctx.columnArgExpr()] - - def visitColumnArgExpr(self, ctx: HogQLParser.ColumnArgExprContext): - return self.visitChildren(ctx) - def visitColumnLambdaExpr(self, ctx: HogQLParser.ColumnLambdaExprContext): return ast.Lambda( args=[self.visit(identifier) for identifier in ctx.identifier()], - expr=self.visit(ctx.columnExpr()), + expr=self.visit(ctx.columnExpr() or ctx.block()), ) def visitWithExprList(self, ctx: HogQLParser.WithExprListContext): @@ -1084,6 +1081,11 @@ def visitColumnExprNullish(self, ctx: HogQLParser.ColumnExprNullishContext): args=[self.visit(ctx.columnExpr(0)), self.visit(ctx.columnExpr(1))], ) + def visitColumnExprCall(self, ctx: HogQLParser.ColumnExprCallContext): + return ast.ExprCall( + expr=self.visit(ctx.columnExpr()), args=self.visit(ctx.columnExprList()) if ctx.columnExprList() else [] + ) + def visitHogqlxTagElementClosed(self, ctx: HogQLParser.HogqlxTagElementClosedContext): kind = self.visit(ctx.identifier()) attributes = [self.visit(a) for a in ctx.hogqlxTagAttribute()] if ctx.hogqlxTagAttribute() else [] diff --git a/posthog/hogql/property.py b/posthog/hogql/property.py index c292a89fb2e07..bce1453e561f8 100644 --- a/posthog/hogql/property.py +++ b/posthog/hogql/property.py @@ -275,7 +275,7 @@ def property_to_expr( | DataWarehousePersonPropertyFilter ), team: Team, - scope: Literal["event", "person", "session", "replay", "replay_entity", "replay_pdi"] = "event", + scope: Literal["event", "person", "session", "replay", "replay_entity"] = "event", ) -> ast.Expr: if isinstance(property, dict): try: @@ -345,6 +345,7 @@ def property_to_expr( or property.type == "data_warehouse" or property.type == "data_warehouse_person_property" or property.type == "session" + or property.type == "recording" or property.type == "log_entry" ): if (scope == "person" and property.type != "person") or (scope == "session" and property.type != "session"): @@ -367,7 +368,7 @@ def property_to_expr( raise QueryError("Data warehouse person property filter value must be a string") elif property.type == "group": chain = [f"group_{property.group_type_index}", "properties"] - elif property.type in ["data_warehouse", "log_entry"]: + elif property.type in ["recording", "data_warehouse", "log_entry"]: chain = [] elif property.type == "session" and scope in ["event", "replay"]: chain = ["session"] @@ -376,7 +377,11 @@ def property_to_expr( else: chain = ["properties"] - expr: ast.Expr = ast.Field(chain=[*chain, property.key]) + field = ast.Field(chain=[*chain, property.key]) + expr: ast.Expr = field + + if property.type == "recording" and property.key == "snapshot_source": + expr = ast.Call(name="argMinMerge", args=[field]) if isinstance(value, list): if len(value) == 0: diff --git a/posthog/hogql/resolver.py b/posthog/hogql/resolver.py index 332e81324871f..54b5f5f298756 100644 --- a/posthog/hogql/resolver.py +++ b/posthog/hogql/resolver.py @@ -112,7 +112,7 @@ def __init__( self.database = context.database self.cte_counter = 0 - def visit(self, node: ast.Expr | None) -> ast.Expr: + def visit(self, node: ast.AST | None): if isinstance(node, ast.Expr) and node.type is not None: raise ResolutionError( f"Type already resolved for {type(node).__name__} ({type(node.type).__name__}). Can't run again." @@ -518,6 +518,12 @@ def visit_call(self, node: ast.Call): ) return node + def visit_expr_call(self, node: ast.ExprCall): + raise QueryError("You can only call simple functions in HogQL, not expressions") + + def visit_block(self, node: ast.Block): + raise QueryError("You can not use blocks in HogQL") + def visit_lambda(self, node: ast.Lambda): """Visit each SELECT query or subquery.""" diff --git a/posthog/hogql/test/_test_parser.py b/posthog/hogql/test/_test_parser.py index b1140be100b5d..280ee65ecf3cf 100644 --- a/posthog/hogql/test/_test_parser.py +++ b/posthog/hogql/test/_test_parser.py @@ -199,6 +199,50 @@ def test_tuples(self): self.assertEqual(self._expr("(1)"), ast.Constant(value=1)) def test_lambdas(self): + self.assertEqual( + self._expr("(x, y) -> x * y"), + ast.Lambda( + args=["x", "y"], + expr=ast.ArithmeticOperation( + op=ast.ArithmeticOperationOp.Mult, + left=ast.Field(chain=["x"]), + right=ast.Field(chain=["y"]), + ), + ), + ) + self.assertEqual( + self._expr("x, y -> x * y"), + ast.Lambda( + args=["x", "y"], + expr=ast.ArithmeticOperation( + op=ast.ArithmeticOperationOp.Mult, + left=ast.Field(chain=["x"]), + right=ast.Field(chain=["y"]), + ), + ), + ) + self.assertEqual( + self._expr("(x) -> x * y"), + ast.Lambda( + args=["x"], + expr=ast.ArithmeticOperation( + op=ast.ArithmeticOperationOp.Mult, + left=ast.Field(chain=["x"]), + right=ast.Field(chain=["y"]), + ), + ), + ) + self.assertEqual( + self._expr("x -> x * y"), + ast.Lambda( + args=["x"], + expr=ast.ArithmeticOperation( + op=ast.ArithmeticOperationOp.Mult, + left=ast.Field(chain=["x"]), + right=ast.Field(chain=["y"]), + ), + ), + ) self.assertEqual( self._expr("arrayMap(x -> x * 2)"), ast.Call( @@ -248,6 +292,54 @@ def test_lambdas(self): ), ) + def test_lambda_blocks(self): + self.assertEqual( + self._expr("(x, y) -> { print('hello'); return x * y }"), + ast.Lambda( + args=["x", "y"], + expr=ast.Block( + declarations=[ + ast.ExprStatement(expr=ast.Call(name="print", args=[ast.Constant(value="hello")])), + ast.ReturnStatement( + expr=ast.ArithmeticOperation( + op=ast.ArithmeticOperationOp.Mult, + left=ast.Field(chain=["x"]), + right=ast.Field(chain=["y"]), + ) + ), + ] + ), + ), + ) + + def test_call_expr(self): + self.assertEqual( + self._expr("asd.asd(123)"), + ast.ExprCall( + expr=ast.Field(chain=["asd", "asd"]), + args=[ast.Constant(value=123)], + ), + ) + self.assertEqual( + self._expr("asd['asd'](123)"), + ast.ExprCall( + expr=ast.ArrayAccess(array=ast.Field(chain=["asd"]), property=ast.Constant(value="asd")), + args=[ast.Constant(value=123)], + ), + ) + self.assertEqual( + self._expr("(x -> x * 2)(3)"), + ast.ExprCall( + expr=ast.Lambda( + args=["x"], + expr=ast.ArithmeticOperation( + op=ast.ArithmeticOperationOp.Mult, left=ast.Field(chain=["x"]), right=ast.Constant(value=2) + ), + ), + args=[ast.Constant(value=3)], + ), + ) + def test_strings(self): self.assertEqual(self._expr("'null'"), ast.Constant(value="null")) self.assertEqual(self._expr("'n''ull'"), ast.Constant(value="n'ull")) diff --git a/posthog/hogql/test/test_metadata.py b/posthog/hogql/test/test_metadata.py index 1c0eb3ad20377..e19778e6f3f20 100644 --- a/posthog/hogql/test/test_metadata.py +++ b/posthog/hogql/test/test_metadata.py @@ -2,7 +2,7 @@ from posthog.hogql.metadata import get_hogql_metadata from posthog.models import PropertyDefinition, Cohort -from posthog.schema import HogQLMetadata, HogQLMetadataResponse, HogQLQuery +from posthog.schema import HogQLMetadata, HogQLMetadataResponse, HogQLQuery, HogLanguage from posthog.test.base import APIBaseTest, ClickhouseTestMixin from django.test import override_settings @@ -14,7 +14,7 @@ def _expr(self, query: str, table: str = "events", debug=True) -> HogQLMetadataR return get_hogql_metadata( query=HogQLMetadata( kind="HogQLMetadata", - language="hogQLExpr", + language=HogLanguage.HOG_QL_EXPR, query=query, sourceQuery=HogQLQuery(query=f"select * from {table}"), response=None, @@ -25,19 +25,21 @@ def _expr(self, query: str, table: str = "events", debug=True) -> HogQLMetadataR def _select(self, query: str) -> HogQLMetadataResponse: return get_hogql_metadata( - query=HogQLMetadata(kind="HogQLMetadata", language="hogQL", query=query, response=None), + query=HogQLMetadata(kind="HogQLMetadata", language=HogLanguage.HOG_QL, query=query, response=None), team=self.team, ) def _program(self, query: str, globals: Optional[dict] = None) -> HogQLMetadataResponse: return get_hogql_metadata( - query=HogQLMetadata(kind="HogQLMetadata", language="hog", query=query, globals=globals, response=None), + query=HogQLMetadata( + kind="HogQLMetadata", language=HogLanguage.HOG, query=query, globals=globals, response=None + ), team=self.team, ) def _template(self, query: str) -> HogQLMetadataResponse: return get_hogql_metadata( - query=HogQLMetadata(kind="HogQLMetadata", language="hogTemplate", query=query, response=None), + query=HogQLMetadata(kind="HogQLMetadata", language=HogLanguage.HOG_TEMPLATE, query=query, response=None), team=self.team, ) diff --git a/posthog/hogql/test/test_printer.py b/posthog/hogql/test/test_printer.py index 83efcb2aa2129..9088ed73a88d1 100644 --- a/posthog/hogql/test/test_printer.py +++ b/posthog/hogql/test/test_printer.py @@ -398,7 +398,7 @@ def test_expr_parse_errors(self): ) self._assert_expr_error( "quantile()(event)", - "Aggregation 'quantile' requires parameters in addition to arguments", + "Aggregation 'quantile' expects 1 parameter, found 0", ) self._assert_expr_error( "quantile(0.5, 2)(event)", @@ -444,12 +444,13 @@ def test_expr_parse_errors_poe_off(self): def test_expr_syntax_errors(self): self._assert_expr_error("(", "no viable alternative at input '('") - self._assert_expr_error("())", "no viable alternative at input '()'") + self._assert_expr_error("())", "mismatched input ')' expecting '->'") self._assert_expr_error("(3 57", "no viable alternative at input '(3 57'") self._assert_expr_error("select query from events", "mismatched input 'query' expecting ") self._assert_expr_error("this makes little sense", "mismatched input 'makes' expecting ") self._assert_expr_error("1;2", "mismatched input ';' expecting ") - self._assert_expr_error("b.a(bla)", "mismatched input '(' expecting ") + self._assert_expr_error("b.a(bla)", "You can only call simple functions in HogQL, not expressions") + self._assert_expr_error("a -> { print(2) }", "You can not use blocks in HogQL") def test_logic(self): self.assertEqual( diff --git a/posthog/hogql/visitor.py b/posthog/hogql/visitor.py index 025ecc5f52e92..2a628181ae225 100644 --- a/posthog/hogql/visitor.py +++ b/posthog/hogql/visitor.py @@ -102,6 +102,11 @@ def visit_call(self, node: ast.Call): for expr in node.params: self.visit(expr) + def visit_expr_call(self, node: ast.ExprCall): + self.visit(node.expr) + for expr in node.args: + self.visit(expr) + def visit_sample_expr(self, node: ast.SampleExpr): self.visit(node.sample_value) self.visit(node.offset_value) @@ -507,6 +512,15 @@ def visit_call(self, node: ast.Call): distinct=node.distinct, ) + def visit_expr_call(self, node: ast.ExprCall): + return ast.ExprCall( + start=None if self.clear_locations else node.start, + end=None if self.clear_locations else node.end, + type=None if self.clear_types else node.type, + expr=self.visit(node.expr), + args=[self.visit(arg) for arg in node.args], + ) + def visit_ratio_expr(self, node: ast.RatioExpr): return ast.RatioExpr( start=None if self.clear_locations else node.start, diff --git a/posthog/management/commands/test_migrations_are_safe.py b/posthog/management/commands/test_migrations_are_safe.py index 0a921cc98d2da..41ef0df6f90db 100644 --- a/posthog/management/commands/test_migrations_are_safe.py +++ b/posthog/management/commands/test_migrations_are_safe.py @@ -42,6 +42,8 @@ def validate_migration_sql(sql) -> bool: and "CREATE TABLE" not in operation_sql and "ADD CONSTRAINT" not in operation_sql and "-- not-null-ignore" not in operation_sql + # Ignore for brand-new tables + and (table_being_altered not in tables_created_so_far or table_being_altered not in new_tables) ): print( f"\n\n\033[91mFound a non-null field or default added to an existing model. This will lock up the table while migrating. Please add 'null=True, blank=True' to the field.\nSource: `{operation_sql}`" diff --git a/posthog/migrations/0460_alertconfiguration_threshold_alertsubscription_and_more.py b/posthog/migrations/0460_alertconfiguration_threshold_alertsubscription_and_more.py new file mode 100644 index 0000000000000..f53046b41046f --- /dev/null +++ b/posthog/migrations/0460_alertconfiguration_threshold_alertsubscription_and_more.py @@ -0,0 +1,161 @@ +# Generated by Django 4.2.14 on 2024-08-26 10:25 + +from django.conf import settings +from django.db import migrations, models +import django.db.models.deletion +import django.db.models.expressions +import posthog.models.utils + + +class Migration(migrations.Migration): + dependencies = [ + ("posthog", "0459_convert_personsnode_insights_to_actorsquery"), + ] + + operations = [ + migrations.CreateModel( + name="AlertConfiguration", + fields=[ + ("created_at", models.DateTimeField(auto_now_add=True)), + ( + "id", + models.UUIDField( + default=posthog.models.utils.UUIDT, editable=False, primary_key=True, serialize=False + ), + ), + ("name", models.CharField(blank=True, max_length=255)), + ("condition", models.JSONField(default=dict)), + ( + "state", + models.CharField( + choices=[("firing", "Firing"), ("inactive", "Inactive")], default="inactive", max_length=10 + ), + ), + ("enabled", models.BooleanField(default=True)), + ("last_notified_at", models.DateTimeField(blank=True, null=True)), + ( + "created_by", + models.ForeignKey( + blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL + ), + ), + ("insight", models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to="posthog.insight")), + ], + options={ + "abstract": False, + }, + ), + migrations.CreateModel( + name="Threshold", + fields=[ + ("created_at", models.DateTimeField(auto_now_add=True)), + ( + "id", + models.UUIDField( + default=posthog.models.utils.UUIDT, editable=False, primary_key=True, serialize=False + ), + ), + ("name", models.CharField(blank=True, max_length=255)), + ("configuration", models.JSONField(default=dict)), + ( + "created_by", + models.ForeignKey( + blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL + ), + ), + ("insight", models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to="posthog.insight")), + ("team", models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to="posthog.team")), + ], + options={ + "abstract": False, + }, + ), + migrations.CreateModel( + name="AlertSubscription", + fields=[ + ("created_at", models.DateTimeField(auto_now_add=True)), + ( + "id", + models.UUIDField( + default=posthog.models.utils.UUIDT, editable=False, primary_key=True, serialize=False + ), + ), + ("subscribed", models.BooleanField(default=True)), + ( + "alert_configuration", + models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to="posthog.alertconfiguration"), + ), + ( + "created_by", + models.ForeignKey( + blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL + ), + ), + ( + "user", + models.ForeignKey( + limit_choices_to={ + "is_active": True, + "organization_id": django.db.models.expressions.OuterRef( + "alert_configuration__team__organization_id" + ), + }, + on_delete=django.db.models.deletion.CASCADE, + related_name="alert_subscriptions", + to=settings.AUTH_USER_MODEL, + ), + ), + ], + options={ + "unique_together": {("user", "alert_configuration")}, + }, + ), + migrations.AddField( + model_name="alertconfiguration", + name="subscribed_users", + field=models.ManyToManyField( + related_name="alert_configurations", through="posthog.AlertSubscription", to=settings.AUTH_USER_MODEL + ), + ), + migrations.AddField( + model_name="alertconfiguration", + name="team", + field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to="posthog.team"), + ), + migrations.AddField( + model_name="alertconfiguration", + name="threshold", + field=models.ForeignKey( + blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, to="posthog.threshold" + ), + ), + migrations.CreateModel( + name="AlertCheck", + fields=[ + ( + "id", + models.UUIDField( + default=posthog.models.utils.UUIDT, editable=False, primary_key=True, serialize=False + ), + ), + ("created_at", models.DateTimeField(auto_now_add=True)), + ("calculated_value", models.FloatField(blank=True, null=True)), + ("condition", models.JSONField(default=dict)), + ("targets_notified", models.JSONField(default=dict)), + ("error", models.JSONField(blank=True, null=True)), + ( + "state", + models.CharField( + choices=[("firing", "Firing"), ("not_met", "Not Met")], default="not_met", max_length=10 + ), + ), + ( + "alert_configuration", + models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to="posthog.alertconfiguration"), + ), + ], + options={ + "abstract": False, + }, + ), + ] diff --git a/posthog/models/__init__.py b/posthog/models/__init__.py index 36bc77625ad95..674d665288439 100644 --- a/posthog/models/__init__.py +++ b/posthog/models/__init__.py @@ -17,7 +17,7 @@ from .action.action_step import ActionStep from .activity_logging.activity_log import ActivityLog from .activity_logging.notification_viewed import NotificationViewed -from .alert import Alert +from .alert import AlertConfiguration from .annotation import Annotation from .async_deletion import AsyncDeletion, DeletionType from .async_migration import AsyncMigration, AsyncMigrationError, MigrationStatus @@ -75,7 +75,7 @@ from .user_scene_personalisation import UserScenePersonalisation __all__ = [ - "Alert", + "AlertConfiguration", "Action", "ActionStep", "ActivityLog", diff --git a/posthog/models/alert.py b/posthog/models/alert.py index 973228820c752..807bc0a0437aa 100644 --- a/posthog/models/alert.py +++ b/posthog/models/alert.py @@ -1,20 +1,213 @@ +from datetime import datetime, UTC, timedelta +from typing import Any, Optional + from django.db import models +from django.core.exceptions import ValidationError + +from posthog.hogql_queries.legacy_compatibility.flagged_conversion_manager import conversion_to_query_based from posthog.models.insight import Insight +from posthog.models.utils import UUIDModel, CreatedMetaFields +from posthog.schema import AlertCondition, InsightThreshold def are_alerts_supported_for_insight(insight: Insight) -> bool: - query = insight.query - if query is None or query.get("kind") != "TrendsQuery": - return False - if query.get("trendsFilter", {}).get("display") != "BoldNumber": - return False + with conversion_to_query_based(insight): + query = insight.query + while query.get("source"): + query = query["source"] + if query is None or query.get("kind") != "TrendsQuery": + return False + if query.get("trendsFilter", {}).get("display") != "BoldNumber": + return False return True +class ConditionValidator: + def __init__(self, threshold: Optional[InsightThreshold], condition: AlertCondition): + self.threshold = threshold + self.condition = condition + + def validate(self, calculated_value: float) -> list[str]: + validators: Any = [ + self.validate_absolute_threshold, + ] + matches = [] + for validator in validators: + matches += validator(calculated_value) + return matches + + def validate_absolute_threshold(self, calculated_value: float) -> list[str]: + if not self.threshold or not self.threshold.absoluteThreshold: + return [] + + absolute_threshold = self.threshold.absoluteThreshold + if absolute_threshold.lower is not None and calculated_value < absolute_threshold.lower: + return [f"The trend value ({calculated_value}) is below the lower threshold ({absolute_threshold.lower})"] + if absolute_threshold.upper is not None and calculated_value > absolute_threshold.upper: + return [f"The trend value ({calculated_value}) is above the upper threshold ({absolute_threshold.upper})"] + return [] + + class Alert(models.Model): + """ + @deprecated("AlertConfiguration should be used instead.") + """ + team = models.ForeignKey("Team", on_delete=models.CASCADE) insight = models.ForeignKey("posthog.Insight", on_delete=models.CASCADE) name = models.CharField(max_length=100) target_value = models.TextField() anomaly_condition = models.JSONField(default=dict) + + +class Threshold(CreatedMetaFields, UUIDModel): + """ + Threshold holds the configuration for a threshold. This can either be attached to an alert, or used as a standalone + object for other purposes. + """ + + team = models.ForeignKey("Team", on_delete=models.CASCADE) + insight = models.ForeignKey("posthog.Insight", on_delete=models.CASCADE) + + name = models.CharField(max_length=255, blank=True) + configuration = models.JSONField(default=dict) + + def clean(self): + config = InsightThreshold.model_validate(self.configuration) + if not config or not config.absoluteThreshold: + return + if config.absoluteThreshold.lower is not None and config.absoluteThreshold.upper is not None: + if config.absoluteThreshold.lower > config.absoluteThreshold.upper: + raise ValidationError("Lower threshold must be less than upper threshold") + + +class AlertConfiguration(CreatedMetaFields, UUIDModel): + ALERTS_PER_TEAM = 10 + + team = models.ForeignKey("Team", on_delete=models.CASCADE) + insight = models.ForeignKey("posthog.Insight", on_delete=models.CASCADE) + + name = models.CharField(max_length=255, blank=True) + subscribed_users = models.ManyToManyField( + "posthog.User", + through="posthog.AlertSubscription", + through_fields=("alert_configuration", "user"), + related_name="alert_configurations", + ) + + # The threshold to evaluate the alert against. If null, the alert must have other conditions to trigger. + threshold = models.ForeignKey(Threshold, on_delete=models.CASCADE, null=True, blank=True) + condition = models.JSONField(default=dict) + + STATE_CHOICES = [ + ("firing", "Firing"), + ("inactive", "Inactive"), + ] + state = models.CharField(max_length=10, choices=STATE_CHOICES, default="inactive") + enabled = models.BooleanField(default=True) + + last_notified_at = models.DateTimeField(null=True, blank=True) + + def __str__(self): + return f"{self.name} (Team: {self.team})" + + def save(self, *args, **kwargs): + if not self.enabled: + # When disabling an alert, set the state to inactive + self.state = "inactive" + if "update_fields" in kwargs: + kwargs["update_fields"].append("state") + + super().save(*args, **kwargs) + + def evaluate_condition(self, calculated_value) -> list[str]: + threshold = InsightThreshold.model_validate(self.threshold.configuration) if self.threshold else None + condition = AlertCondition.model_validate(self.condition) + validator = ConditionValidator(threshold=threshold, condition=condition) + return validator.validate(calculated_value) + + def add_check( + self, *, calculated_value: Optional[float], error: Optional[dict] = None + ) -> tuple["AlertCheck", list[str]]: + """Add a new AlertCheck, managing state transitions and cooldown.""" + matches = self.evaluate_condition(calculated_value) if calculated_value is not None else [] + targets_notified = {} + + # Determine the appropriate state for this check + if matches: + if self.state != "firing": + # Transition to firing state and send a notification + check_state = "firing" + self.last_notified_at = datetime.now(UTC) + targets_notified = {"users": list(self.subscribed_users.all().values_list("email", flat=True))} + else: + check_state = "firing" # Already firing, no new notification + matches = [] # Don't send duplicate notifications + else: + check_state = "not_met" + self.state = "inactive" # Set the Alert to inactive if the threshold is no longer met + # Optionally send a resolved notification + + alert_check = AlertCheck.objects.create( + alert_configuration=self, + calculated_value=calculated_value, + condition=self.condition, + targets_notified=targets_notified, + state=check_state, + error=error, + ) + + # Update the Alert state + if check_state == "firing": + self.state = "firing" + elif check_state == "not_met": + self.state = "inactive" + + self.save() + return alert_check, matches + + +class AlertSubscription(CreatedMetaFields, UUIDModel): + user = models.ForeignKey( + "User", + on_delete=models.CASCADE, + limit_choices_to={ + "is_active": True, + "organization_id": models.OuterRef("alert_configuration__team__organization_id"), + }, + related_name="alert_subscriptions", + ) + alert_configuration = models.ForeignKey(AlertConfiguration, on_delete=models.CASCADE) + subscribed = models.BooleanField(default=True) + + def __str__(self): + return f"AlertSubscription for {self.alert_configuration.name} by {self.user.email}" + + class Meta: + unique_together = ["user", "alert_configuration"] + + +class AlertCheck(UUIDModel): + alert_configuration = models.ForeignKey(AlertConfiguration, on_delete=models.CASCADE) + created_at = models.DateTimeField(auto_now_add=True) + calculated_value = models.FloatField(null=True, blank=True) + condition = models.JSONField(default=dict) # Snapshot of the condition at the time of the check + targets_notified = models.JSONField(default=dict) + error = models.JSONField(null=True, blank=True) + + STATE_CHOICES = [ + ("firing", "Firing"), + ("not_met", "Not Met"), + ] + state = models.CharField(max_length=10, choices=STATE_CHOICES, default="not_met") + + def __str__(self): + return f"AlertCheck for {self.alert_configuration.name} at {self.created_at}" + + @classmethod + def clean_up_old_checks(cls) -> int: + retention_days = 14 + oldest_allowed_date = datetime.now(UTC) - timedelta(days=retention_days) + rows_count, _ = cls.objects.filter(created_at__lt=oldest_allowed_date).delete() + return rows_count diff --git a/posthog/models/filters/mixins/session_recordings.py b/posthog/models/filters/mixins/session_recordings.py index d4968a8634852..aad2202dfdafe 100644 --- a/posthog/models/filters/mixins/session_recordings.py +++ b/posthog/models/filters/mixins/session_recordings.py @@ -4,8 +4,9 @@ from posthog.hogql import ast from posthog.constants import PERSON_UUID_FILTER, SESSION_RECORDINGS_FILTER_IDS, PropertyOperatorType from posthog.models.filters.mixins.common import BaseParamMixin +from posthog.models.filters.mixins.property import PropertyMixin from posthog.models.filters.mixins.utils import cached_property -from posthog.models.property import Property +from posthog.models.property import PropertyGroup class PersonUUIDMixin(BaseParamMixin): @@ -14,10 +15,10 @@ def person_uuid(self) -> Optional[str]: return self._data.get(PERSON_UUID_FILTER, None) -class SessionRecordingsMixin(BaseParamMixin): +class SessionRecordingsMixin(PropertyMixin, BaseParamMixin): @cached_property - def console_search_query(self) -> str | None: - return self._data.get("console_search_query", None) + def order(self) -> str: + return self._data.get("order", "start_time") # Supports a legacy use case where events were ORed not ANDed # Can be removed and replaced with ast_operand once the new universal replay filtering is out @@ -27,8 +28,10 @@ def events_operand(self) -> type[Union[ast.And, ast.Or]]: return ast.And if operand == "AND" else ast.Or @cached_property - def _operand(self) -> Literal["AND"] | Literal["OR"]: - return self._data.get("operand", "AND") + def console_log_filters(self) -> PropertyGroup: + property_group = self._parse_data(key="console_log_filters") + property_group.type = self.property_operand + return property_group @cached_property def property_operand(self) -> PropertyOperatorType: @@ -39,36 +42,8 @@ def ast_operand(self) -> type[Union[ast.And, ast.Or]]: return ast.And if self._operand == "AND" else ast.Or @cached_property - def console_logs_filter(self) -> list[Literal["error", "warn", "info"]]: - user_value = self._data.get("console_logs", None) or [] - if isinstance(user_value, str): - user_value = json.loads(user_value) - valid_values = [x for x in user_value if x in ["error", "warn", "info"]] - return valid_values - - @cached_property - def duration_type_filter(self) -> Literal["duration", "active_seconds", "inactive_seconds"]: - user_value = self._data.get("duration_type_filter", None) - if user_value in ["duration", "active_seconds", "inactive_seconds"]: - return user_value - else: - return "duration" - - @cached_property - def recording_duration_filter(self) -> Optional[Property]: - duration_filter_data_str = self._data.get("session_recording_duration", None) - if duration_filter_data_str: - filter_data = json.loads(duration_filter_data_str) - return Property(**filter_data) - return None - - @cached_property - def snapshot_source_filter(self) -> Optional[Property]: - snapshot_source_data_str = self._data.get("snapshot_source", None) - if isinstance(snapshot_source_data_str, str): - filter_data = json.loads(snapshot_source_data_str) - return Property(**filter_data) - return None + def _operand(self) -> Literal["AND"] | Literal["OR"]: + return self._data.get("operand", "AND") @cached_property def session_ids(self) -> Optional[list[str]]: @@ -92,3 +67,7 @@ def session_ids(self) -> Optional[list[str]]: # If the property is at all present, we assume that the user wants to filter by it return [] + + @cached_property + def having_predicates(self) -> PropertyGroup: + return self._parse_data(key="having_predicates") diff --git a/posthog/schema.py b/posthog/schema.py index 56720edb0fae7..45584a0fcc811 100644 --- a/posthog/schema.py +++ b/posthog/schema.py @@ -12,14 +12,6 @@ class SchemaRoot(RootModel[Any]): root: Any -class AbsoluteThreshold(BaseModel): - model_config = ConfigDict( - extra="forbid", - ) - lower: Optional[float] = None - upper: Optional[float] = None - - class MathGroupTypeIndex(float, Enum): NUMBER_0 = 0 NUMBER_1 = 1 @@ -36,11 +28,43 @@ class AggregationAxisFormat(StrEnum): PERCENTAGE_SCALED = "percentage_scaled" -class AnomalyCondition(BaseModel): +class AlertCheck(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + calculated_value: float + created_at: str + id: str + state: str + targets_notified: bool + + +class AlertCondition(BaseModel): + pass model_config = ConfigDict( extra="forbid", ) - absoluteThreshold: AbsoluteThreshold + + +class AlertTypeBase(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + condition: AlertCondition + enabled: bool + insight: float + name: str + + +class AlertTypeWrite(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + condition: AlertCondition + enabled: bool + insight: float + name: str + subscribed_users: list[int] class Kind(StrEnum): @@ -560,6 +584,19 @@ class GoalLine(BaseModel): value: float +class HedgehogColorOptions(StrEnum): + GREEN = "green" + RED = "red" + BLUE = "blue" + PURPLE = "purple" + DARK = "dark" + LIGHT = "light" + SEPIA = "sepia" + INVERT = "invert" + INVERT_HUE = "invert-hue" + GREYSCALE = "greyscale" + + class HogLanguage(StrEnum): HOG = "hog" HOG_JSON = "hogJson" @@ -701,6 +738,14 @@ class InsightNodeKind(StrEnum): LIFECYCLE_QUERY = "LifecycleQuery" +class InsightsThresholdAbsolute(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + lower: Optional[float] = None + upper: Optional[float] = None + + class IntervalType(StrEnum): MINUTE = "minute" HOUR = "hour" @@ -716,6 +761,22 @@ class LifecycleToggle(StrEnum): DORMANT = "dormant" +class MatchedRecordingEvent(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + uuid: str + + +class MinimalHedgehogConfig(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + accessories: list[str] + color: Optional[HedgehogColorOptions] = None + use_as_profile: bool + + class MultipleBreakdownType(StrEnum): PERSON = "person" EVENT = "event" @@ -738,6 +799,7 @@ class NodeKind(StrEnum): FUNNELS_ACTORS_QUERY = "FunnelsActorsQuery" FUNNEL_CORRELATION_ACTORS_QUERY = "FunnelCorrelationActorsQuery" SESSIONS_TIMELINE_QUERY = "SessionsTimelineQuery" + RECORDINGS_QUERY = "RecordingsQuery" SESSION_ATTRIBUTION_EXPLORER_QUERY = "SessionAttributionExplorerQuery" ERROR_TRACKING_QUERY = "ErrorTrackingQuery" DATA_TABLE_NODE = "DataTableNode" @@ -816,6 +878,19 @@ class PathsFilterLegacy(BaseModel): step_limit: Optional[int] = None +class PersonType(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + created_at: Optional[str] = None + distinct_ids: list[str] + id: Optional[str] = None + is_identified: Optional[bool] = None + name: Optional[str] = None + properties: dict[str, Any] + uuid: Optional[str] = None + + class PropertyFilterType(StrEnum): META = "meta" EVENT = "event" @@ -1047,6 +1122,17 @@ class SessionPropertyFilter(BaseModel): value: Optional[Union[str, float, list[Union[str, float]]]] = None +class SnapshotSource(StrEnum): + WEB = "web" + MOBILE = "mobile" + UNKNOWN = "unknown" + + +class Storage(StrEnum): + OBJECT_STORAGE_LTS = "object_storage_lts" + OBJECT_STORAGE = "object_storage" + + class StepOrderValue(StrEnum): STRICT = "strict" UNORDERED = "unordered" @@ -1258,6 +1344,20 @@ class TrendsQueryResponse(BaseModel): ) +class UserBasicType(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + distinct_id: str + email: str + first_name: str + hedgehog_config: Optional[MinimalHedgehogConfig] = None + id: float + is_email_verified: Optional[Any] = None + last_name: Optional[str] = None + uuid: str + + class ActionsPie(BaseModel): model_config = ConfigDict( extra="forbid", @@ -2474,6 +2574,13 @@ class InsightActorsQueryBase(BaseModel): response: Optional[ActorsQueryResponse] = None +class InsightThreshold(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + absoluteThreshold: Optional[InsightsThresholdAbsolute] = None + + class LifecycleFilter(BaseModel): model_config = ConfigDict( extra="forbid", @@ -2524,6 +2631,14 @@ class LogEntryPropertyFilter(BaseModel): value: Optional[Union[str, float, list[Union[str, float]]]] = None +class MatchedRecording(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + events: list[MatchedRecordingEvent] + session_id: Optional[str] = None + + class MultipleBreakdownOptions(BaseModel): model_config = ConfigDict( extra="forbid", @@ -3232,6 +3347,35 @@ class SessionAttributionExplorerQuery(BaseModel): response: Optional[SessionAttributionExplorerQueryResponse] = None +class SessionRecordingType(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + active_seconds: Optional[float] = None + click_count: Optional[float] = None + console_error_count: Optional[float] = None + console_log_count: Optional[float] = None + console_warn_count: Optional[float] = None + distinct_id: Optional[str] = None + email: Optional[str] = None + end_time: str = Field(..., description="When the recording ends in ISO format.") + id: str + inactive_seconds: Optional[float] = None + keypress_count: Optional[float] = None + matching_events: Optional[list[MatchedRecording]] = Field(default=None, description="List of matching events. *") + mouse_activity_count: Optional[float] = Field( + default=None, description="count of all mouse activity in the recording, not just clicks" + ) + person: Optional[PersonType] = None + recording_duration: float = Field(..., description="Length of recording in seconds.") + snapshot_source: SnapshotSource + start_time: str = Field(..., description="When the recording starts in ISO format.") + start_url: Optional[str] = None + storage: Optional[Storage] = Field(default=None, description="Where this recording information was loaded from") + summary: Optional[str] = None + viewed: bool = Field(..., description="Whether this recording has been viewed already.") + + class SessionsTimelineQueryResponse(BaseModel): model_config = ConfigDict( extra="forbid", @@ -3315,6 +3459,31 @@ class WebTopClicksQuery(BaseModel): useSessionsTable: Optional[bool] = None +class Threshold(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + configuration: InsightThreshold + + +class AlertType(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + checks: list[AlertCheck] + condition: AlertCondition + created_at: str + created_by: UserBasicType + enabled: bool + id: str + insight: float + last_notified_at: str + name: str + state: str + subscribed_users: list[UserBasicType] + threshold: Threshold + + class AnyResponseType( RootModel[ Union[ @@ -4040,6 +4209,14 @@ class QueryResponseAlternative24(BaseModel): ) +class RecordingsQueryResponse(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + has_next: bool + results: list[SessionRecordingType] + + class RetentionQueryResponse(BaseModel): model_config = ConfigDict( extra="forbid", @@ -4197,6 +4374,67 @@ class PropertyGroupFilter(BaseModel): values: list[PropertyGroupFilterValue] +class RecordingsQuery(BaseModel): + model_config = ConfigDict( + extra="forbid", + ) + actions: Optional[list[dict[str, Any]]] = None + console_log_filters: Optional[list[LogEntryPropertyFilter]] = None + date_from: Optional[str] = None + date_to: Optional[str] = None + events: Optional[list[dict[str, Any]]] = None + filter_test_accounts: Optional[bool] = None + having_predicates: Optional[ + list[ + Union[ + EventPropertyFilter, + PersonPropertyFilter, + ElementPropertyFilter, + SessionPropertyFilter, + CohortPropertyFilter, + RecordingPropertyFilter, + LogEntryPropertyFilter, + GroupPropertyFilter, + FeaturePropertyFilter, + HogQLPropertyFilter, + EmptyPropertyFilter, + DataWarehousePropertyFilter, + DataWarehousePersonPropertyFilter, + ] + ] + ] = None + kind: Literal["RecordingsQuery"] = "RecordingsQuery" + limit: Optional[int] = None + modifiers: Optional[HogQLQueryModifiers] = Field( + default=None, description="Modifiers used when performing the query" + ) + offset: Optional[int] = None + operand: Optional[FilterLogicalOperator] = None + order: Union[DurationType, str] + person_uuid: Optional[str] = None + properties: Optional[ + list[ + Union[ + EventPropertyFilter, + PersonPropertyFilter, + ElementPropertyFilter, + SessionPropertyFilter, + CohortPropertyFilter, + RecordingPropertyFilter, + LogEntryPropertyFilter, + GroupPropertyFilter, + FeaturePropertyFilter, + HogQLPropertyFilter, + EmptyPropertyFilter, + DataWarehousePropertyFilter, + DataWarehousePersonPropertyFilter, + ] + ] + ] = None + response: Optional[RecordingsQueryResponse] = None + session_ids: Optional[list[str]] = None + + class RetentionQuery(BaseModel): model_config = ConfigDict( extra="forbid", diff --git a/posthog/session_recordings/queries/session_recording_list_from_filters.py b/posthog/session_recordings/queries/session_recording_list_from_filters.py index 893cd1f4c9537..92d868e83f0b2 100644 --- a/posthog/session_recordings/queries/session_recording_list_from_filters.py +++ b/posthog/session_recordings/queries/session_recording_list_from_filters.py @@ -2,7 +2,7 @@ from datetime import datetime, timedelta from posthog.hogql import ast -from posthog.hogql.ast import Constant, CompareOperation +from posthog.hogql.ast import CompareOperation from posthog.hogql.parser import parse_select from posthog.hogql.property import entity_to_expr, property_to_expr from posthog.hogql.query import execute_hogql_query @@ -132,8 +132,7 @@ def get_query(self): ) def _order_by_clause(self) -> ast.Field: - order = self._filter.target_entity_order or "start_time" - return ast.Field(chain=[order]) + return ast.Field(chain=[self._filter.order]) def _where_predicates(self) -> Union[ast.And, ast.Or]: exprs: list[ast.Expr] = [ @@ -208,36 +207,14 @@ def _where_predicates(self) -> Union[ast.And, ast.Or]: ) optional_exprs.append(property_to_expr(remaining_properties, team=self._team, scope="replay")) - console_logs_predicates: list[ast.Expr] = [] - if self._filter.console_logs_filter: - console_logs_predicates.append( - ast.CompareOperation( - op=ast.CompareOperationOp.In, - left=ast.Field(chain=["level"]), - right=ast.Constant(value=self._filter.console_logs_filter), - ) - ) - - if self._filter.console_search_query: - console_logs_predicates.append( - ast.CompareOperation( - op=ast.CompareOperationOp.Gt, - left=ast.Call( - name="positionCaseInsensitive", - args=[ - ast.Field(chain=["message"]), - ast.Constant(value=self._filter.console_search_query), - ], - ), - right=ast.Constant(value=0), - ) - ) - - if console_logs_predicates: + if self._filter.console_log_filters.values: + # print(self._filter.console_log_filters.type) console_logs_subquery = ast.SelectQuery( select=[ast.Field(chain=["log_source_id"])], select_from=ast.JoinExpr(table=ast.Field(chain=["console_logs_log_entries"])), - where=self._filter.ast_operand(exprs=console_logs_predicates), + where=self._filter.ast_operand( + exprs=[property_to_expr(self._filter.console_log_filters, team=self._team)] + ), ) optional_exprs.append( @@ -253,39 +230,8 @@ def _where_predicates(self) -> Union[ast.And, ast.Or]: return ast.And(exprs=exprs) - def _having_predicates(self) -> ast.And | Constant: - exprs: list[ast.Expr] = [] - - if self._filter.recording_duration_filter: - op = ( - ast.CompareOperationOp.GtEq - if self._filter.recording_duration_filter.operator == "gt" - or self._filter.recording_duration_filter.operator == "gte" - else ast.CompareOperationOp.LtEq - ) - exprs.append( - ast.CompareOperation( - op=op, - left=ast.Field(chain=[self._filter.duration_type_filter]), - right=ast.Constant(value=self._filter.recording_duration_filter.value), - ), - ) - - if self._filter.snapshot_source_filter: - op = ( - ast.CompareOperationOp.In - if self._filter.snapshot_source_filter.operator == "exact" - else ast.CompareOperationOp.NotIn - ) - exprs.append( - ast.CompareOperation( - op=op, - left=ast.Call(name="argMinMerge", args=[ast.Field(chain=["s", "snapshot_source"])]), - right=ast.Constant(value=self._filter.snapshot_source_filter.value), - ), - ) - - return ast.And(exprs=exprs) if exprs else ast.Constant(value=True) + def _having_predicates(self) -> ast.Expr: + return property_to_expr(self._filter.having_predicates, team=self._team, scope="replay") def _strip_person_and_event_properties(self, property_group: PropertyGroup) -> PropertyGroup | None: property_groups_to_keep = [g for g in property_group.flat if not is_person_property(g)] @@ -344,7 +290,7 @@ def person_properties(self) -> PropertyGroup | None: @cached_property def _where_predicates(self) -> ast.Expr: return ( - property_to_expr(self.person_properties, team=self._team, scope="replay_pdi") + property_to_expr(self.person_properties, team=self._team) if self.person_properties else ast.Constant(value=True) ) diff --git a/posthog/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr b/posthog/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr index ff61bf1a8a80a..9dd57f56763e8 100644 --- a/posthog/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr +++ b/posthog/session_recordings/queries/test/__snapshots__/test_session_recording_list_from_filters.ambr @@ -180,7 +180,7 @@ GROUP BY events.`$session_id` HAVING hasAll(groupUniqArray(events.event), ['$pageview', 'custom-event'])))) GROUP BY s.session_id - HAVING ifNull(greaterOrEquals(duration, 60), 0) + HAVING 1 ORDER BY start_time DESC LIMIT 51 OFFSET 0 SETTINGS readonly=2, @@ -466,7 +466,7 @@ FROM session_replay_events AS s WHERE and(equals(s.team_id, 2), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) GROUP BY s.session_id - HAVING ifNull(greaterOrEquals(duration, 60), 0) + HAVING ifNull(greater(duration, 60), 0) ORDER BY start_time DESC LIMIT 51 OFFSET 0 SETTINGS readonly=2, @@ -498,7 +498,7 @@ FROM session_replay_events AS s WHERE and(equals(s.team_id, 2), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) GROUP BY s.session_id - HAVING ifNull(greaterOrEquals(active_seconds, 60), 0) + HAVING ifNull(greater(active_seconds, '60'), 0) ORDER BY start_time DESC LIMIT 51 OFFSET 0 SETTINGS readonly=2, @@ -530,7 +530,7 @@ FROM session_replay_events AS s WHERE and(equals(s.team_id, 2), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) GROUP BY s.session_id - HAVING ifNull(greaterOrEquals(inactive_seconds, 60), 0) + HAVING ifNull(greater(inactive_seconds, '60'), 0) ORDER BY start_time DESC LIMIT 51 OFFSET 0 SETTINGS readonly=2, @@ -978,7 +978,7 @@ FROM session_replay_events AS s WHERE and(equals(s.team_id, 2), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) GROUP BY s.session_id - HAVING ifNull(greaterOrEquals(duration, 60), 0) + HAVING ifNull(greater(duration, 60), 0) ORDER BY start_time DESC LIMIT 51 OFFSET 0 SETTINGS readonly=2, @@ -1010,7 +1010,7 @@ FROM session_replay_events AS s WHERE and(equals(s.team_id, 2), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) GROUP BY s.session_id - HAVING ifNull(lessOrEquals(duration, 60), 0) + HAVING ifNull(less(duration, 60), 0) ORDER BY start_time DESC LIMIT 51 OFFSET 0 SETTINGS readonly=2, @@ -1190,7 +1190,7 @@ GROUP BY events.`$session_id` HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) GROUP BY s.session_id - HAVING ifNull(greaterOrEquals(duration, 60), 0) + HAVING ifNull(greater(duration, 60), 0) ORDER BY start_time DESC LIMIT 51 OFFSET 0 SETTINGS readonly=2, @@ -1227,7 +1227,7 @@ GROUP BY events.`$session_id` HAVING hasAll(groupUniqArray(events.event), ['$pageview'])))) GROUP BY s.session_id - HAVING ifNull(greaterOrEquals(active_seconds, 60), 0) + HAVING ifNull(greater(active_seconds, 60), 0) ORDER BY start_time DESC LIMIT 51 OFFSET 0 SETTINGS readonly=2, @@ -2508,7 +2508,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE and(ifNull(in(console_logs_log_entries.level, ['warn', 'error']), 0), ifNull(greater(positionCaseInsensitive(console_logs_log_entries.message, 'message 4'), 0), 0))))) + WHERE or(or(ifNull(equals(console_logs_log_entries.level, 'warn'), 0), ifNull(equals(console_logs_log_entries.level, 'error'), 0)), ifNull(equals(console_logs_log_entries.message, 'message 4'), 0))))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -2546,7 +2546,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE and(ifNull(in(console_logs_log_entries.level, ['warn', 'error']), 0), ifNull(greater(positionCaseInsensitive(console_logs_log_entries.message, 'message 5'), 0), 0))))) + WHERE and(or(ifNull(equals(console_logs_log_entries.level, 'warn'), 0), ifNull(equals(console_logs_log_entries.level, 'error'), 0)), ifNull(ilike(console_logs_log_entries.message, '%message 5%'), 0))))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -2584,45 +2584,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE and(ifNull(in(console_logs_log_entries.level, ['warn', 'error']), 0), ifNull(greater(positionCaseInsensitive(console_logs_log_entries.message, 'MESSAGE 5'), 0), 0))))) - GROUP BY s.session_id - HAVING 1 - ORDER BY start_time DESC - LIMIT 51 - OFFSET 0 SETTINGS readonly=2, - max_execution_time=60, - allow_experimental_object_type=1, - format_csv_allow_double_quotes=0, - max_ast_elements=4000000, - max_expanded_ast_elements=4000000, - max_bytes_before_external_group_by=0 - ''' -# --- -# name: TestSessionRecordingsListFromFilters.test_filter_for_recordings_by_console_text.3 - ''' - SELECT s.session_id AS session_id, - any(s.team_id), - any(s.distinct_id), - min(toTimeZone(s.min_first_timestamp, 'UTC')) AS start_time, - max(toTimeZone(s.max_last_timestamp, 'UTC')) AS end_time, - dateDiff('SECOND', start_time, end_time) AS duration, - argMinMerge(s.first_url) AS first_url, - sum(s.click_count), - sum(s.keypress_count), - sum(s.mouse_activity_count), - divide(sum(s.active_milliseconds), 1000) AS active_seconds, - minus(duration, active_seconds) AS inactive_seconds, - sum(s.console_log_count) AS console_log_count, - sum(s.console_warn_count) AS console_warn_count, - sum(s.console_error_count) AS console_error_count - FROM session_replay_events AS s - WHERE and(equals(s.team_id, 2), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-31 20:00:00.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-14 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-21 20:00:00.000000', 6, 'UTC')), 0), in(s.session_id, - (SELECT console_logs_log_entries.log_source_id AS log_source_id - FROM - (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message - FROM log_entries - WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE and(ifNull(in(console_logs_log_entries.level, ['info']), 0), ifNull(greater(positionCaseInsensitive(console_logs_log_entries.message, 'message 5'), 0), 0))))) + WHERE and(ifNull(equals(console_logs_log_entries.level, 'info'), 0), ifNull(ilike(console_logs_log_entries.message, '%message 5%'), 0))))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -2656,7 +2618,7 @@ FROM session_replay_events AS s WHERE and(equals(s.team_id, 2), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) GROUP BY s.session_id - HAVING ifNull(in(argMinMerge(s.snapshot_source), ['web']), 0) + HAVING ifNull(equals(argMinMerge(s.snapshot_source), 'web'), 0) ORDER BY start_time DESC LIMIT 51 OFFSET 0 SETTINGS readonly=2, @@ -2688,7 +2650,7 @@ FROM session_replay_events AS s WHERE and(equals(s.team_id, 2), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-11 13:46:23.000000', 6, 'UTC')), 0), ifNull(greaterOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2020-12-25 00:00:00.000000', 6, 'UTC')), 0), ifNull(lessOrEquals(toTimeZone(s.min_first_timestamp, 'UTC'), toDateTime64('2021-01-01 13:46:23.000000', 6, 'UTC')), 0)) GROUP BY s.session_id - HAVING ifNull(in(argMinMerge(s.snapshot_source), ['mobile']), 0) + HAVING ifNull(equals(argMinMerge(s.snapshot_source), 'mobile'), 0) ORDER BY start_time DESC LIMIT 51 OFFSET 0 SETTINGS readonly=2, @@ -2724,7 +2686,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(in(console_logs_log_entries.level, ['error']), 0)))) + WHERE ifNull(equals(console_logs_log_entries.level, 'error'), 0)))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -2762,7 +2724,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(in(console_logs_log_entries.level, ['info']), 0)))) + WHERE ifNull(equals(console_logs_log_entries.level, 'info'), 0)))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -2800,7 +2762,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(in(console_logs_log_entries.level, ['info']), 0)))) + WHERE ifNull(equals(console_logs_log_entries.level, 'info'), 0)))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -2838,7 +2800,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(in(console_logs_log_entries.level, ['warn']), 0)))) + WHERE ifNull(equals(console_logs_log_entries.level, 'warn'), 0)))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -2876,7 +2838,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(in(console_logs_log_entries.level, ['warn']), 0)))) + WHERE ifNull(equals(console_logs_log_entries.level, 'warn'), 0)))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -2914,7 +2876,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(in(console_logs_log_entries.level, ['info']), 0)))) + WHERE ifNull(equals(console_logs_log_entries.level, 'info'), 0)))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -2952,7 +2914,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(in(console_logs_log_entries.level, ['warn', 'error']), 0)))) + WHERE or(ifNull(equals(console_logs_log_entries.level, 'warn'), 0), ifNull(equals(console_logs_log_entries.level, 'error'), 0))))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -2990,7 +2952,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE ifNull(in(console_logs_log_entries.level, ['info']), 0)))) + WHERE ifNull(equals(console_logs_log_entries.level, 'info'), 0)))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -3524,7 +3486,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE and(ifNull(in(console_logs_log_entries.level, ['warn']), 0), ifNull(greater(positionCaseInsensitive(console_logs_log_entries.message, 'random'), 0), 0))))) + WHERE and(ifNull(equals(console_logs_log_entries.level, 'warn'), 0), ifNull(equals(console_logs_log_entries.message, 'random'), 0))))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC @@ -3562,7 +3524,7 @@ (SELECT log_entries.log_source_id AS log_source_id, log_entries.level AS level, log_entries.message AS message FROM log_entries WHERE and(equals(log_entries.team_id, 2), equals(log_entries.log_source, 'session_replay'))) AS console_logs_log_entries - WHERE or(ifNull(in(console_logs_log_entries.level, ['warn']), 0), ifNull(greater(positionCaseInsensitive(console_logs_log_entries.message, 'random'), 0), 0))))) + WHERE or(ifNull(equals(console_logs_log_entries.level, 'warn'), 0), ifNull(equals(console_logs_log_entries.message, 'random'), 0))))) GROUP BY s.session_id HAVING 1 ORDER BY start_time DESC diff --git a/posthog/session_recordings/queries/test/test_session_recording_list_from_filters.py b/posthog/session_recordings/queries/test/test_session_recording_list_from_filters.py index 9cd6a5709101d..5f57396df1a7a 100644 --- a/posthog/session_recordings/queries/test/test_session_recording_list_from_filters.py +++ b/posthog/session_recordings/queries/test/test_session_recording_list_from_filters.py @@ -234,8 +234,7 @@ def test_basic_query_active_sessions( (session_recordings, _, _) = self._filter_recordings_by( { - "duration_type_filter": "duration", - "session_recording_duration": '{"type":"recording","key":"duration","value":60,"operator":"gt"}', + "having_predicates": '[{"type":"recording","key":"duration","value":60,"operator":"gt"}]', } ) @@ -249,8 +248,7 @@ def test_basic_query_active_sessions( (session_recordings, _, _) = self._filter_recordings_by( { - "duration_type_filter": "active_seconds", - "session_recording_duration": '{"type":"recording","key":"duration","value":60,"operator":"gt"}', + "having_predicates": '[{"type":"recording","key":"active_seconds","value":"60","operator":"gt"}]', } ) @@ -260,8 +258,7 @@ def test_basic_query_active_sessions( (session_recordings, _, _) = self._filter_recordings_by( { - "duration_type_filter": "inactive_seconds", - "session_recording_duration": '{"type":"recording","key":"duration","value":60,"operator":"gt"}', + "having_predicates": '[{"type":"recording","key":"inactive_seconds","value":"60","operator":"gt"}]', } ) @@ -423,21 +420,21 @@ def test_basic_query_with_ordering(self): ) (session_recordings) = self._filter_recordings_by( - {"no_filter": None, "limit": 3, "offset": 0, "entity_order": "active_seconds"} + {"no_filter": None, "limit": 3, "offset": 0, "order": "active_seconds"} ) ordered_by_activity = [(r["session_id"], r["active_seconds"]) for r in session_recordings.results] assert ordered_by_activity == [(session_id_two, 1.0), (session_id_one, 0.002)] (session_recordings) = self._filter_recordings_by( - {"no_filter": None, "limit": 3, "offset": 0, "entity_order": "console_error_count"} + {"no_filter": None, "limit": 3, "offset": 0, "order": "console_error_count"} ) ordered_by_errors = [(r["session_id"], r["console_error_count"]) for r in session_recordings.results] assert ordered_by_errors == [(session_id_one, 1012), (session_id_two, 430)] (session_recordings) = self._filter_recordings_by( - {"no_filter": None, "limit": 3, "offset": 0, "entity_order": "start_time"} + {"no_filter": None, "limit": 3, "offset": 0, "order": "start_time"} ) ordered_by_default = [(r["session_id"], r["start_time"]) for r in session_recordings.results] @@ -854,7 +851,6 @@ def test_event_filter_with_active_sessions( (session_recordings, _, _) = self._filter_recordings_by( { - "duration_type_filter": "duration", "events": [ { "id": "$pageview", @@ -863,7 +859,7 @@ def test_event_filter_with_active_sessions( "name": "$pageview", } ], - "session_recording_duration": '{"type":"recording","key":"duration","value":60,"operator":"gt"}', + "having_predicates": '[{"type":"recording","key":"duration","value":60,"operator":"gt"}]', } ) @@ -873,7 +869,6 @@ def test_event_filter_with_active_sessions( (session_recordings, _, _) = self._filter_recordings_by( { - "duration_type_filter": "active_seconds", "events": [ { "id": "$pageview", @@ -882,7 +877,7 @@ def test_event_filter_with_active_sessions( "name": "$pageview", } ], - "session_recording_duration": '{"type":"recording","key":"duration","value":60,"operator":"gt"}', + "having_predicates": '[{"type":"recording","key":"active_seconds","value":60,"operator":"gt"}]', } ) @@ -1300,12 +1295,12 @@ def test_duration_filter(self): ) (session_recordings, _, _) = self._filter_recordings_by( - {"session_recording_duration": '{"type":"recording","key":"duration","value":60,"operator":"gt"}'} + {"having_predicates": '[{"type":"recording","key":"duration","value":60,"operator":"gt"}]'} ) assert [r["session_id"] for r in session_recordings] == [session_id_two] (session_recordings, _, _) = self._filter_recordings_by( - {"session_recording_duration": '{"type":"recording","key":"duration","value":60,"operator":"lt"}'} + {"having_predicates": '[{"type":"recording","key":"duration","value":60,"operator":"lt"}]'} ) assert [r["session_id"] for r in session_recordings] == [session_id_one] @@ -1500,13 +1495,19 @@ def test_operand_or_filters(self): ) (session_recordings, _, _) = self._filter_recordings_by( - {"console_logs": ["warn"], "console_search_query": "random"} + { + "console_log_filters": '[{"key": "level", "value": ["warn"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "random", "operator": "exact", "type": "log_entry"}]', + "operand": "AND", + } ) assert len(session_recordings) == 1 assert session_recordings[0]["session_id"] == session_with_both_log_filters (session_recordings, _, _) = self._filter_recordings_by( - {"console_logs": ["warn"], "console_search_query": "random", "operand": "OR"} + { + "console_log_filters": '[{"key": "level", "value": ["warn"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "random", "operator": "exact", "type": "log_entry"}]', + "operand": "OR", + } ) assert len(session_recordings) == 2 @@ -2513,7 +2514,14 @@ def test_filter_for_recordings_with_console_logs(self): team_id=self.team.id, ) - (session_recordings, _, _) = self._filter_recordings_by({"console_logs": ["info"]}) + # (session_recordings, _, _) = self._filter_recordings_by({"console_logs": ["info"]}) + + (session_recordings, _, _) = self._filter_recordings_by( + { + "console_log_filters": '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}]', + "operand": "AND", + } + ) actual = sorted( [(sr["session_id"], sr["console_log_count"]) for sr in session_recordings], @@ -2524,7 +2532,12 @@ def test_filter_for_recordings_with_console_logs(self): (with_logs_session_id, 4), ] - (session_recordings, _, _) = self._filter_recordings_by({"console_logs": ["warn"]}) + (session_recordings, _, _) = self._filter_recordings_by( + { + "console_log_filters": '[{"key": "level", "value": ["warn"], "operator": "exact", "type": "log_entry"}]', + "operand": "AND", + } + ) assert session_recordings == [] @snapshot_clickhouse_queries @@ -2557,7 +2570,12 @@ def test_filter_for_recordings_with_console_warns(self): team_id=self.team.id, ) - (session_recordings, _, _) = self._filter_recordings_by({"console_logs": ["warn"]}) + (session_recordings, _, _) = self._filter_recordings_by( + { + "console_log_filters": '[{"key": "level", "value": ["warn"], "operator": "exact", "type": "log_entry"}]', + "operand": "AND", + } + ) assert sorted( [(sr["session_id"], sr["console_warn_count"]) for sr in session_recordings], @@ -2566,7 +2584,12 @@ def test_filter_for_recordings_with_console_warns(self): (with_logs_session_id, 4), ] - (session_recordings, _, _) = self._filter_recordings_by({"console_logs": ["info"]}) + (session_recordings, _, _) = self._filter_recordings_by( + { + "console_log_filters": '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}]', + "operand": "AND", + } + ) assert session_recordings == [] @@ -2600,7 +2623,12 @@ def test_filter_for_recordings_with_console_errors(self): team_id=self.team.id, ) - (session_recordings, _, _) = self._filter_recordings_by({"console_logs": ["error"]}) + (session_recordings, _, _) = self._filter_recordings_by( + { + "console_log_filters": '[{"key": "level", "value": ["error"], "operator": "exact", "type": "log_entry"}]', + "operand": "AND", + } + ) assert sorted( [(sr["session_id"], sr["console_error_count"]) for sr in session_recordings], @@ -2609,7 +2637,12 @@ def test_filter_for_recordings_with_console_errors(self): (with_logs_session_id, 4), ] - (session_recordings, _, _) = self._filter_recordings_by({"console_logs": ["info"]}) + (session_recordings, _, _) = self._filter_recordings_by( + { + "console_log_filters": '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}]', + "operand": "AND", + } + ) assert session_recordings == [] @@ -2690,7 +2723,12 @@ def test_filter_for_recordings_with_mixed_console_counts(self): }, ) - (session_recordings, _, _) = self._filter_recordings_by({"console_logs": ["warn", "error"]}) + (session_recordings, _, _) = self._filter_recordings_by( + { + "console_log_filters": '[{"key": "level", "value": ["warn", "error"], "operator": "exact", "type": "log_entry"}]', + "operand": "AND", + } + ) assert sorted([sr["session_id"] for sr in session_recordings]) == sorted( [ @@ -2700,7 +2738,12 @@ def test_filter_for_recordings_with_mixed_console_counts(self): ] ) - (session_recordings, _, _) = self._filter_recordings_by({"console_logs": ["info"]}) + (session_recordings, _, _) = self._filter_recordings_by( + { + "console_log_filters": '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}]', + "operand": "AND", + } + ) assert sorted([sr["session_id"] for sr in session_recordings]) == sorted( [ @@ -2786,8 +2829,8 @@ def test_filter_for_recordings_by_console_text(self): (session_recordings, _, _) = self._filter_recordings_by( { # there are 5 warn and 4 error logs, message 4 matches in both - "console_logs": ["warn", "error"], - "console_search_query": "message 4", + "console_log_filters": '[{"key": "level", "value": ["warn", "error"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "message 4", "operator": "exact", "type": "log_entry"}]', + "operand": "OR", } ) @@ -2802,22 +2845,8 @@ def test_filter_for_recordings_by_console_text(self): (session_recordings, _, _) = self._filter_recordings_by( { # there are 5 warn and 4 error logs, message 5 matches only matches in warn - "console_logs": ["warn", "error"], - "console_search_query": "message 5", - } - ) - - assert sorted([sr["session_id"] for sr in session_recordings]) == sorted( - [ - with_warns_session_id, - ] - ) - - (session_recordings, _, _) = self._filter_recordings_by( - { - # match is case-insensitive - "console_logs": ["warn", "error"], - "console_search_query": "MESSAGE 5", + "console_log_filters": '[{"key": "level", "value": ["warn", "error"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "message 5", "operator": "icontains", "type": "log_entry"}]', + "operand": "AND", } ) @@ -2830,12 +2859,12 @@ def test_filter_for_recordings_by_console_text(self): (session_recordings, _, _) = self._filter_recordings_by( { # message 5 does not match log level "info" - "console_logs": ["info"], - "console_search_query": "message 5", + "console_log_filters": '[{"key": "level", "value": ["info"], "operator": "exact", "type": "log_entry"}, {"key": "message", "value": "message 5", "operator": "icontains", "type": "log_entry"}]', + "operand": "AND", } ) - assert sorted([sr["session_id"] for sr in session_recordings]) == sorted([]) + assert sorted([sr["session_id"] for sr in session_recordings]) == [] @snapshot_clickhouse_queries def test_filter_for_recordings_by_snapshot_source(self): @@ -2860,14 +2889,14 @@ def test_filter_for_recordings_by_snapshot_source(self): (session_recordings, _, _) = self._filter_recordings_by( { - "snapshot_source": '{"key": "snapshot_source", "value": ["web"], "operator": "exact", "type": "recording"}' + "having_predicates": '[{"key": "snapshot_source", "value": ["web"], "operator": "exact", "type": "recording"}]' } ) assert [r["session_id"] for r in session_recordings] == [session_id_one] (session_recordings, _, _) = self._filter_recordings_by( { - "snapshot_source": '{"key": "snapshot_source", "value": ["mobile"], "operator": "exact", "type": "recording"}' + "having_predicates": '[{"key": "snapshot_source", "value": ["mobile"], "operator": "exact", "type": "recording"}]' } ) assert [r["session_id"] for r in session_recordings] == [session_id_two] diff --git a/posthog/session_recordings/test/test_session_recordings.py b/posthog/session_recordings/test/test_session_recordings.py index 5efc1b1e58f4f..f6868001660d3 100644 --- a/posthog/session_recordings/test/test_session_recordings.py +++ b/posthog/session_recordings/test/test_session_recordings.py @@ -4,6 +4,7 @@ from datetime import datetime, timedelta, UTC from unittest.mock import ANY, patch, MagicMock, call from urllib.parse import urlencode +from typing import cast from parameterized import parameterized from dateutil.parser import parse @@ -23,6 +24,7 @@ from posthog.session_recordings.queries.test.session_replay_sql import ( produce_replay_summary, ) +from posthog.models.property import Property from posthog.test.base import ( APIBaseTest, ClickhouseTestMixin, @@ -171,11 +173,17 @@ def test_can_list_recordings_even_when_the_person_has_multiple_distinct_ids(self def test_console_log_filters_are_correctly_passed_to_listing(self, mock_summary_lister): mock_summary_lister.return_value.run.return_value = ([], False) - self.client.get(f'/api/projects/{self.team.id}/session_recordings?console_logs=["warn", "error"]') + params_string = urlencode( + { + "console_log_filters": '[{"key": "console_log_level", "value": ["warn", "error"], "operator": "exact", "type": "recording"}]' + } + ) + self.client.get(f"/api/projects/{self.team.id}/session_recordings?{params_string}") assert len(mock_summary_lister.call_args_list) == 1 filter_passed_to_mock: SessionRecordingsFilter = mock_summary_lister.call_args_list[0].kwargs["filter"] - assert filter_passed_to_mock.console_logs_filter == ["warn", "error"] + console_filter = cast(Property, filter_passed_to_mock.console_log_filters.values[0]) + assert console_filter.value == ["warn", "error"] @snapshot_postgres_queries def test_listing_recordings_is_not_nplus1_for_persons(self): diff --git a/posthog/tasks/alerts/checks.py b/posthog/tasks/alerts/checks.py index e5c34f578a048..cbc4fb4604da5 100644 --- a/posthog/tasks/alerts/checks.py +++ b/posthog/tasks/alerts/checks.py @@ -1,107 +1,138 @@ +from datetime import datetime, timedelta, UTC +from typing import Optional + from celery import shared_task -from celery.canvas import group, chain +from celery.canvas import chain +from django.db import transaction from django.utils import timezone -import math import structlog +from sentry_sdk import capture_exception from posthog.api.services.query import ExecutionMode from posthog.caching.calculate_results import calculate_for_query_based_insight from posthog.email import EmailMessage +from posthog.errors import CHQueryErrorTooManySimultaneousQueries from posthog.hogql_queries.legacy_compatibility.flagged_conversion_manager import ( conversion_to_query_based, ) -from posthog.models import Alert -from posthog.schema import AnomalyCondition +from posthog.models import AlertConfiguration, Team +from posthog.models.alert import AlertCheck +from posthog.tasks.utils import CeleryQueue logger = structlog.get_logger(__name__) def check_all_alerts() -> None: - alert_ids = list(Alert.objects.all().values_list("id", flat=True)) + # TODO: Consider aligning insight calculation with cache warming of insights, see warming.py + # Currently it's implicitly aligned by alerts obviously also using cache if available + + # Use a fixed expiration time since tasks in the chain are executed sequentially + expire_after = datetime.now(UTC) + timedelta(minutes=30) + + teams = Team.objects.filter(alertconfiguration__isnull=False).distinct() - group_count = 10 - # All groups but the last one will have a group_size size. - # The last group will have at most group_size size. - group_size = int(math.ceil(len(alert_ids) / group_count)) + for team in teams: + alert_ids = list(AlertConfiguration.objects.filter(team=team, enabled=True).values_list("id", flat=True)) - groups = [] - for i in range(0, len(alert_ids), group_size): - alert_id_group = alert_ids[i : i + group_size] - chained_calls = chain([check_alert_task.si(alert_id) for alert_id in alert_id_group]) - groups.append(chained_calls) + # We chain the task execution to prevent queries *for a single team* running at the same time + chain(*(check_alert_task.si(str(alert_id)).set(expires=expire_after) for alert_id in alert_ids))() - group(groups).apply_async() +@transaction.atomic +def check_alert(alert_id: str) -> None: + try: + alert = AlertConfiguration.objects.get(id=alert_id, enabled=True) + except AlertConfiguration.DoesNotExist: + logger.warning("Alert not found or not enabled", alert_id=alert_id) + return -def check_alert(alert_id: int) -> None: - alert = Alert.objects.get(pk=alert_id) insight = alert.insight + error: Optional[dict] = None + + try: + with conversion_to_query_based(insight): + calculation_result = calculate_for_query_based_insight( + insight, + execution_mode=ExecutionMode.RECENT_CACHE_CALCULATE_BLOCKING_IF_STALE, + user=None, + ) + + if not calculation_result.result: + raise RuntimeError(f"No results for alert {alert.id}") + + result = calculation_result.result[0] + aggregated_value = result["aggregated_value"] + except Exception as e: + event_id = capture_exception(e) + error = { + "sentry_event_id": event_id, + "message": str(e), + } + aggregated_value = None + + # Lock alert to prevent concurrent state changes + alert = AlertConfiguration.objects.select_for_update().get(id=alert_id, enabled=True) + check, matches = alert.add_check(calculated_value=aggregated_value, error=error) + + if not check.state == "firing": + logger.info("Check state is %s", check.state, alert_id=alert.id) + return - with conversion_to_query_based(insight): - calculation_result = calculate_for_query_based_insight( - insight, - execution_mode=ExecutionMode.RECENT_CACHE_CALCULATE_BLOCKING_IF_STALE, - user=None, - ) - - if not calculation_result.result: - raise RuntimeError(f"No results for alert {alert.id}") - - anomaly_condition = AnomalyCondition.model_validate(alert.anomaly_condition) - thresholds = anomaly_condition.absoluteThreshold - - result = calculation_result.result[0] - aggregated_value = result["aggregated_value"] - anomalies_descriptions = [] - - if thresholds.lower is not None and aggregated_value < thresholds.lower: - anomalies_descriptions += [ - f"The trend value ({aggregated_value}) is below the lower threshold ({thresholds.lower})" - ] - if thresholds.upper is not None and aggregated_value > thresholds.upper: - anomalies_descriptions += [ - f"The trend value ({aggregated_value}) is above the upper threshold ({thresholds.upper})" - ] - - if not anomalies_descriptions: - logger.info("No threshold met", alert_id=alert.id) + if not matches: + # We might be firing but have no (new) matches to notify about return - send_notifications(alert, anomalies_descriptions) + send_notifications(alert, matches) -@shared_task(ignore_result=True) +@shared_task( + ignore_result=True, + expires=60 * 60, +) def check_all_alerts_task() -> None: check_all_alerts() -@shared_task(ignore_result=True) -def check_alert_task(alert_id: int) -> None: +@shared_task( + ignore_result=True, + queue=CeleryQueue.ANALYTICS_LIMITED.value, # Important! Prevents Clickhouse from being overwhelmed + autoretry_for=(CHQueryErrorTooManySimultaneousQueries,), + retry_backoff=1, + retry_backoff_max=10, + max_retries=10, + expires=60 * 60, +) +def check_alert_task(alert_id: str) -> None: check_alert(alert_id) -def send_notifications(alert: Alert, anomalies_descriptions: list[str]) -> None: - subject = f"PostHog alert {alert.name} has anomalies" - campaign_key = f"alert-anomaly-notification-{alert.id}-{timezone.now().timestamp()}" +@shared_task(ignore_result=True) +def checks_cleanup_task() -> None: + AlertCheck.clean_up_old_checks() + + +def send_notifications(alert: AlertConfiguration, matches: list[str]) -> None: + subject = f"PostHog alert {alert.name} is firing" + campaign_key = f"alert-firing-notification-{alert.id}-{timezone.now().timestamp()}" insight_url = f"/project/{alert.team.pk}/insights/{alert.insight.short_id}" alert_url = f"{insight_url}/alerts/{alert.id}" message = EmailMessage( campaign_key=campaign_key, subject=subject, - template_name="alert_anomaly", + template_name="alert_check_firing", template_context={ - "anomalies_descriptions": anomalies_descriptions, + "match_descriptions": matches, "insight_url": insight_url, "insight_name": alert.insight.name, "alert_url": alert_url, "alert_name": alert.name, }, ) - targets = list(filter(len, alert.target_value.split(","))) + targets = alert.subscribed_users.all().values_list("email", flat=True) if not targets: raise RuntimeError(f"no targets configured for the alert {alert.id}") for target in targets: message.add_recipient(email=target) - logger.info(f"Send notifications about {len(anomalies_descriptions)} anomalies", alert_id=alert.id) + logger.info(f"Send notifications about {len(matches)} anomalies", alert_id=alert.id) message.send() diff --git a/posthog/tasks/alerts/test/test_checks.py b/posthog/tasks/alerts/test/test_checks.py index fb5f93b3cb166..2504c90dc3a3f 100644 --- a/posthog/tasks/alerts/test/test_checks.py +++ b/posthog/tasks/alerts/test/test_checks.py @@ -1,16 +1,16 @@ -import pytest from typing import Optional from unittest.mock import MagicMock, patch from freezegun import freeze_time +from posthog.models.alert import AlertCheck from posthog.models.instance_setting import set_instance_setting from posthog.tasks.alerts.checks import send_notifications, check_alert from posthog.test.base import APIBaseTest, _create_event, flush_persons_and_events, ClickhouseDestroyTablesMixin from posthog.api.test.dashboards import DashboardAPI from posthog.schema import ChartDisplayType, EventsNode, TrendsQuery, TrendsFilter from posthog.tasks.test.utils_email_tests import mock_email_messages -from posthog.models import Alert +from posthog.models import AlertConfiguration @freeze_time("2024-06-02T08:55:00.000Z") @@ -41,20 +41,29 @@ def setUp(self) -> None: data={ "name": "alert name", "insight": self.insight["id"], - "target_value": "a@b.c,d@e.f", - "anomaly_condition": {"absoluteThreshold": {}}, + "subscribed_users": [self.user.id], + "threshold": {"configuration": {"absoluteThreshold": {}}}, }, ).json() def set_thresholds(self, lower: Optional[int] = None, upper: Optional[int] = None) -> None: self.client.patch( f"/api/projects/{self.team.id}/alerts/{self.alert['id']}", - data={"anomaly_condition": {"absoluteThreshold": {"lower": lower, "upper": upper}}}, + data={"threshold": {"configuration": {"absoluteThreshold": {"lower": lower, "upper": upper}}}}, ) def get_anomalies_descriptions(self, mock_send_notifications: MagicMock, call_index: int) -> list[str]: return mock_send_notifications.call_args_list[call_index].args[1] + def test_alert_is_not_triggered_when_disabled(self, mock_send_notifications: MagicMock) -> None: + self.set_thresholds(lower=1) + + self.client.patch(f"/api/projects/{self.team.id}/alerts/{self.alert['id']}", data={"enabled": False}) + + check_alert(self.alert["id"]) + + assert mock_send_notifications.call_count == 0 + def test_alert_is_triggered_for_values_above_higher_threshold(self, mock_send_notifications: MagicMock) -> None: self.set_thresholds(upper=0) @@ -70,7 +79,7 @@ def test_alert_is_triggered_for_values_above_higher_threshold(self, mock_send_no assert mock_send_notifications.call_count == 1 alert = mock_send_notifications.call_args_list[0].args[0] - assert alert.id == self.alert["id"] + assert str(alert.id) == self.alert["id"] anomalies_descriptions = self.get_anomalies_descriptions(mock_send_notifications, call_index=0) assert len(anomalies_descriptions) == 1 @@ -100,6 +109,77 @@ def test_alert_is_triggered_for_value_below_lower_threshold(self, mock_send_noti anomalies = self.get_anomalies_descriptions(mock_send_notifications, call_index=0) assert "The trend value (0) is below the lower threshold (1.0)" in anomalies + def test_alert_triggers_but_does_not_send_notification_during_firing( + self, mock_send_notifications: MagicMock + ) -> None: + self.set_thresholds(lower=1) + + check_alert(self.alert["id"]) + + assert mock_send_notifications.call_count == 1 + assert AlertCheck.objects.filter(alert_configuration=self.alert["id"]).latest("created_at").state == "firing" + + with freeze_time("2024-06-02T09:00:00.000Z"): + check_alert(self.alert["id"]) + + assert mock_send_notifications.call_count == 1 + assert ( + AlertCheck.objects.filter(alert_configuration=self.alert["id"]).latest("created_at").state == "firing" + ) + + with freeze_time("2024-06-02T09:55:00.000Z"): + self.set_thresholds(lower=0) + + check_alert(self.alert["id"]) + + assert mock_send_notifications.call_count == 1 + assert ( + AlertCheck.objects.filter(alert_configuration=self.alert["id"]).latest("created_at").state == "not_met" + ) + + with freeze_time("2024-06-02T11:00:00.000Z"): + self.set_thresholds(lower=1) + + check_alert(self.alert["id"]) + + assert mock_send_notifications.call_count == 2 + assert ( + AlertCheck.objects.filter(alert_configuration=self.alert["id"]).latest("created_at").state == "firing" + ) + + # test clean up old checks (> 14 days) + with freeze_time("2024-06-20T11:00:00.000Z"): + AlertCheck.clean_up_old_checks() + assert AlertCheck.objects.filter(alert_configuration=self.alert["id"]).count() == 0 + + def test_alert_is_set_to_inactive_when_disabled(self, mock_send_notifications: MagicMock) -> None: + self.set_thresholds(lower=1) + + check_alert(self.alert["id"]) + + assert mock_send_notifications.call_count == 1 + assert AlertCheck.objects.filter(alert_configuration=self.alert["id"]).latest("created_at").state == "firing" + + self.client.patch(f"/api/projects/{self.team.id}/alerts/{self.alert['id']}", data={"enabled": False}) + + # Check that the alert is set to inactive and checks are not triggered + check_alert(self.alert["id"]) + + assert mock_send_notifications.call_count == 1 + assert AlertConfiguration.objects.get(pk=self.alert["id"]).state == "inactive" + + def test_alert_is_set_to_inactive_when_threshold_changes(self, mock_send_notifications: MagicMock) -> None: + self.set_thresholds(lower=1) + + check_alert(self.alert["id"]) + + assert mock_send_notifications.call_count == 1 + assert AlertCheck.objects.filter(alert_configuration=self.alert["id"]).latest("created_at").state == "firing" + + self.set_thresholds(lower=2) + + assert AlertConfiguration.objects.get(pk=self.alert["id"]).state == "inactive" + def test_alert_is_not_triggered_for_normal_values(self, mock_send_notifications: MagicMock) -> None: self.set_thresholds(lower=0, upper=1) @@ -124,12 +204,30 @@ def test_error_while_calculating_no_alert(self, mock_send_notifications: MagicMo } )[1] - self.client.patch(f"/api/projects/{self.team.id}/alerts/{self.alert['id']}", data={"insight": insight["id"]}) + # Change with ORM to bypass API validation + AlertConfiguration.objects.filter(pk=self.alert["id"]).update(insight=insight["id"]) - with pytest.raises(KeyError): - check_alert(self.alert["id"]) + check_alert(self.alert["id"]) assert mock_send_notifications.call_count == 0 + latest_alert_check = AlertCheck.objects.filter(alert_configuration=self.alert["id"]).latest("created_at") + assert latest_alert_check.error["message"] == "'aggregated_value'" + + # mock calculate_for_query_based_insight to raise a different exception + with patch( + "posthog.tasks.alerts.checks.calculate_for_query_based_insight" + ) as mock_calculate_for_query_based_insight: + mock_calculate_for_query_based_insight.side_effect = Exception("Some error") + + with freeze_time("2024-06-02T09:00:00.000Z"): + check_alert(self.alert["id"]) + assert mock_send_notifications.call_count == 0 + + latest_alert_check = AlertCheck.objects.filter(alert_configuration=self.alert["id"]).latest( + "created_at" + ) + assert latest_alert_check.error["message"] == "Some error" + def test_alert_with_insight_with_filter(self, mock_send_notifications: MagicMock) -> None: insight = self.dashboard_api.create_insight( data={"name": "insight", "filters": {"events": [{"id": "$pageview"}], "display": "BoldNumber"}} @@ -147,13 +245,12 @@ def test_alert_with_insight_with_filter(self, mock_send_notifications: MagicMock @patch("posthog.tasks.alerts.checks.EmailMessage") def test_send_emails(self, MockEmailMessage: MagicMock, mock_send_notifications: MagicMock) -> None: mocked_email_messages = mock_email_messages(MockEmailMessage) - alert = Alert.objects.get(pk=self.alert["id"]) + alert = AlertConfiguration.objects.get(pk=self.alert["id"]) send_notifications(alert, ["first anomaly description", "second anomaly description"]) assert len(mocked_email_messages) == 1 email = mocked_email_messages[0] - assert len(email.to) == 2 - assert email.to[0]["recipient"] == "a@b.c" - assert email.to[1]["recipient"] == "d@e.f" + assert len(email.to) == 1 + assert email.to[0]["recipient"] == "user1@posthog.com" assert "first anomaly description" in email.html_body assert "second anomaly description" in email.html_body diff --git a/posthog/tasks/scheduled.py b/posthog/tasks/scheduled.py index 53127d9ce726d..df765689fb362 100644 --- a/posthog/tasks/scheduled.py +++ b/posthog/tasks/scheduled.py @@ -8,7 +8,7 @@ from posthog.caching.warming import schedule_warming_for_teams_task from posthog.celery import app -from posthog.tasks.alerts.checks import check_all_alerts_task +from posthog.tasks.alerts.checks import check_all_alerts_task, checks_cleanup_task from posthog.tasks.integrations import refresh_integrations from posthog.tasks.tasks import ( calculate_cohort, @@ -244,9 +244,15 @@ def setup_periodic_tasks(sender: Celery, **kwargs: Any) -> None: ) sender.add_periodic_task( - crontab(hour="*", minute="20"), + crontab(hour="*", minute="45"), check_all_alerts_task.s(), - name="detect alerts' anomalies and notify about them", + name="check alerts for matches and send notifications", + ) + + sender.add_periodic_task( + crontab(hour="8", minute="0"), + checks_cleanup_task.s(), + name="clean up old alert checks", ) if settings.EE_AVAILABLE: diff --git a/posthog/templates/email/alert_anomaly.html b/posthog/templates/email/alert_anomaly.html deleted file mode 100644 index 49636488288dc..0000000000000 --- a/posthog/templates/email/alert_anomaly.html +++ /dev/null @@ -1,10 +0,0 @@ -{% extends "email/base.html" %} {% load posthog_assets %} {% block section %} -

    - The {{ alert_name }} alert detected following anomalies for {{ insight_name }}: -

      - {% for anomaly_description in anomalies_descriptions %} -
    • {{ anomaly_description }}
    • - {% endfor %} -
    -

    -{% endblock %}{% load posthog_filters %} diff --git a/posthog/templates/email/alert_check_firing.html b/posthog/templates/email/alert_check_firing.html new file mode 100644 index 0000000000000..52b10b504e175 --- /dev/null +++ b/posthog/templates/email/alert_check_firing.html @@ -0,0 +1,14 @@ +{% extends "email/base.html" %} {% load posthog_assets %} {% block section %} +

    + The {{ alert_name }} alert is firing for {{ insight_name }}: +

    +
      + {% for item in match_descriptions %} +
    • {{ item }}
    • + {% endfor %} +
    + +{% endblock %}{% load posthog_filters %} diff --git a/requirements.in b/requirements.in index 112239492eb0e..ac315f2c5e719 100644 --- a/requirements.in +++ b/requirements.in @@ -97,7 +97,7 @@ phonenumberslite==8.13.6 openai==1.10.0 tiktoken==0.6.0 nh3==0.2.14 -hogql-parser==1.0.36 +hogql-parser==1.0.38 zxcvbn==4.4.28 zstd==1.5.5.1 xmlsec==1.3.13 # Do not change this version - it will break SAML diff --git a/requirements.txt b/requirements.txt index 4e887d218f265..8e64d768b0ec7 100644 --- a/requirements.txt +++ b/requirements.txt @@ -275,7 +275,7 @@ h11==0.13.0 # wsproto hexbytes==1.0.0 # via dlt -hogql-parser==1.0.36 +hogql-parser==1.0.38 # via -r requirements.in httpcore==1.0.2 # via httpx diff --git a/rust/capture/src/config.rs b/rust/capture/src/config.rs index 98c5dbc1d66ff..c6ca77461d11c 100644 --- a/rust/capture/src/config.rs +++ b/rust/capture/src/config.rs @@ -68,6 +68,8 @@ pub struct KafkaConfig { pub kafka_producer_queue_mib: u32, // Size of the in-memory producer queue in mebibytes #[envconfig(default = "20000")] pub kafka_message_timeout_ms: u32, // Time before we stop retrying producing a message: 20 seconds + #[envconfig(default = "1000000")] + pub kafka_producer_message_max_bytes: u32, // message.max.bytes - max kafka message size we will produce #[envconfig(default = "none")] pub kafka_compression_codec: String, // none, gzip, snappy, lz4, zstd pub kafka_hosts: String, diff --git a/rust/capture/src/sinks/kafka.rs b/rust/capture/src/sinks/kafka.rs index 7b3dcdbb8f74e..498b2b71e837a 100644 --- a/rust/capture/src/sinks/kafka.rs +++ b/rust/capture/src/sinks/kafka.rs @@ -130,6 +130,10 @@ impl KafkaSink { .set("statistics.interval.ms", "10000") .set("partitioner", "murmur2_random") // Compatibility with python-kafka .set("linger.ms", config.kafka_producer_linger_ms.to_string()) + .set( + "message.max.bytes", + config.kafka_producer_message_max_bytes.to_string(), + ) .set( "message.timeout.ms", config.kafka_message_timeout_ms.to_string(), @@ -328,7 +332,9 @@ mod tests { use std::num::NonZeroU32; use time::Duration; - async fn start_on_mocked_sink() -> (MockCluster<'static, DefaultProducerContext>, KafkaSink) { + async fn start_on_mocked_sink( + message_max_bytes: Option, + ) -> (MockCluster<'static, DefaultProducerContext>, KafkaSink) { let registry = HealthRegistry::new("liveness"); let handle = registry .register("one".to_string(), Duration::seconds(30)) @@ -343,6 +349,7 @@ mod tests { kafka_producer_linger_ms: 0, kafka_producer_queue_mib: 50, kafka_message_timeout_ms: 500, + kafka_producer_message_max_bytes: message_max_bytes.unwrap_or(1000000), kafka_compression_codec: "none".to_string(), kafka_hosts: cluster.bootstrap_servers(), kafka_topic: "events_plugin_ingestion".to_string(), @@ -361,7 +368,7 @@ mod tests { // Uses a mocked Kafka broker that allows injecting write errors, to check error handling. // We test different cases in a single test to amortize the startup cost of the producer. - let (cluster, sink) = start_on_mocked_sink().await; + let (cluster, sink) = start_on_mocked_sink(Some(3000000)).await; let event: ProcessedEvent = ProcessedEvent { data_type: DataType::AnalyticsMain, uuid: uuid_v7(), @@ -389,7 +396,7 @@ mod tests { .await .expect("failed to send initial event batch"); - // Producer should reject a 2MB message, twice the default `message.max.bytes` + // Producer should accept a 2MB message as we set message.max.bytes to 3MB let big_data = rand::thread_rng() .sample_iter(Alphanumeric) .take(2_000_000) @@ -406,6 +413,27 @@ mod tests { token: "token1".to_string(), session_id: None, }; + sink.send(big_event) + .await + .expect("failed to send event larger than default max size"); + + // Producer should reject a 4MB message + let big_data = rand::thread_rng() + .sample_iter(Alphanumeric) + .take(4_000_000) + .map(char::from) + .collect(); + let big_event: ProcessedEvent = ProcessedEvent { + data_type: DataType::AnalyticsMain, + uuid: uuid_v7(), + distinct_id: "id1".to_string(), + ip: "".to_string(), + data: big_data, + now: "".to_string(), + sent_at: None, + token: "token1".to_string(), + session_id: None, + }; match sink.send(big_event).await { Err(CaptureError::EventTooBig) => {} // Expected Err(err) => panic!("wrong error code {}", err), diff --git a/rust/capture/tests/common.rs b/rust/capture/tests/common.rs index bf76616229292..ee0b8659b42e5 100644 --- a/rust/capture/tests/common.rs +++ b/rust/capture/tests/common.rs @@ -41,6 +41,7 @@ pub static DEFAULT_CONFIG: Lazy = Lazy::new(|| Config { kafka_producer_linger_ms: 0, // Send messages as soon as possible kafka_producer_queue_mib: 10, kafka_message_timeout_ms: 10000, // 10s, ACKs can be slow on low volumes, should be tuned + kafka_producer_message_max_bytes: 1000000, // 1MB, rdkafka default kafka_compression_codec: "none".to_string(), kafka_hosts: "kafka:9092".to_string(), kafka_topic: "events_plugin_ingestion".to_string(),