@rawdash/connector-google-analytics 0.15.0 → 0.16.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +84 -147
- package/dist/index.d.ts +171 -3
- package/dist/index.js +148 -13
- package/dist/index.js.map +1 -1
- package/package.json +4 -4
package/README.md
CHANGED
|
@@ -1,119 +1,104 @@
|
|
|
1
|
+
<!-- This file is generated from connector metadata by scripts/generate-connector-docs.ts. Do not edit by hand. -->
|
|
2
|
+
|
|
1
3
|
# @rawdash/connector-google-analytics
|
|
2
4
|
|
|
3
|
-
|
|
5
|
+
[](https://www.npmjs.com/package/@rawdash/connector-google-analytics)
|
|
6
|
+
[](https://github.com/rawdash/rawdash/blob/main/LICENSE)
|
|
4
7
|
|
|
5
|
-
|
|
8
|
+
Sync daily GA4 traffic, acquisition, top pages, events, conversions, and geography metrics from a Google Analytics 4 property.
|
|
6
9
|
|
|
7
|
-
|
|
10
|
+
## Install
|
|
8
11
|
|
|
9
|
-
|
|
12
|
+
```sh
|
|
13
|
+
npm install @rawdash/connector-google-analytics
|
|
14
|
+
```
|
|
10
15
|
|
|
11
|
-
|
|
12
|
-
2. Navigate to **IAM & Admin → Service Accounts** and click **Create Service Account**.
|
|
13
|
-
3. Give it a name (e.g. `rawdash-ga4-reader`) and click **Create and continue**.
|
|
14
|
-
4. Skip the optional role assignment on this screen — click **Done**.
|
|
15
|
-
5. In Google Analytics, go to **Admin → Account Access Management** and add the service account email with the **Viewer** role.
|
|
16
|
-
6. Back in Cloud Console, open the service account → **Keys** → **Add key → Create new key → JSON**.
|
|
17
|
-
7. Download the `.json` file and store its contents as the secret `GA_SERVICE_ACCOUNT_JSON`.
|
|
16
|
+
## Authentication
|
|
18
17
|
|
|
19
|
-
|
|
18
|
+
Authenticate against the GA4 Data API with either a Google service account JSON key (recommended) or an OAuth 2.0 refresh-token tuple. The identity must have at least the Analytics Viewer role on the property.
|
|
20
19
|
|
|
21
|
-
1.
|
|
22
|
-
2.
|
|
23
|
-
3.
|
|
24
|
-
4. Store the refresh token as `GA_REFRESH_TOKEN`, the client ID as `GA_CLIENT_ID`, and the client secret as `GA_CLIENT_SECRET`.
|
|
20
|
+
1. Find your GA4 Property ID under Google Analytics -> Admin -> Property settings (numeric, e.g. 123456789).
|
|
21
|
+
2. Recommended: create a service account at Google Cloud -> IAM & Admin -> Service Accounts, generate a JSON key, and grant it the Analytics Viewer role on the property. Store the JSON as a secret and reference it as serviceAccountJson: secret("GA4_SERVICE_ACCOUNT_JSON").
|
|
22
|
+
3. Alternative: provide an OAuth 2.0 refresh token with the analytics.readonly scope together with its clientId and clientSecret from the Google Cloud Console.
|
|
25
23
|
|
|
26
24
|
## Configuration
|
|
27
25
|
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
26
|
+
| Field | Type | Required | Description |
|
|
27
|
+
| -------------------- | ------ | -------- | ------------------------------------------------------------------------------------------------------------------------------------------------------- |
|
|
28
|
+
| `propertyId` | string | Yes | Numeric ID of your GA4 property (e.g. 123456789). Find it in Google Analytics → Admin → Property settings. |
|
|
29
|
+
| `serviceAccountJson` | secret | No | Contents of the JSON key file for a Google service account with the Analytics Viewer role. Create one at Google Cloud → IAM & Admin → Service Accounts. |
|
|
30
|
+
| `refreshToken` | secret | No | Google OAuth 2.0 refresh token with analytics.readonly scope. Required if not using serviceAccountJson. |
|
|
31
|
+
| `clientId` | string | No | OAuth 2.0 client ID from Google Cloud Console. Required when using refreshToken auth. |
|
|
32
|
+
| `clientSecret` | secret | No | OAuth 2.0 client secret from Google Cloud Console. Required when using refreshToken auth. |
|
|
33
|
+
| `lookbackDays` | number | No | How many calendar days to fetch on a full sync. Defaults to 90. |
|
|
34
|
+
|
|
35
|
+
## Resources
|
|
36
|
+
|
|
37
|
+
- **`ga4_traffic_by_day`** _(metric)_ - Daily site traffic totals - sessions, total users, new users, page views, and engagement rate.
|
|
38
|
+
- Endpoint: `POST /v1beta/properties/{propertyId}:runReport`
|
|
39
|
+
- Unit: sessions
|
|
40
|
+
- Granularity: day
|
|
41
|
+
- Dimensions: `date`
|
|
42
|
+
- **`ga4_traffic_by_source`** _(metric)_ - Daily sessions and conversions broken down by acquisition source and medium.
|
|
43
|
+
- Endpoint: `POST /v1beta/properties/{propertyId}:runReport`
|
|
44
|
+
- Unit: sessions
|
|
45
|
+
- Granularity: day
|
|
46
|
+
- Dimensions: `date`, `sessionSource`, `sessionMedium`
|
|
47
|
+
- **`ga4_top_pages`** _(metric)_ - Daily page views and average session duration bucketed by page path.
|
|
48
|
+
- Endpoint: `POST /v1beta/properties/{propertyId}:runReport`
|
|
49
|
+
- Unit: page_views
|
|
50
|
+
- Granularity: day
|
|
51
|
+
- Dimensions: `date`, `pagePath`
|
|
52
|
+
- **`ga4_events`** _(metric)_ - Daily event counts and the users that triggered them, bucketed by event name.
|
|
53
|
+
- Endpoint: `POST /v1beta/properties/{propertyId}:runReport`
|
|
54
|
+
- Unit: events
|
|
55
|
+
- Granularity: day
|
|
56
|
+
- Dimensions: `date`, `eventName`
|
|
57
|
+
- **`ga4_conversions`** _(metric)_ - Daily conversion counts and total revenue bucketed by conversion event name.
|
|
58
|
+
- Endpoint: `POST /v1beta/properties/{propertyId}:runReport`
|
|
59
|
+
- Unit: conversions
|
|
60
|
+
- Granularity: day
|
|
61
|
+
- Dimensions: `date`, `eventName`
|
|
62
|
+
- **`ga4_geo`** _(metric)_ - Daily sessions and total users bucketed by visitor country.
|
|
63
|
+
- Endpoint: `POST /v1beta/properties/{propertyId}:runReport`
|
|
64
|
+
- Unit: sessions
|
|
65
|
+
- Granularity: day
|
|
66
|
+
- Dimensions: `date`, `country`
|
|
67
|
+
|
|
68
|
+
## Example
|
|
44
69
|
|
|
45
70
|
```ts
|
|
46
|
-
import {
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
71
|
+
import {
|
|
72
|
+
defineConfig,
|
|
73
|
+
defineDashboard,
|
|
74
|
+
defineMetric,
|
|
75
|
+
secret,
|
|
76
|
+
} from '@rawdash/core';
|
|
77
|
+
|
|
78
|
+
const googleAnalytics = {
|
|
79
|
+
name: 'googleAnalytics',
|
|
50
80
|
connectorId: 'google-analytics',
|
|
51
81
|
config: {
|
|
52
82
|
propertyId: '123456789',
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
clientSecret: secret('GA_CLIENT_SECRET'),
|
|
83
|
+
serviceAccountJson: secret('GA4_SERVICE_ACCOUNT_JSON'),
|
|
84
|
+
lookbackDays: 90,
|
|
56
85
|
},
|
|
57
86
|
};
|
|
58
|
-
```
|
|
59
|
-
|
|
60
|
-
Register the connector class when mounting the engine:
|
|
61
|
-
|
|
62
|
-
```ts
|
|
63
|
-
import { GA4Connector } from '@rawdash/connector-google-analytics';
|
|
64
|
-
import { mountEngine } from '@rawdash/hono';
|
|
65
|
-
|
|
66
|
-
mountEngine(config, {
|
|
67
|
-
connectorRegistry: { 'google-analytics': GA4Connector },
|
|
68
|
-
});
|
|
69
|
-
```
|
|
70
|
-
|
|
71
|
-
Then wire it into `defineConfig`:
|
|
72
|
-
|
|
73
|
-
```ts
|
|
74
|
-
import { defineConfig, defineDashboard, defineMetric } from '@rawdash/core';
|
|
75
87
|
|
|
76
88
|
export default defineConfig({
|
|
77
|
-
connectors: [
|
|
89
|
+
connectors: [googleAnalytics],
|
|
78
90
|
dashboards: {
|
|
79
|
-
|
|
91
|
+
traffic: defineDashboard({
|
|
80
92
|
widgets: {
|
|
81
|
-
|
|
82
|
-
kind: 'stat',
|
|
83
|
-
title: 'Sessions today',
|
|
84
|
-
metric: defineMetric({
|
|
85
|
-
connector: ga4,
|
|
86
|
-
shape: 'metric',
|
|
87
|
-
name: 'ga4_traffic_by_day',
|
|
88
|
-
field: 'sessions',
|
|
89
|
-
fn: 'sum',
|
|
90
|
-
window: '1d',
|
|
91
|
-
}),
|
|
92
|
-
},
|
|
93
|
-
sessions_over_time: {
|
|
93
|
+
sessions: {
|
|
94
94
|
kind: 'timeseries',
|
|
95
|
-
title: '
|
|
95
|
+
title: 'Daily sessions',
|
|
96
96
|
window: '30d',
|
|
97
97
|
metric: defineMetric({
|
|
98
|
-
connector:
|
|
98
|
+
connector: googleAnalytics,
|
|
99
99
|
shape: 'metric',
|
|
100
100
|
name: 'ga4_traffic_by_day',
|
|
101
|
-
field: 'sessions',
|
|
102
|
-
fn: 'sum',
|
|
103
|
-
window: '30d',
|
|
104
|
-
groupBy: { field: 'ts', granularity: 'day' },
|
|
105
|
-
}),
|
|
106
|
-
},
|
|
107
|
-
traffic_by_source: {
|
|
108
|
-
kind: 'distribution',
|
|
109
|
-
title: 'Traffic by source',
|
|
110
|
-
metric: defineMetric({
|
|
111
|
-
connector: ga4,
|
|
112
|
-
shape: 'metric',
|
|
113
|
-
name: 'ga4_traffic_by_source',
|
|
114
|
-
field: 'sessions',
|
|
115
101
|
fn: 'sum',
|
|
116
|
-
window: '30d',
|
|
117
102
|
}),
|
|
118
103
|
},
|
|
119
104
|
},
|
|
@@ -122,69 +107,21 @@ export default defineConfig({
|
|
|
122
107
|
});
|
|
123
108
|
```
|
|
124
109
|
|
|
125
|
-
##
|
|
110
|
+
## Rate limits
|
|
126
111
|
|
|
127
|
-
|
|
112
|
+
GA4 Data API quota is 200,000 tokens/day per property (default); 429 responses are retried automatically with exponential backoff.
|
|
128
113
|
|
|
129
|
-
|
|
130
|
-
| ----------------------- | ---------------------------------- | --------------------------------------------------------------- |
|
|
131
|
-
| `ga4_traffic_by_day` | date | sessions, totalUsers, newUsers, screenPageViews, engagementRate |
|
|
132
|
-
| `ga4_traffic_by_source` | date, sessionSource, sessionMedium | sessions, conversions |
|
|
133
|
-
| `ga4_top_pages` | date, pagePath | screenPageViews, averageSessionDuration |
|
|
134
|
-
| `ga4_events` | date, eventName | eventCount, totalUsers |
|
|
135
|
-
| `ga4_conversions` | date, eventName | conversions, totalRevenue |
|
|
136
|
-
| `ga4_geo` | date, country | sessions, totalUsers |
|
|
114
|
+
## Limitations
|
|
137
115
|
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
## Schemas
|
|
141
|
-
|
|
142
|
-
`GA4Connector.schemas` declares the Zod schema for each `request()` resource — one per `runReport` phase, plus the OAuth token exchange. Used by the cloud shape-drift pipeline to populate `connector_baselines`, and by the package's property tests.
|
|
143
|
-
|
|
144
|
-
| Resource | Represents |
|
|
145
|
-
| ------------------- | --------------------------------------------------------------------------- |
|
|
146
|
-
| `oauth_token` | `POST https://oauth2.googleapis.com/token` — access-token exchange response |
|
|
147
|
-
| `traffic_by_day` | `properties/{id}:runReport` with `dimensions=[date]` |
|
|
148
|
-
| `traffic_by_source` | `runReport` with `dimensions=[date, sessionSource, sessionMedium]` |
|
|
149
|
-
| `top_pages` | `runReport` with `dimensions=[date, pagePath]` |
|
|
150
|
-
| `events` | `runReport` with `dimensions=[date, eventName]` |
|
|
151
|
-
| `conversions` | `runReport` with `dimensions=[date, eventName]` |
|
|
152
|
-
| `geo` | `runReport` with `dimensions=[date, country]` |
|
|
153
|
-
|
|
154
|
-
## Sync behaviour
|
|
155
|
-
|
|
156
|
-
- **Backfill** (`mode: 'full'`): fetches a rolling window (default 90 days, configurable via `lookbackDays`) for all six resources.
|
|
157
|
-
- **Incremental** (`mode: 'latest'`): fetches the trailing 30 days to catch late-arriving attribution data (GA4 can attribute conversions up to 3 days after the session).
|
|
158
|
-
- Both modes **clear existing metric data** for each resource before re-inserting, preventing duplicate rows from accumulating across sync runs.
|
|
159
|
-
- **Pagination**: uses the GA4 Data API `offset`/`limit` model with 10 000 rows per page. Interrupted syncs return a cursor and resume from the same phase and offset.
|
|
160
|
-
- **Rate limits**: the GA4 Data API quota is 200 000 tokens/day per property (default). 429 responses are handled automatically by the built-in HTTP client with exponential back-off.
|
|
161
|
-
|
|
162
|
-
## Registering in the MCP server
|
|
163
|
-
|
|
164
|
-
```ts
|
|
165
|
-
import {
|
|
166
|
-
GA4Connector,
|
|
167
|
-
configFields,
|
|
168
|
-
} from '@rawdash/connector-google-analytics';
|
|
169
|
-
|
|
170
|
-
createMcpServer({
|
|
171
|
-
// ...
|
|
172
|
-
connectorFactories: [
|
|
173
|
-
{
|
|
174
|
-
id: 'google-analytics',
|
|
175
|
-
configFields,
|
|
176
|
-
create: GA4Connector.create,
|
|
177
|
-
},
|
|
178
|
-
],
|
|
179
|
-
});
|
|
180
|
-
```
|
|
116
|
+
- Incremental syncs use a 30-day window because GA4 can attribute conversions up to 3 days after the session.
|
|
117
|
+
- Report pagination is 10,000 rows per page.
|
|
181
118
|
|
|
182
|
-
##
|
|
119
|
+
## Links
|
|
183
120
|
|
|
184
|
-
|
|
121
|
+
- [Rawdash docs](https://rawdash.dev/docs/connectors/)
|
|
122
|
+
- [Google Analytics API docs](https://developers.google.com/analytics/devguides/reporting/data/v1)
|
|
123
|
+
- [GitHub](https://github.com/rawdash/rawdash)
|
|
185
124
|
|
|
186
|
-
|
|
187
|
-
2. Pipe them through `connector.sync()` against an `InMemoryStorage` instance.
|
|
188
|
-
3. Assert universal invariants — non-empty entity ids, finite event timestamps, no `undefined` leaking into storage, no thrown errors on any valid input — plus per-resource counts.
|
|
125
|
+
## License
|
|
189
126
|
|
|
190
|
-
|
|
127
|
+
Apache-2.0
|
package/dist/index.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { BaseConnector, ConnectorContext, SyncOptions, StorageHandle, SyncResult } from '@rawdash/core';
|
|
1
|
+
import { BaseConnector, ConnectorContext, SyncOptions, StorageHandle, SyncResult, ConnectorDoc } from '@rawdash/core';
|
|
2
2
|
import { z } from 'zod';
|
|
3
3
|
|
|
4
4
|
declare const configFields: z.ZodObject<{
|
|
@@ -15,6 +15,7 @@ declare const configFields: z.ZodObject<{
|
|
|
15
15
|
}, z.core.$strip>>;
|
|
16
16
|
lookbackDays: z.ZodOptional<z.ZodNumber>;
|
|
17
17
|
}, z.core.$strip>;
|
|
18
|
+
declare const doc: ConnectorDoc;
|
|
18
19
|
interface GA4Settings {
|
|
19
20
|
propertyId: string;
|
|
20
21
|
lookbackDays?: number;
|
|
@@ -56,6 +57,168 @@ declare function rowToMetricSample(row: GA4ReportRow, dimensionHeaders: string[]
|
|
|
56
57
|
};
|
|
57
58
|
declare class GA4Connector extends BaseConnector<GA4Settings, GA4Credentials> {
|
|
58
59
|
static readonly id = "google-analytics";
|
|
60
|
+
static readonly resources: {
|
|
61
|
+
readonly ga4_traffic_by_day: {
|
|
62
|
+
readonly shape: "metric";
|
|
63
|
+
readonly description: "Daily site traffic totals - sessions, total users, new users, page views, and engagement rate.";
|
|
64
|
+
readonly unit: "sessions";
|
|
65
|
+
readonly granularity: "day";
|
|
66
|
+
readonly endpoint: "POST /v1beta/properties/{propertyId}:runReport";
|
|
67
|
+
readonly dimensions: [{
|
|
68
|
+
readonly name: "date";
|
|
69
|
+
readonly description: "Calendar day of the metric sample.";
|
|
70
|
+
}];
|
|
71
|
+
readonly responses: {
|
|
72
|
+
readonly oauth_token: z.ZodObject<{
|
|
73
|
+
access_token: z.ZodString;
|
|
74
|
+
expires_in: z.ZodOptional<z.ZodNumber>;
|
|
75
|
+
}, z.core.$strip>;
|
|
76
|
+
readonly traffic_by_day: z.ZodObject<{
|
|
77
|
+
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
78
|
+
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
79
|
+
value: z.ZodString;
|
|
80
|
+
}, z.core.$strip>, ...z.ZodType<unknown, unknown, z.core.$ZodTypeInternals<unknown, unknown>>[]], null>;
|
|
81
|
+
metricValues: z.ZodArray<z.ZodObject<{
|
|
82
|
+
value: z.ZodString;
|
|
83
|
+
}, z.core.$strip>>;
|
|
84
|
+
}, z.core.$strip>>>;
|
|
85
|
+
}, z.core.$strip>;
|
|
86
|
+
};
|
|
87
|
+
};
|
|
88
|
+
readonly ga4_traffic_by_source: {
|
|
89
|
+
readonly shape: "metric";
|
|
90
|
+
readonly description: "Daily sessions and conversions broken down by acquisition source and medium.";
|
|
91
|
+
readonly unit: "sessions";
|
|
92
|
+
readonly granularity: "day";
|
|
93
|
+
readonly endpoint: "POST /v1beta/properties/{propertyId}:runReport";
|
|
94
|
+
readonly dimensions: [{
|
|
95
|
+
readonly name: "date";
|
|
96
|
+
readonly description: "Calendar day of the metric sample.";
|
|
97
|
+
}, {
|
|
98
|
+
readonly name: "sessionSource";
|
|
99
|
+
readonly description: "Origin of the session (e.g. google, direct, newsletter).";
|
|
100
|
+
}, {
|
|
101
|
+
readonly name: "sessionMedium";
|
|
102
|
+
readonly description: "Acquisition medium of the session (e.g. organic, cpc, referral).";
|
|
103
|
+
}];
|
|
104
|
+
readonly responses: {
|
|
105
|
+
readonly traffic_by_source: z.ZodObject<{
|
|
106
|
+
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
107
|
+
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
108
|
+
value: z.ZodString;
|
|
109
|
+
}, z.core.$strip>, ...z.ZodType<unknown, unknown, z.core.$ZodTypeInternals<unknown, unknown>>[]], null>;
|
|
110
|
+
metricValues: z.ZodArray<z.ZodObject<{
|
|
111
|
+
value: z.ZodString;
|
|
112
|
+
}, z.core.$strip>>;
|
|
113
|
+
}, z.core.$strip>>>;
|
|
114
|
+
}, z.core.$strip>;
|
|
115
|
+
};
|
|
116
|
+
};
|
|
117
|
+
readonly ga4_top_pages: {
|
|
118
|
+
readonly shape: "metric";
|
|
119
|
+
readonly description: "Daily page views and average session duration bucketed by page path.";
|
|
120
|
+
readonly unit: "page_views";
|
|
121
|
+
readonly granularity: "day";
|
|
122
|
+
readonly endpoint: "POST /v1beta/properties/{propertyId}:runReport";
|
|
123
|
+
readonly dimensions: [{
|
|
124
|
+
readonly name: "date";
|
|
125
|
+
readonly description: "Calendar day of the metric sample.";
|
|
126
|
+
}, {
|
|
127
|
+
readonly name: "pagePath";
|
|
128
|
+
readonly description: "URL path of the page that was viewed.";
|
|
129
|
+
}];
|
|
130
|
+
readonly responses: {
|
|
131
|
+
readonly top_pages: z.ZodObject<{
|
|
132
|
+
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
133
|
+
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
134
|
+
value: z.ZodString;
|
|
135
|
+
}, z.core.$strip>, ...z.ZodType<unknown, unknown, z.core.$ZodTypeInternals<unknown, unknown>>[]], null>;
|
|
136
|
+
metricValues: z.ZodArray<z.ZodObject<{
|
|
137
|
+
value: z.ZodString;
|
|
138
|
+
}, z.core.$strip>>;
|
|
139
|
+
}, z.core.$strip>>>;
|
|
140
|
+
}, z.core.$strip>;
|
|
141
|
+
};
|
|
142
|
+
};
|
|
143
|
+
readonly ga4_events: {
|
|
144
|
+
readonly shape: "metric";
|
|
145
|
+
readonly description: "Daily event counts and the users that triggered them, bucketed by event name.";
|
|
146
|
+
readonly unit: "events";
|
|
147
|
+
readonly granularity: "day";
|
|
148
|
+
readonly endpoint: "POST /v1beta/properties/{propertyId}:runReport";
|
|
149
|
+
readonly dimensions: [{
|
|
150
|
+
readonly name: "date";
|
|
151
|
+
readonly description: "Calendar day of the metric sample.";
|
|
152
|
+
}, {
|
|
153
|
+
readonly name: "eventName";
|
|
154
|
+
readonly description: "GA4 event name (e.g. page_view, scroll, click).";
|
|
155
|
+
}];
|
|
156
|
+
readonly responses: {
|
|
157
|
+
readonly events: z.ZodObject<{
|
|
158
|
+
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
159
|
+
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
160
|
+
value: z.ZodString;
|
|
161
|
+
}, z.core.$strip>, ...z.ZodType<unknown, unknown, z.core.$ZodTypeInternals<unknown, unknown>>[]], null>;
|
|
162
|
+
metricValues: z.ZodArray<z.ZodObject<{
|
|
163
|
+
value: z.ZodString;
|
|
164
|
+
}, z.core.$strip>>;
|
|
165
|
+
}, z.core.$strip>>>;
|
|
166
|
+
}, z.core.$strip>;
|
|
167
|
+
};
|
|
168
|
+
};
|
|
169
|
+
readonly ga4_conversions: {
|
|
170
|
+
readonly shape: "metric";
|
|
171
|
+
readonly description: "Daily conversion counts and total revenue bucketed by conversion event name.";
|
|
172
|
+
readonly unit: "conversions";
|
|
173
|
+
readonly granularity: "day";
|
|
174
|
+
readonly endpoint: "POST /v1beta/properties/{propertyId}:runReport";
|
|
175
|
+
readonly dimensions: [{
|
|
176
|
+
readonly name: "date";
|
|
177
|
+
readonly description: "Calendar day of the metric sample.";
|
|
178
|
+
}, {
|
|
179
|
+
readonly name: "eventName";
|
|
180
|
+
readonly description: "GA4 conversion event name (e.g. purchase, generate_lead).";
|
|
181
|
+
}];
|
|
182
|
+
readonly responses: {
|
|
183
|
+
readonly conversions: z.ZodObject<{
|
|
184
|
+
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
185
|
+
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
186
|
+
value: z.ZodString;
|
|
187
|
+
}, z.core.$strip>, ...z.ZodType<unknown, unknown, z.core.$ZodTypeInternals<unknown, unknown>>[]], null>;
|
|
188
|
+
metricValues: z.ZodArray<z.ZodObject<{
|
|
189
|
+
value: z.ZodString;
|
|
190
|
+
}, z.core.$strip>>;
|
|
191
|
+
}, z.core.$strip>>>;
|
|
192
|
+
}, z.core.$strip>;
|
|
193
|
+
};
|
|
194
|
+
};
|
|
195
|
+
readonly ga4_geo: {
|
|
196
|
+
readonly shape: "metric";
|
|
197
|
+
readonly description: "Daily sessions and total users bucketed by visitor country.";
|
|
198
|
+
readonly unit: "sessions";
|
|
199
|
+
readonly granularity: "day";
|
|
200
|
+
readonly endpoint: "POST /v1beta/properties/{propertyId}:runReport";
|
|
201
|
+
readonly dimensions: [{
|
|
202
|
+
readonly name: "date";
|
|
203
|
+
readonly description: "Calendar day of the metric sample.";
|
|
204
|
+
}, {
|
|
205
|
+
readonly name: "country";
|
|
206
|
+
readonly description: "Country the session originated from.";
|
|
207
|
+
}];
|
|
208
|
+
readonly responses: {
|
|
209
|
+
readonly geo: z.ZodObject<{
|
|
210
|
+
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
211
|
+
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
212
|
+
value: z.ZodString;
|
|
213
|
+
}, z.core.$strip>, ...z.ZodType<unknown, unknown, z.core.$ZodTypeInternals<unknown, unknown>>[]], null>;
|
|
214
|
+
metricValues: z.ZodArray<z.ZodObject<{
|
|
215
|
+
value: z.ZodString;
|
|
216
|
+
}, z.core.$strip>>;
|
|
217
|
+
}, z.core.$strip>>>;
|
|
218
|
+
}, z.core.$strip>;
|
|
219
|
+
};
|
|
220
|
+
};
|
|
221
|
+
};
|
|
59
222
|
static readonly schemas: {
|
|
60
223
|
readonly oauth_token: z.ZodObject<{
|
|
61
224
|
access_token: z.ZodString;
|
|
@@ -71,6 +234,7 @@ declare class GA4Connector extends BaseConnector<GA4Settings, GA4Credentials> {
|
|
|
71
234
|
}, z.core.$strip>>;
|
|
72
235
|
}, z.core.$strip>>>;
|
|
73
236
|
}, z.core.$strip>;
|
|
237
|
+
} & {
|
|
74
238
|
readonly traffic_by_source: z.ZodObject<{
|
|
75
239
|
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
76
240
|
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
@@ -81,6 +245,7 @@ declare class GA4Connector extends BaseConnector<GA4Settings, GA4Credentials> {
|
|
|
81
245
|
}, z.core.$strip>>;
|
|
82
246
|
}, z.core.$strip>>>;
|
|
83
247
|
}, z.core.$strip>;
|
|
248
|
+
} & {
|
|
84
249
|
readonly top_pages: z.ZodObject<{
|
|
85
250
|
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
86
251
|
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
@@ -91,6 +256,7 @@ declare class GA4Connector extends BaseConnector<GA4Settings, GA4Credentials> {
|
|
|
91
256
|
}, z.core.$strip>>;
|
|
92
257
|
}, z.core.$strip>>>;
|
|
93
258
|
}, z.core.$strip>;
|
|
259
|
+
} & {
|
|
94
260
|
readonly events: z.ZodObject<{
|
|
95
261
|
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
96
262
|
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
@@ -101,6 +267,7 @@ declare class GA4Connector extends BaseConnector<GA4Settings, GA4Credentials> {
|
|
|
101
267
|
}, z.core.$strip>>;
|
|
102
268
|
}, z.core.$strip>>>;
|
|
103
269
|
}, z.core.$strip>;
|
|
270
|
+
} & {
|
|
104
271
|
readonly conversions: z.ZodObject<{
|
|
105
272
|
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
106
273
|
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
@@ -111,6 +278,7 @@ declare class GA4Connector extends BaseConnector<GA4Settings, GA4Credentials> {
|
|
|
111
278
|
}, z.core.$strip>>;
|
|
112
279
|
}, z.core.$strip>>>;
|
|
113
280
|
}, z.core.$strip>;
|
|
281
|
+
} & {
|
|
114
282
|
readonly geo: z.ZodObject<{
|
|
115
283
|
rows: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
116
284
|
dimensionValues: z.ZodTuple<[z.ZodObject<{
|
|
@@ -121,7 +289,7 @@ declare class GA4Connector extends BaseConnector<GA4Settings, GA4Credentials> {
|
|
|
121
289
|
}, z.core.$strip>>;
|
|
122
290
|
}, z.core.$strip>>>;
|
|
123
291
|
}, z.core.$strip>;
|
|
124
|
-
}
|
|
292
|
+
} & Readonly<Record<string, z.ZodType<unknown, unknown, z.core.$ZodTypeInternals<unknown, unknown>>>>;
|
|
125
293
|
static create(input: unknown, ctx?: ConnectorContext): GA4Connector;
|
|
126
294
|
readonly id = "google-analytics";
|
|
127
295
|
readonly credentials: {
|
|
@@ -149,4 +317,4 @@ declare class GA4Connector extends BaseConnector<GA4Settings, GA4Credentials> {
|
|
|
149
317
|
sync(options: SyncOptions, storage: StorageHandle, signal?: AbortSignal): Promise<SyncResult>;
|
|
150
318
|
}
|
|
151
319
|
|
|
152
|
-
export { GA4Connector, type GA4Settings, configFields, GA4Connector as default, rowToMetricSample };
|
|
320
|
+
export { GA4Connector, type GA4Settings, configFields, GA4Connector as default, doc, rowToMetricSample };
|
package/dist/index.js
CHANGED
|
@@ -1,7 +1,17 @@
|
|
|
1
|
+
// ../../connector-shared/dist/index.js
|
|
2
|
+
var HTTP_CLIENT_VERSION = "0.0.0";
|
|
3
|
+
var DEFAULT_USER_AGENT = `rawdash-connector/${HTTP_CLIENT_VERSION} (+https://rawdash.dev)`;
|
|
4
|
+
function connectorUserAgent(connectorId) {
|
|
5
|
+
return `rawdash-connector-${connectorId}/${HTTP_CLIENT_VERSION} (+https://rawdash.dev)`;
|
|
6
|
+
}
|
|
7
|
+
|
|
1
8
|
// src/google-analytics.ts
|
|
2
9
|
import {
|
|
3
10
|
BaseConnector,
|
|
4
|
-
defineConfigFields
|
|
11
|
+
defineConfigFields,
|
|
12
|
+
defineConnectorDoc,
|
|
13
|
+
defineResources,
|
|
14
|
+
schemasFromResources
|
|
5
15
|
} from "@rawdash/core";
|
|
6
16
|
import { z } from "zod";
|
|
7
17
|
var configFields = defineConfigFields(
|
|
@@ -43,6 +53,30 @@ var configFields = defineConfigFields(
|
|
|
43
53
|
}
|
|
44
54
|
)
|
|
45
55
|
);
|
|
56
|
+
var doc = defineConnectorDoc({
|
|
57
|
+
displayName: "Google Analytics",
|
|
58
|
+
category: "analytics",
|
|
59
|
+
brandColor: "#E37400",
|
|
60
|
+
tagline: "Sync daily GA4 traffic, acquisition, top pages, events, conversions, and geography metrics from a Google Analytics 4 property.",
|
|
61
|
+
vendor: {
|
|
62
|
+
name: "Google Analytics",
|
|
63
|
+
apiDocs: "https://developers.google.com/analytics/devguides/reporting/data/v1",
|
|
64
|
+
website: "https://analytics.google.com"
|
|
65
|
+
},
|
|
66
|
+
auth: {
|
|
67
|
+
summary: "Authenticate against the GA4 Data API with either a Google service account JSON key (recommended) or an OAuth 2.0 refresh-token tuple. The identity must have at least the Analytics Viewer role on the property.",
|
|
68
|
+
setup: [
|
|
69
|
+
"Find your GA4 Property ID under Google Analytics -> Admin -> Property settings (numeric, e.g. 123456789).",
|
|
70
|
+
'Recommended: create a service account at Google Cloud -> IAM & Admin -> Service Accounts, generate a JSON key, and grant it the Analytics Viewer role on the property. Store the JSON as a secret and reference it as serviceAccountJson: secret("GA4_SERVICE_ACCOUNT_JSON").',
|
|
71
|
+
"Alternative: provide an OAuth 2.0 refresh token with the analytics.readonly scope together with its clientId and clientSecret from the Google Cloud Console."
|
|
72
|
+
]
|
|
73
|
+
},
|
|
74
|
+
rateLimit: "GA4 Data API quota is 200,000 tokens/day per property (default); 429 responses are retried automatically with exponential backoff.",
|
|
75
|
+
limitations: [
|
|
76
|
+
"Incremental syncs use a 30-day window because GA4 can attribute conversions up to 3 days after the session.",
|
|
77
|
+
"Report pagination is 10,000 rows per page."
|
|
78
|
+
]
|
|
79
|
+
});
|
|
46
80
|
var ga4Credentials = {
|
|
47
81
|
serviceAccountJson: {
|
|
48
82
|
description: "Google service account JSON key (base64 or raw JSON)",
|
|
@@ -215,8 +249,20 @@ var INCREMENTAL_LOOKBACK_DAYS = 30;
|
|
|
215
249
|
function getDateRange(options, lookbackDays) {
|
|
216
250
|
const now = Date.now();
|
|
217
251
|
const endDate = toGA4Date(new Date(now));
|
|
218
|
-
|
|
219
|
-
|
|
252
|
+
if (options.mode === "latest" && options.since) {
|
|
253
|
+
const startMs2 = now - (INCREMENTAL_LOOKBACK_DAYS - 1) * MS_PER_DAY;
|
|
254
|
+
return { startDate: toGA4Date(new Date(startMs2)), endDate };
|
|
255
|
+
}
|
|
256
|
+
if (options.since) {
|
|
257
|
+
const sinceMs = new Date(options.since).getTime();
|
|
258
|
+
if (Number.isFinite(sinceMs)) {
|
|
259
|
+
const days = Math.max(1, Math.ceil((now - sinceMs) / MS_PER_DAY));
|
|
260
|
+
const cappedDays = Math.min(days, lookbackDays);
|
|
261
|
+
const startMs2 = now - (cappedDays - 1) * MS_PER_DAY;
|
|
262
|
+
return { startDate: toGA4Date(new Date(startMs2)), endDate };
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
const startMs = now - (lookbackDays - 1) * MS_PER_DAY;
|
|
220
266
|
return { startDate: toGA4Date(new Date(startMs)), endDate };
|
|
221
267
|
}
|
|
222
268
|
function rowToMetricSample(row, dimensionHeaders, metricHeaders, metricName) {
|
|
@@ -263,17 +309,105 @@ var tokenResponseSchema = z.object({
|
|
|
263
309
|
access_token: z.string().min(1),
|
|
264
310
|
expires_in: z.number().int().positive().optional()
|
|
265
311
|
});
|
|
312
|
+
var googleAnalyticsResources = defineResources({
|
|
313
|
+
ga4_traffic_by_day: {
|
|
314
|
+
shape: "metric",
|
|
315
|
+
description: "Daily site traffic totals - sessions, total users, new users, page views, and engagement rate.",
|
|
316
|
+
unit: "sessions",
|
|
317
|
+
granularity: "day",
|
|
318
|
+
endpoint: "POST /v1beta/properties/{propertyId}:runReport",
|
|
319
|
+
dimensions: [
|
|
320
|
+
{ name: "date", description: "Calendar day of the metric sample." }
|
|
321
|
+
],
|
|
322
|
+
responses: {
|
|
323
|
+
oauth_token: tokenResponseSchema,
|
|
324
|
+
traffic_by_day: reportSchema(1)
|
|
325
|
+
}
|
|
326
|
+
},
|
|
327
|
+
ga4_traffic_by_source: {
|
|
328
|
+
shape: "metric",
|
|
329
|
+
description: "Daily sessions and conversions broken down by acquisition source and medium.",
|
|
330
|
+
unit: "sessions",
|
|
331
|
+
granularity: "day",
|
|
332
|
+
endpoint: "POST /v1beta/properties/{propertyId}:runReport",
|
|
333
|
+
dimensions: [
|
|
334
|
+
{ name: "date", description: "Calendar day of the metric sample." },
|
|
335
|
+
{
|
|
336
|
+
name: "sessionSource",
|
|
337
|
+
description: "Origin of the session (e.g. google, direct, newsletter)."
|
|
338
|
+
},
|
|
339
|
+
{
|
|
340
|
+
name: "sessionMedium",
|
|
341
|
+
description: "Acquisition medium of the session (e.g. organic, cpc, referral)."
|
|
342
|
+
}
|
|
343
|
+
],
|
|
344
|
+
responses: { traffic_by_source: reportSchema(3) }
|
|
345
|
+
},
|
|
346
|
+
ga4_top_pages: {
|
|
347
|
+
shape: "metric",
|
|
348
|
+
description: "Daily page views and average session duration bucketed by page path.",
|
|
349
|
+
unit: "page_views",
|
|
350
|
+
granularity: "day",
|
|
351
|
+
endpoint: "POST /v1beta/properties/{propertyId}:runReport",
|
|
352
|
+
dimensions: [
|
|
353
|
+
{ name: "date", description: "Calendar day of the metric sample." },
|
|
354
|
+
{
|
|
355
|
+
name: "pagePath",
|
|
356
|
+
description: "URL path of the page that was viewed."
|
|
357
|
+
}
|
|
358
|
+
],
|
|
359
|
+
responses: { top_pages: reportSchema(2) }
|
|
360
|
+
},
|
|
361
|
+
ga4_events: {
|
|
362
|
+
shape: "metric",
|
|
363
|
+
description: "Daily event counts and the users that triggered them, bucketed by event name.",
|
|
364
|
+
unit: "events",
|
|
365
|
+
granularity: "day",
|
|
366
|
+
endpoint: "POST /v1beta/properties/{propertyId}:runReport",
|
|
367
|
+
dimensions: [
|
|
368
|
+
{ name: "date", description: "Calendar day of the metric sample." },
|
|
369
|
+
{
|
|
370
|
+
name: "eventName",
|
|
371
|
+
description: "GA4 event name (e.g. page_view, scroll, click)."
|
|
372
|
+
}
|
|
373
|
+
],
|
|
374
|
+
responses: { events: reportSchema(2) }
|
|
375
|
+
},
|
|
376
|
+
ga4_conversions: {
|
|
377
|
+
shape: "metric",
|
|
378
|
+
description: "Daily conversion counts and total revenue bucketed by conversion event name.",
|
|
379
|
+
unit: "conversions",
|
|
380
|
+
granularity: "day",
|
|
381
|
+
endpoint: "POST /v1beta/properties/{propertyId}:runReport",
|
|
382
|
+
dimensions: [
|
|
383
|
+
{ name: "date", description: "Calendar day of the metric sample." },
|
|
384
|
+
{
|
|
385
|
+
name: "eventName",
|
|
386
|
+
description: "GA4 conversion event name (e.g. purchase, generate_lead)."
|
|
387
|
+
}
|
|
388
|
+
],
|
|
389
|
+
responses: { conversions: reportSchema(2) }
|
|
390
|
+
},
|
|
391
|
+
ga4_geo: {
|
|
392
|
+
shape: "metric",
|
|
393
|
+
description: "Daily sessions and total users bucketed by visitor country.",
|
|
394
|
+
unit: "sessions",
|
|
395
|
+
granularity: "day",
|
|
396
|
+
endpoint: "POST /v1beta/properties/{propertyId}:runReport",
|
|
397
|
+
dimensions: [
|
|
398
|
+
{ name: "date", description: "Calendar day of the metric sample." },
|
|
399
|
+
{
|
|
400
|
+
name: "country",
|
|
401
|
+
description: "Country the session originated from."
|
|
402
|
+
}
|
|
403
|
+
],
|
|
404
|
+
responses: { geo: reportSchema(2) }
|
|
405
|
+
}
|
|
406
|
+
});
|
|
266
407
|
var GA4Connector = class _GA4Connector extends BaseConnector {
|
|
267
408
|
static id = "google-analytics";
|
|
268
|
-
static
|
|
269
|
-
|
|
270
|
-
traffic_by_day: reportSchema(1),
|
|
271
|
-
traffic_by_source: reportSchema(3),
|
|
272
|
-
top_pages: reportSchema(2),
|
|
273
|
-
events: reportSchema(2),
|
|
274
|
-
conversions: reportSchema(2),
|
|
275
|
-
geo: reportSchema(2)
|
|
276
|
-
};
|
|
409
|
+
static resources = googleAnalyticsResources;
|
|
410
|
+
static schemas = schemasFromResources(googleAnalyticsResources);
|
|
277
411
|
static create(input, ctx) {
|
|
278
412
|
const parsed = configFields.parse(input);
|
|
279
413
|
return new _GA4Connector(
|
|
@@ -351,7 +485,7 @@ var GA4Connector = class _GA4Connector extends BaseConnector {
|
|
|
351
485
|
headers: {
|
|
352
486
|
Authorization: `Bearer ${accessToken}`,
|
|
353
487
|
"Content-Type": "application/json",
|
|
354
|
-
"User-Agent": "
|
|
488
|
+
"User-Agent": connectorUserAgent("google-analytics")
|
|
355
489
|
},
|
|
356
490
|
body: JSON.stringify(body),
|
|
357
491
|
signal
|
|
@@ -433,6 +567,7 @@ export {
|
|
|
433
567
|
GA4Connector,
|
|
434
568
|
configFields,
|
|
435
569
|
index_default as default,
|
|
570
|
+
doc,
|
|
436
571
|
rowToMetricSample
|
|
437
572
|
};
|
|
438
573
|
//# sourceMappingURL=index.js.map
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/google-analytics.ts","../src/index.ts"],"sourcesContent":["import {\n BaseConnector,\n type ConnectorContext,\n type CredentialsSchema,\n type StorageHandle,\n type SyncOptions,\n type SyncResult,\n defineConfigFields,\n} from '@rawdash/core';\nimport { z } from 'zod';\n\nexport const configFields = defineConfigFields(\n z\n .object({\n propertyId: z\n .string()\n .trim()\n .regex(/^\\d+$/, 'GA4 Property ID must be digits only')\n .meta({\n label: 'GA4 Property ID',\n description:\n 'Numeric ID of your GA4 property (e.g. 123456789). Find it in Google Analytics → Admin → Property settings.',\n placeholder: '123456789',\n }),\n serviceAccountJson: z.object({ $secret: z.string() }).optional().meta({\n label: 'Service Account JSON (recommended)',\n description:\n 'Contents of the JSON key file for a Google service account with the Analytics Viewer role. Create one at Google Cloud → IAM & Admin → Service Accounts.',\n secret: true,\n }),\n refreshToken: z.object({ $secret: z.string() }).optional().meta({\n label: 'OAuth Refresh Token',\n description:\n 'Google OAuth 2.0 refresh token with analytics.readonly scope. Required if not using serviceAccountJson.',\n secret: true,\n }),\n clientId: z.string().optional().meta({\n label: 'OAuth Client ID',\n description:\n 'OAuth 2.0 client ID from Google Cloud Console. Required when using refreshToken auth.',\n placeholder: '…apps.googleusercontent.com',\n }),\n clientSecret: z.object({ $secret: z.string() }).optional().meta({\n label: 'OAuth Client Secret',\n description:\n 'OAuth 2.0 client secret from Google Cloud Console. Required when using refreshToken auth.',\n secret: true,\n }),\n lookbackDays: z.number().int().positive().optional().meta({\n label: 'Lookback days (full sync)',\n description:\n 'How many calendar days to fetch on a full sync. Defaults to 90.',\n placeholder: '90',\n }),\n })\n .refine(\n (val) =>\n val.serviceAccountJson !== undefined ||\n (val.refreshToken !== undefined &&\n val.clientId !== undefined &&\n val.clientSecret !== undefined),\n {\n message:\n 'Provide either serviceAccountJson or the full OAuth tuple (refreshToken + clientId + clientSecret)',\n },\n ),\n);\n\n// ---------------------------------------------------------------------------\n// Settings / credentials\n// ---------------------------------------------------------------------------\n\nexport interface GA4Settings {\n propertyId: string;\n lookbackDays?: number;\n}\n\nconst ga4Credentials = {\n serviceAccountJson: {\n description: 'Google service account JSON key (base64 or raw JSON)',\n auth: 'optional' as const,\n },\n refreshToken: {\n description: 'Google OAuth 2.0 refresh token',\n auth: 'optional' as const,\n },\n clientId: {\n description: 'Google OAuth 2.0 client ID',\n auth: 'optional' as const,\n },\n clientSecret: {\n description: 'Google OAuth 2.0 client secret',\n auth: 'optional' as const,\n },\n} satisfies CredentialsSchema;\n\ntype GA4Credentials = typeof ga4Credentials;\n\n// ---------------------------------------------------------------------------\n// Sync phases + cursor\n// ---------------------------------------------------------------------------\n\nconst PHASE_ORDER = [\n 'traffic_by_day',\n 'traffic_by_source',\n 'top_pages',\n 'events',\n 'conversions',\n 'geo',\n] as const;\n\ntype GA4Phase = (typeof PHASE_ORDER)[number];\n\ninterface GA4DateRange {\n startDate: string;\n endDate: string;\n}\n\ninterface GA4SyncCursor {\n phase: GA4Phase;\n // dateRange always populated, even when we abort between phases, so a\n // resumed run uses the original window for every remaining phase.\n dateRange: GA4DateRange;\n}\n\nconst GA4_DATE_RE = /^\\d{4}-\\d{2}-\\d{2}$/;\n\nfunction isGA4DateString(value: unknown): value is string {\n return typeof value === 'string' && GA4_DATE_RE.test(value);\n}\n\nfunction isGA4DateRange(value: unknown): value is GA4DateRange {\n if (typeof value !== 'object' || value === null) {\n return false;\n }\n const v = value as { startDate?: unknown; endDate?: unknown };\n return isGA4DateString(v.startDate) && isGA4DateString(v.endDate);\n}\n\nfunction isGA4SyncCursor(value: unknown): value is GA4SyncCursor {\n if (typeof value !== 'object' || value === null) {\n return false;\n }\n const v = value as { phase?: unknown; dateRange?: unknown };\n if (typeof v.phase !== 'string') {\n return false;\n }\n if (!(PHASE_ORDER as readonly string[]).includes(v.phase)) {\n return false;\n }\n return isGA4DateRange(v.dateRange);\n}\n\n// ---------------------------------------------------------------------------\n// Phase configs — dimensions + metrics for each resource\n// ---------------------------------------------------------------------------\n\ninterface PhaseConfig {\n dimensions: string[];\n metrics: string[];\n metricName: string;\n}\n\nconst PHASE_CONFIGS: Record<GA4Phase, PhaseConfig> = {\n traffic_by_day: {\n dimensions: ['date'],\n metrics: [\n 'sessions',\n 'totalUsers',\n 'newUsers',\n 'screenPageViews',\n 'engagementRate',\n ],\n metricName: 'ga4_traffic_by_day',\n },\n traffic_by_source: {\n dimensions: ['date', 'sessionSource', 'sessionMedium'],\n metrics: ['sessions', 'conversions'],\n metricName: 'ga4_traffic_by_source',\n },\n top_pages: {\n dimensions: ['date', 'pagePath'],\n metrics: ['screenPageViews', 'averageSessionDuration'],\n metricName: 'ga4_top_pages',\n },\n events: {\n dimensions: ['date', 'eventName'],\n metrics: ['eventCount', 'totalUsers'],\n metricName: 'ga4_events',\n },\n conversions: {\n dimensions: ['date', 'eventName'],\n metrics: ['conversions', 'totalRevenue'],\n metricName: 'ga4_conversions',\n },\n geo: {\n dimensions: ['date', 'country'],\n metrics: ['sessions', 'totalUsers'],\n metricName: 'ga4_geo',\n },\n};\n\nconst ROWS_PER_PAGE = 10_000;\n\n// ---------------------------------------------------------------------------\n// GA4 Data API types\n// ---------------------------------------------------------------------------\n\nexport interface GA4DimensionValue {\n value: string;\n}\n\nexport interface GA4MetricValue {\n value: string;\n}\n\nexport interface GA4ReportRow {\n dimensionValues: GA4DimensionValue[];\n metricValues: GA4MetricValue[];\n}\n\ninterface GA4ReportResponse {\n rows?: GA4ReportRow[];\n rowCount?: number;\n dimensionHeaders?: Array<{ name: string }>;\n metricHeaders?: Array<{ name: string; type: string }>;\n}\n\n// ---------------------------------------------------------------------------\n// Service account / OAuth token helpers\n// ---------------------------------------------------------------------------\n\ninterface ServiceAccountKey {\n client_email: string;\n private_key: string;\n token_uri?: string;\n}\n\ninterface TokenResponse {\n access_token: string;\n expires_in?: number;\n}\n\nfunction base64urlFromBytes(bytes: Uint8Array): string {\n let binary = '';\n for (let i = 0; i < bytes.length; i++) {\n binary += String.fromCharCode(bytes[i]!);\n }\n return btoa(binary).replace(/\\+/g, '-').replace(/\\//g, '_').replace(/=/g, '');\n}\n\nfunction base64urlFromString(str: string): string {\n return base64urlFromBytes(new TextEncoder().encode(str));\n}\n\nasync function signRS256JWT(\n payload: Record<string, unknown>,\n privateKeyPem: string,\n): Promise<string> {\n const header = { alg: 'RS256', typ: 'JWT' };\n const headerB64 = base64urlFromString(JSON.stringify(header));\n const payloadB64 = base64urlFromString(JSON.stringify(payload));\n const signingInput = `${headerB64}.${payloadB64}`;\n\n const pemContent = privateKeyPem\n .replace(/-----BEGIN PRIVATE KEY-----/g, '')\n .replace(/-----END PRIVATE KEY-----/g, '')\n .replace(/\\s/g, '');\n const der = Uint8Array.from(atob(pemContent), (c) => c.charCodeAt(0));\n\n const key = await globalThis.crypto.subtle.importKey(\n 'pkcs8',\n der,\n { name: 'RSASSA-PKCS1-v1_5', hash: 'SHA-256' },\n false,\n ['sign'],\n );\n\n const signature = await globalThis.crypto.subtle.sign(\n 'RSASSA-PKCS1-v1_5',\n key,\n new TextEncoder().encode(signingInput),\n );\n\n return `${signingInput}.${base64urlFromBytes(new Uint8Array(signature))}`;\n}\n\nfunction parseServiceAccountJson(value: string): ServiceAccountKey {\n const trimmed = value.trim();\n if (trimmed.startsWith('{')) {\n return JSON.parse(trimmed) as ServiceAccountKey;\n }\n const binary = atob(trimmed);\n const bytes = new Uint8Array(binary.length);\n for (let i = 0; i < binary.length; i++) {\n bytes[i] = binary.charCodeAt(i);\n }\n const decoded = new TextDecoder().decode(bytes);\n return JSON.parse(decoded) as ServiceAccountKey;\n}\n\nasync function buildServiceAccountJwt(\n serviceAccountJson: string,\n): Promise<{ url: string; body: string }> {\n const sa = parseServiceAccountJson(serviceAccountJson);\n const now = Math.floor(Date.now() / 1000);\n const jwt = await signRS256JWT(\n {\n iss: sa.client_email,\n scope: 'https://www.googleapis.com/auth/analytics.readonly',\n aud: sa.token_uri ?? 'https://oauth2.googleapis.com/token',\n exp: now + 3600,\n iat: now,\n },\n sa.private_key,\n );\n\n const body = new URLSearchParams({\n grant_type: 'urn:ietf:params:oauth:grant-type:jwt-bearer',\n assertion: jwt,\n }).toString();\n\n return {\n url: sa.token_uri ?? 'https://oauth2.googleapis.com/token',\n body,\n };\n}\n\n// ---------------------------------------------------------------------------\n// Date helpers\n// ---------------------------------------------------------------------------\n\nfunction toGA4Date(date: Date): string {\n const y = date.getUTCFullYear();\n const m = String(date.getUTCMonth() + 1).padStart(2, '0');\n const d = String(date.getUTCDate()).padStart(2, '0');\n return `${y}-${m}-${d}`;\n}\n\nfunction ga4DateToMs(ga4Date: string): number {\n // GA4 dates arrive as 'YYYYMMDD'\n const y = ga4Date.slice(0, 4);\n const m = ga4Date.slice(4, 6);\n const d = ga4Date.slice(6, 8);\n return Date.UTC(Number(y), Number(m) - 1, Number(d));\n}\n\nconst MS_PER_DAY = 24 * 60 * 60 * 1000;\nconst INCREMENTAL_LOOKBACK_DAYS = 30;\n\nfunction getDateRange(\n options: SyncOptions,\n lookbackDays: number,\n): GA4DateRange {\n const now = Date.now();\n const endDate = toGA4Date(new Date(now));\n const days =\n options.mode === 'latest' && options.since\n ? INCREMENTAL_LOOKBACK_DAYS\n : lookbackDays;\n const startMs = now - (days - 1) * MS_PER_DAY;\n return { startDate: toGA4Date(new Date(startMs)), endDate };\n}\n\n// ---------------------------------------------------------------------------\n// Row conversion\n// ---------------------------------------------------------------------------\n\nexport function rowToMetricSample(\n row: GA4ReportRow,\n dimensionHeaders: string[],\n metricHeaders: string[],\n metricName: string,\n): {\n name: string;\n ts: number;\n value: number;\n attributes: Record<string, string | number>;\n} {\n const dims: Record<string, string> = {};\n for (let i = 0; i < dimensionHeaders.length; i++) {\n dims[dimensionHeaders[i]!] = row.dimensionValues[i]?.value ?? '';\n }\n\n const mets: Record<string, number> = {};\n for (let i = 0; i < metricHeaders.length; i++) {\n mets[metricHeaders[i]!] =\n parseFloat(row.metricValues[i]?.value ?? '0') || 0;\n }\n\n const dateStr = dims['date'] ?? '19700101';\n const ts = ga4DateToMs(dateStr);\n const primaryValue = mets[metricHeaders[0]!] ?? 0;\n\n return {\n name: metricName,\n ts,\n value: primaryValue,\n attributes: { ...dims, ...mets },\n };\n}\n\n// ---------------------------------------------------------------------------\n// GA4Connector\n// ---------------------------------------------------------------------------\n\n// ---------------------------------------------------------------------------\n// Schemas — describe the per-resource API response shape consumed by request()\n// ---------------------------------------------------------------------------\n\nconst dateDimensionValue = z.object({\n value: z.string().regex(/^(19|20)\\d{2}(0[1-9]|1[0-2])(0[1-9]|[12]\\d|3[01])$/),\n});\n\nconst stringDimensionValue = z.object({ value: z.string() });\nconst numericMetricValue = z.object({\n value: z.string().regex(/^-?\\d+(\\.\\d+)?$/),\n});\n\nfunction reportSchema(dimensionCount: number) {\n const dims =\n dimensionCount === 1\n ? z.tuple([dateDimensionValue])\n : z.tuple([\n dateDimensionValue,\n ...Array(dimensionCount - 1).fill(stringDimensionValue),\n ] as [typeof dateDimensionValue, ...z.ZodType[]]);\n return z.object({\n rows: z\n .array(\n z.object({\n dimensionValues: dims,\n metricValues: z.array(numericMetricValue).nonempty(),\n }),\n )\n .optional(),\n });\n}\n\nconst tokenResponseSchema = z.object({\n access_token: z.string().min(1),\n expires_in: z.number().int().positive().optional(),\n});\n\nexport class GA4Connector extends BaseConnector<GA4Settings, GA4Credentials> {\n static readonly id = 'google-analytics';\n\n static readonly schemas = {\n oauth_token: tokenResponseSchema,\n traffic_by_day: reportSchema(1),\n traffic_by_source: reportSchema(3),\n top_pages: reportSchema(2),\n events: reportSchema(2),\n conversions: reportSchema(2),\n geo: reportSchema(2),\n } as const;\n\n static create(input: unknown, ctx?: ConnectorContext): GA4Connector {\n const parsed = configFields.parse(input);\n return new GA4Connector(\n {\n propertyId: parsed.propertyId,\n lookbackDays: parsed.lookbackDays,\n },\n {\n serviceAccountJson: parsed.serviceAccountJson,\n refreshToken: parsed.refreshToken,\n clientId: parsed.clientId,\n clientSecret: parsed.clientSecret,\n },\n ctx,\n );\n }\n\n readonly id = 'google-analytics';\n override readonly credentials = ga4Credentials;\n\n private cachedToken: { token: string; expiresAt: number } | null = null;\n\n private async fetchOAuthToken(\n url: string,\n body: string,\n signal: AbortSignal | undefined,\n ): Promise<{ token: string; expiresAt: number }> {\n const res = await this.post<TokenResponse>(url, {\n resource: 'oauth_token',\n headers: { 'Content-Type': 'application/x-www-form-urlencoded' },\n body,\n signal,\n });\n const expiresIn = res.body.expires_in ?? 3600;\n return {\n token: res.body.access_token,\n expiresAt: Date.now() + (expiresIn - 60) * 1000,\n };\n }\n\n private async getAccessToken(signal?: AbortSignal): Promise<string> {\n if (this.cachedToken && Date.now() < this.cachedToken.expiresAt) {\n return this.cachedToken.token;\n }\n\n const { serviceAccountJson, refreshToken, clientId, clientSecret } =\n this.creds;\n\n if (serviceAccountJson) {\n const { url, body } = await buildServiceAccountJwt(serviceAccountJson);\n this.cachedToken = await this.fetchOAuthToken(url, body, signal);\n return this.cachedToken.token;\n }\n\n if (refreshToken && clientId && clientSecret) {\n const body = new URLSearchParams({\n grant_type: 'refresh_token',\n refresh_token: refreshToken,\n client_id: clientId,\n client_secret: clientSecret,\n }).toString();\n this.cachedToken = await this.fetchOAuthToken(\n 'https://oauth2.googleapis.com/token',\n body,\n signal,\n );\n return this.cachedToken.token;\n }\n\n throw new Error(\n 'GA4 connector: provide either serviceAccountJson or (refreshToken + clientId + clientSecret)',\n );\n }\n\n private async runReport(\n accessToken: string,\n phase: GA4Phase,\n dateRange: { startDate: string; endDate: string },\n offset: number,\n signal?: AbortSignal,\n ): Promise<GA4ReportResponse> {\n const { dimensions, metrics } = PHASE_CONFIGS[phase];\n const url = `https://analyticsdata.googleapis.com/v1beta/properties/${this.settings.propertyId}:runReport`;\n\n const body: Record<string, unknown> = {\n dimensions: dimensions.map((name) => ({ name })),\n metrics: metrics.map((name) => ({ name })),\n dateRanges: [\n { startDate: dateRange.startDate, endDate: dateRange.endDate },\n ],\n limit: ROWS_PER_PAGE,\n offset,\n };\n\n const res = await this.post<GA4ReportResponse>(url, {\n resource: phase,\n headers: {\n Authorization: `Bearer ${accessToken}`,\n 'Content-Type': 'application/json',\n 'User-Agent':\n 'rawdash/connector-google-analytics (+https://rawdash.dev)',\n },\n body: JSON.stringify(body),\n signal,\n });\n return res.body;\n }\n\n async sync(\n options: SyncOptions,\n storage: StorageHandle,\n signal?: AbortSignal,\n ): Promise<SyncResult> {\n const lookbackDays = this.settings.lookbackDays ?? 90;\n\n const cursor = isGA4SyncCursor(options.cursor) ? options.cursor : undefined;\n // Restore the originally-computed window on resume so phases stay aligned\n // across midnight rollovers and lookbackDays changes between runs.\n const dateRange = cursor?.dateRange ?? getDateRange(options, lookbackDays);\n\n let accessToken: string | null = null;\n const getToken = async (sig?: AbortSignal): Promise<string> => {\n if (!accessToken) {\n accessToken = await this.getAccessToken(sig);\n }\n return accessToken;\n };\n\n const runReportWithRetry = async (\n phase: GA4Phase,\n offset: number,\n sig: AbortSignal | undefined,\n ): Promise<GA4ReportResponse> => {\n const token = await getToken(sig);\n try {\n return await this.runReport(token, phase, dateRange, offset, sig);\n } catch (err) {\n console.warn(\n `[ga4] runReport failed, refreshing token and retrying once`,\n err,\n );\n accessToken = null;\n const freshToken = await getToken(sig);\n return this.runReport(freshToken, phase, dateRange, offset, sig);\n }\n };\n\n const drainPhase = async (phase: GA4Phase): Promise<GA4ReportRow[]> => {\n const allRows: GA4ReportRow[] = [];\n let offset = 0;\n for (;;) {\n const response = await runReportWithRetry(phase, offset, signal);\n const rows = response.rows ?? [];\n allRows.push(...rows);\n offset += rows.length;\n if (rows.length === 0) {\n break;\n }\n // Prefer the API's authoritative rowCount when available; fall back\n // to a short-page heuristic only when GA4 omits it, so a missing\n // field can't truncate a multi-page dataset to its first page.\n const done =\n typeof response.rowCount === 'number'\n ? offset >= response.rowCount\n : rows.length < ROWS_PER_PAGE;\n if (done) {\n break;\n }\n }\n return allRows;\n };\n\n const resumeIdx = cursor ? PHASE_ORDER.indexOf(cursor.phase) : -1;\n const startIdx = resumeIdx >= 0 ? resumeIdx : 0;\n\n for (let i = startIdx; i < PHASE_ORDER.length; i++) {\n const phase = PHASE_ORDER[i]!;\n if (signal?.aborted) {\n return { done: false, cursor: { phase, dateRange } };\n }\n\n // Drain every page of this phase in-memory before writing so the commit\n // is one atomic call. A mid-phase failure restarts this phase from\n // scratch on the next sync; the clear-and-replace below wipes partial\n // state. If the abort signal trips mid-drain, surface a resumable\n // cursor instead of throwing the AbortError up to the caller.\n let rows: GA4ReportRow[];\n try {\n rows = await drainPhase(phase);\n } catch (err) {\n if (signal?.aborted) {\n return { done: false, cursor: { phase, dateRange } };\n }\n throw err;\n }\n const cfg = PHASE_CONFIGS[phase];\n const samples = rows.map((row) =>\n rowToMetricSample(row, cfg.dimensions, cfg.metrics, cfg.metricName),\n );\n // Scoping by name ensures stale rows are wiped even when samples is empty.\n await storage.metrics(samples, { names: [cfg.metricName] });\n }\n\n return { done: true };\n }\n}\n","import { GA4Connector } from './google-analytics';\n\nexport {\n configFields,\n GA4Connector,\n rowToMetricSample,\n} from './google-analytics';\nexport type { GA4Settings } from './google-analytics';\nexport default GA4Connector;\n"],"mappings":"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","names":[]}
|
|
1
|
+
{"version":3,"sources":["../../../connector-shared/src/errors.ts","../../../connector-shared/src/retry.ts","../../../connector-shared/src/version.ts","../../../connector-shared/src/request.ts","../../../connector-shared/src/rate-limit.ts","../../../connector-shared/src/sanitize.ts","../../../connector-shared/src/epoch.ts","../../../connector-shared/src/pagination.ts","../../../connector-shared/src/logger.ts","../src/google-analytics.ts","../src/index.ts"],"sourcesContent":["import type { HttpResponse } from './types';\n\nexport type HttpErrorKind =\n | 'transient'\n | 'rate_limit'\n | 'auth'\n | 'upstream_bug'\n | 'client_bug';\n\nexport abstract class HttpClientError extends Error {\n abstract readonly kind: HttpErrorKind;\n readonly response?: HttpResponse;\n\n constructor(message: string, response?: HttpResponse) {\n super(message);\n this.name = new.target.name;\n this.response = response;\n }\n}\n\nexport class TransientError extends HttpClientError {\n readonly kind = 'transient' as const;\n}\n\nexport class RateLimitError extends HttpClientError {\n readonly kind = 'rate_limit' as const;\n readonly retryAfter?: Date;\n\n constructor(message: string, response?: HttpResponse, retryAfter?: Date) {\n super(message, response);\n this.retryAfter = retryAfter;\n }\n}\n\nexport class AuthError extends HttpClientError {\n readonly kind = 'auth' as const;\n}\n\nexport class UpstreamBugError extends HttpClientError {\n readonly kind = 'upstream_bug' as const;\n}\n\nexport class ClientBugError extends HttpClientError {\n readonly kind = 'client_bug' as const;\n}\n\nexport function classifyStatus(status: number): HttpErrorKind {\n if (status === 429) {\n return 'rate_limit';\n }\n if (status === 401 || status === 403) {\n return 'auth';\n }\n if (status === 408) {\n return 'transient';\n }\n if (status >= 500) {\n return 'upstream_bug';\n }\n if (status >= 400) {\n return 'client_bug';\n }\n return 'client_bug';\n}\n\nexport function errorForStatus(\n message: string,\n response: HttpResponse,\n retryAfter?: Date,\n): HttpClientError {\n const kind = classifyStatus(response.status);\n switch (kind) {\n case 'rate_limit':\n return new RateLimitError(message, response, retryAfter);\n case 'auth':\n return new AuthError(message, response);\n case 'transient':\n return new TransientError(message, response);\n case 'upstream_bug':\n return new UpstreamBugError(message, response);\n case 'client_bug':\n return new ClientBugError(message, response);\n }\n}\n","import { HttpClientError, RateLimitError, TransientError } from './errors';\n\nexport interface RetryPolicy {\n maxAttempts?: number;\n initialDelayMs?: number;\n maxDelayMs?: number;\n retryOn?: (status: number | null, err?: Error) => boolean;\n}\n\nexport const defaultRetryOn = (status: number | null, err?: Error): boolean => {\n if (err instanceof RateLimitError) {\n return true;\n }\n if (err instanceof TransientError) {\n return true;\n }\n if (status === null) {\n return err instanceof Error && !(err instanceof HttpClientError);\n }\n if (status === 408 || status === 429) {\n return true;\n }\n if (status >= 500) {\n return true;\n }\n return false;\n};\n\nexport function backoffDelayMs(\n attempt: number,\n policy: Required<Pick<RetryPolicy, 'initialDelayMs' | 'maxDelayMs'>>,\n): number {\n const base = policy.initialDelayMs * 2 ** attempt;\n const jitter = base * 0.25 * Math.random();\n return Math.min(base + jitter, policy.maxDelayMs);\n}\n\nexport function parseRetryAfter(\n headerValue: string | null,\n now: Date = new Date(),\n): Date | undefined {\n if (!headerValue) {\n return undefined;\n }\n const trimmed = headerValue.trim();\n if (/^\\d+$/.test(trimmed)) {\n return new Date(now.getTime() + Number(trimmed) * 1000);\n }\n const parsed = Date.parse(trimmed);\n if (Number.isNaN(parsed)) {\n return undefined;\n }\n return new Date(parsed);\n}\n\nexport function sleep(ms: number, signal?: AbortSignal): Promise<void> {\n if (signal?.aborted) {\n return Promise.reject(signal.reason ?? new Error('Aborted'));\n }\n return new Promise<void>((resolve, reject) => {\n const onAbort = () => {\n clearTimeout(timer);\n reject(signal!.reason ?? new Error('Aborted'));\n };\n const timer = setTimeout(() => {\n signal?.removeEventListener('abort', onAbort);\n resolve();\n }, ms);\n signal?.addEventListener('abort', onAbort, { once: true });\n });\n}\n","export const HTTP_CLIENT_VERSION = '0.0.0';\n\nexport const DEFAULT_USER_AGENT = `rawdash-connector/${HTTP_CLIENT_VERSION} (+https://rawdash.dev)`;\n\nexport function connectorUserAgent(connectorId: string): string {\n return `rawdash-connector-${connectorId}/${HTTP_CLIENT_VERSION} (+https://rawdash.dev)`;\n}\n","import {\n AuthError,\n ClientBugError,\n HttpClientError,\n RateLimitError,\n TransientError,\n UpstreamBugError,\n errorForStatus,\n} from './errors';\nimport { defaultRetryOn, parseRetryAfter, sleep } from './retry';\nimport type { FetchLike, HttpMethod, HttpRequest, HttpResponse } from './types';\nimport { DEFAULT_USER_AGENT } from './version';\n\nconst DEFAULT_TIMEOUT_MS = 10_000;\nconst DEFAULT_MAX_ATTEMPTS = 3;\nconst DEFAULT_INITIAL_DELAY_MS = 1000;\nconst DEFAULT_MAX_DELAY_MS = 60_000;\nconst OBSERVER_TIMEOUT_MS = 250;\n\nexport interface RequestObservation {\n url: string;\n method: HttpMethod;\n status: number;\n resource: string;\n requestId: string;\n body: unknown;\n}\n\nexport type RequestObserver = (\n event: RequestObservation,\n) => void | Promise<void>;\n\nexport interface RequestOptions {\n fetch?: FetchLike;\n observer?: RequestObserver;\n resource: string;\n requestId?: string;\n}\n\nasync function notifyObserver(\n observer: RequestObserver,\n event: RequestObservation,\n): Promise<void> {\n let result: void | Promise<void>;\n try {\n result = observer(event);\n } catch (err) {\n console.warn('[connector-shared] request observer threw:', err);\n return;\n }\n if (!(result instanceof Promise)) {\n return;\n }\n const guarded = result.catch((err) => {\n console.warn('[connector-shared] request observer rejected:', err);\n });\n let timer: ReturnType<typeof setTimeout> | undefined;\n const timeout = new Promise<void>((resolve) => {\n timer = setTimeout(resolve, OBSERVER_TIMEOUT_MS);\n });\n try {\n await Promise.race([guarded, timeout]);\n } finally {\n if (timer) {\n clearTimeout(timer);\n }\n }\n}\n\nfunction newRequestId(): string {\n const c = (globalThis as { crypto?: { randomUUID?: () => string } }).crypto;\n if (c?.randomUUID) {\n return c.randomUUID();\n }\n return `${Date.now().toString(36)}-${Math.random().toString(36).slice(2, 10)}`;\n}\n\nfunction mergeHeaders(\n defaults: Record<string, string>,\n overrides: Record<string, string> | undefined,\n): Record<string, string> {\n const merged: Record<string, string> = {};\n for (const [k, v] of Object.entries(defaults)) {\n merged[k.toLowerCase()] = v;\n }\n if (overrides) {\n for (const [k, v] of Object.entries(overrides)) {\n merged[k.toLowerCase()] = v;\n }\n }\n return merged;\n}\n\nfunction linkTimeoutSignal(\n parent: AbortSignal | undefined,\n timeoutMs: number,\n): { signal: AbortSignal; cancel: () => void } {\n const controller = new AbortController();\n const onParentAbort = () => {\n controller.abort(parent?.reason);\n };\n if (parent) {\n if (parent.aborted) {\n controller.abort(parent.reason);\n } else {\n parent.addEventListener('abort', onParentAbort, { once: true });\n }\n }\n const timer = setTimeout(() => {\n controller.abort(new Error(`Request timed out after ${timeoutMs}ms`));\n }, timeoutMs);\n return {\n signal: controller.signal,\n cancel: () => {\n clearTimeout(timer);\n if (parent) {\n parent.removeEventListener('abort', onParentAbort);\n }\n },\n };\n}\n\nasync function readBody(res: Response, parseJson: boolean): Promise<unknown> {\n if (res.status === 204 || res.status === 205) {\n return null;\n }\n const contentType = res.headers.get('content-type') ?? '';\n if (parseJson && contentType.includes('application/json')) {\n const text = await res.text();\n if (text.length === 0) {\n return null;\n }\n return JSON.parse(text);\n }\n return res.text();\n}\n\nexport async function request<T = unknown>(\n req: HttpRequest,\n options: RequestOptions,\n): Promise<HttpResponse<T>> {\n const fetchImpl: FetchLike = options.fetch ?? (globalThis.fetch as FetchLike);\n const retry = req.retry ?? {};\n const maxAttempts = retry.maxAttempts ?? DEFAULT_MAX_ATTEMPTS;\n const initialDelayMs = retry.initialDelayMs ?? DEFAULT_INITIAL_DELAY_MS;\n const maxDelayMs = retry.maxDelayMs ?? DEFAULT_MAX_DELAY_MS;\n const retryOn = retry.retryOn ?? defaultRetryOn;\n const timeoutMs = req.timeoutMs ?? DEFAULT_TIMEOUT_MS;\n const parseJson = req.parseJson ?? true;\n\n const headers = mergeHeaders(\n {\n 'User-Agent': DEFAULT_USER_AGENT,\n Accept: 'application/json',\n },\n req.headers,\n );\n\n let lastErr: Error | undefined;\n\n for (let attempt = 0; attempt < maxAttempts; attempt++) {\n req.signal?.throwIfAborted();\n\n const { signal, cancel } = linkTimeoutSignal(req.signal, timeoutMs);\n let res: Response;\n try {\n res = await fetchImpl(req.url, {\n method: req.method ?? 'GET',\n headers,\n body: req.body as RequestInit['body'],\n signal,\n });\n } catch (err) {\n cancel();\n if (req.signal?.aborted) {\n throw req.signal.reason ?? err;\n }\n const error = err instanceof Error ? err : new Error(String(err));\n lastErr = error;\n if (attempt < maxAttempts - 1 && retryOn(null, error)) {\n const delay = computeDelay(attempt, initialDelayMs, maxDelayMs);\n await sleep(delay, req.signal);\n continue;\n }\n throw new TransientError(error.message);\n }\n cancel();\n\n const body = await readBody(res, parseJson);\n const httpResponse: HttpResponse<T> = {\n status: res.status,\n headers: res.headers,\n body: body as T,\n };\n if (req.rateLimit) {\n const state = req.rateLimit.parse(res.headers);\n if (state) {\n httpResponse.rateLimitState = state;\n }\n }\n\n if (options.observer) {\n await notifyObserver(options.observer, {\n url: req.url,\n method: req.method ?? 'GET',\n status: res.status,\n resource: options.resource,\n requestId: options.requestId ?? newRequestId(),\n body,\n });\n }\n\n if (res.ok) {\n return httpResponse;\n }\n\n const retryAfter = parseRetryAfter(res.headers.get('retry-after'));\n const message = `HTTP ${res.status} ${res.statusText} for ${req.method ?? 'GET'} ${req.url}`;\n const err = errorForStatus(message, httpResponse, retryAfter);\n\n if (\n attempt < maxAttempts - 1 &&\n retryOn(res.status, err) &&\n !(err instanceof AuthError) &&\n !(err instanceof ClientBugError)\n ) {\n lastErr = err;\n let delay = computeDelay(attempt, initialDelayMs, maxDelayMs);\n if (err instanceof RateLimitError && retryAfter) {\n const wait = retryAfter.getTime() - Date.now();\n if (wait > 0) {\n delay = Math.min(wait, maxDelayMs);\n }\n }\n await sleep(delay, req.signal);\n continue;\n }\n\n throw err;\n }\n\n throw lastErr ?? new UpstreamBugError('Exhausted retry attempts');\n}\n\nfunction computeDelay(\n attempt: number,\n initialDelayMs: number,\n maxDelayMs: number,\n): number {\n const base = initialDelayMs * 2 ** attempt;\n const jitter = base * 0.25 * Math.random();\n return Math.min(base + jitter, maxDelayMs);\n}\n\nexport { HttpClientError };\n","export interface RateLimitState {\n remaining: number;\n resetAt: Date;\n}\n\nexport interface RateLimitPolicy {\n parse(headers: Headers): RateLimitState | null;\n}\n\nexport interface StandardRateLimitPolicyConfig {\n remainingHeader: string;\n resetHeader: string;\n resetUnit: 's' | 'ms';\n resetFallbackMs?: number;\n}\n\nexport function standardRateLimitPolicy(\n config: StandardRateLimitPolicyConfig,\n): RateLimitPolicy {\n const { remainingHeader, resetHeader, resetUnit, resetFallbackMs } = config;\n const multiplier = resetUnit === 's' ? 1000 : 1;\n return {\n parse(h) {\n const remainingRaw = h.get(remainingHeader);\n if (remainingRaw === null || remainingRaw.trim() === '') {\n return null;\n }\n const remaining = Number(remainingRaw);\n if (!Number.isFinite(remaining)) {\n return null;\n }\n const resetRaw = h.get(resetHeader);\n if (resetRaw === null) {\n if (resetFallbackMs === undefined) {\n return null;\n }\n return {\n remaining,\n resetAt: new Date(Date.now() + resetFallbackMs),\n };\n }\n if (resetRaw.trim() === '') {\n return null;\n }\n const reset = Number(resetRaw);\n if (!Number.isFinite(reset) || reset < 0) {\n return null;\n }\n const resetMs = reset * multiplier;\n if (!Number.isFinite(resetMs)) {\n return null;\n }\n return { remaining, resetAt: new Date(resetMs) };\n },\n };\n}\n","export interface SanitizeAllowedUrlOptions {\n url: string | null;\n host: string;\n pathname: string;\n protocol?: 'https:' | 'http:';\n}\n\nexport function sanitizeAllowedUrl(\n options: SanitizeAllowedUrlOptions,\n): string | null {\n const { url, host, pathname, protocol = 'https:' } = options;\n if (url === null) {\n return null;\n }\n try {\n const u = new URL(url);\n if (u.protocol !== protocol || u.host !== host || u.pathname !== pathname) {\n return null;\n }\n return u.toString();\n } catch {\n return null;\n }\n}\n","export type EpochUnit = 'ms' | 's' | 'iso';\n\nexport function parseEpoch(\n value: number | string | null | undefined,\n unit: EpochUnit,\n): number | null {\n if (value === null || value === undefined) {\n return null;\n }\n if (unit === 'iso') {\n if (typeof value !== 'string') {\n return null;\n }\n const ms = new Date(value).getTime();\n return Number.isFinite(ms) ? ms : null;\n }\n if (typeof value === 'string' && value.trim() === '') {\n return null;\n }\n const n = typeof value === 'number' ? value : Number(value);\n if (!Number.isFinite(n)) {\n return null;\n }\n const result = unit === 's' ? n * 1000 : n;\n return Number.isFinite(result) ? result : null;\n}\n","import { request } from './request';\nimport type { HttpRequest } from './types';\n\nexport function parseLinkHeader(header: string | null): Record<string, string> {\n if (!header) {\n return {};\n }\n const result: Record<string, string> = {};\n for (const part of header.split(',')) {\n const match = part.match(/<([^>]+)>\\s*;\\s*rel=\"([^\"]+)\"/);\n if (match) {\n result[match[2]!] = match[1]!;\n }\n }\n return result;\n}\n\nexport async function* paginateLink<T>(\n initial: HttpRequest,\n parse: (body: unknown) => T[],\n options: { resource: string },\n): AsyncIterable<T> {\n let next: string | null = initial.url;\n while (next) {\n const res: Awaited<ReturnType<typeof request>> = await request(\n {\n ...initial,\n url: next,\n },\n { resource: options.resource },\n );\n for (const item of parse(res.body)) {\n yield item;\n }\n const links = parseLinkHeader(res.headers.get('link'));\n next = links['next'] ?? null;\n }\n}\n\nexport async function* paginateCursor<T>(\n initial: HttpRequest,\n parse: (body: unknown) => { items: T[]; nextCursor: string | null },\n buildNext: (req: HttpRequest, cursor: string) => HttpRequest,\n options: { resource: string },\n): AsyncIterable<T> {\n let req: HttpRequest = initial;\n while (true) {\n const res = await request(req, { resource: options.resource });\n const { items, nextCursor } = parse(res.body);\n for (const item of items) {\n yield item;\n }\n if (!nextCursor) {\n return;\n }\n req = buildNext(req, nextCursor);\n }\n}\n\nexport async function* paginatePage<T>(\n initial: HttpRequest,\n parse: (body: unknown) => { items: T[]; hasMore: boolean },\n buildPage: (req: HttpRequest, page: number) => HttpRequest,\n options: { resource: string },\n): AsyncIterable<T> {\n let page = 1;\n while (true) {\n const req = page === 1 ? initial : buildPage(initial, page);\n const res = await request(req, { resource: options.resource });\n const { items, hasMore } = parse(res.body);\n for (const item of items) {\n yield item;\n }\n if (!hasMore || items.length === 0) {\n return;\n }\n page++;\n }\n}\n","export type LogFields = Record<string, unknown>;\n\nexport interface ConnectorLogger {\n info(event: string, fields?: LogFields): void;\n warn(event: string, fields?: LogFields): void;\n}\n\nexport interface ConnectorLoggerOptions {\n scope: string;\n}\n\nconst MAX_VALUE_LEN = 120;\n\nfunction truncate(s: string, max = MAX_VALUE_LEN): string {\n if (s.length <= max) {\n return s;\n }\n return `${s.slice(0, max - 1)}…`;\n}\n\nfunction formatValue(value: unknown): string {\n if (value === null) {\n return 'null';\n }\n if (value === undefined) {\n return '';\n }\n if (typeof value === 'number' || typeof value === 'boolean') {\n return String(value);\n }\n if (typeof value === 'string') {\n const t = truncate(value);\n if (/[\\s\"=]/.test(t)) {\n return JSON.stringify(t);\n }\n return t;\n }\n if (typeof value === 'bigint') {\n return value.toString();\n }\n let json: string | undefined;\n try {\n json = JSON.stringify(value);\n } catch {\n json = undefined;\n }\n return truncate(json ?? String(value));\n}\n\nexport function formatLogFields(fields?: LogFields): string {\n if (!fields) {\n return '';\n }\n const parts: string[] = [];\n for (const [k, v] of Object.entries(fields)) {\n if (v === undefined) {\n continue;\n }\n parts.push(`${k}=${formatValue(v)}`);\n }\n return parts.length > 0 ? ` ${parts.join(' ')}` : '';\n}\n\nexport function formatLogLine(\n scope: string,\n event: string,\n fields?: LogFields,\n): string {\n return `[${scope}] ${event}${formatLogFields(fields)}`;\n}\n\nexport function createDefaultConnectorLogger(\n opts: ConnectorLoggerOptions,\n): ConnectorLogger {\n return {\n info(event, fields) {\n console.info(formatLogLine(opts.scope, event, fields));\n },\n warn(event, fields) {\n console.warn(formatLogLine(opts.scope, event, fields));\n },\n };\n}\n\nconst NOOP_LOGGER: ConnectorLogger = {\n info() {},\n warn() {},\n};\n\nexport function noopConnectorLogger(): ConnectorLogger {\n return NOOP_LOGGER;\n}\n","import { connectorUserAgent } from '@rawdash/connector-shared';\nimport {\n BaseConnector,\n type ConnectorContext,\n type ConnectorDoc,\n type CredentialsSchema,\n type StorageHandle,\n type SyncOptions,\n type SyncResult,\n defineConfigFields,\n defineConnectorDoc,\n defineResources,\n schemasFromResources,\n} from '@rawdash/core';\nimport { z } from 'zod';\n\nexport const configFields = defineConfigFields(\n z\n .object({\n propertyId: z\n .string()\n .trim()\n .regex(/^\\d+$/, 'GA4 Property ID must be digits only')\n .meta({\n label: 'GA4 Property ID',\n description:\n 'Numeric ID of your GA4 property (e.g. 123456789). Find it in Google Analytics → Admin → Property settings.',\n placeholder: '123456789',\n }),\n serviceAccountJson: z.object({ $secret: z.string() }).optional().meta({\n label: 'Service Account JSON (recommended)',\n description:\n 'Contents of the JSON key file for a Google service account with the Analytics Viewer role. Create one at Google Cloud → IAM & Admin → Service Accounts.',\n secret: true,\n }),\n refreshToken: z.object({ $secret: z.string() }).optional().meta({\n label: 'OAuth Refresh Token',\n description:\n 'Google OAuth 2.0 refresh token with analytics.readonly scope. Required if not using serviceAccountJson.',\n secret: true,\n }),\n clientId: z.string().optional().meta({\n label: 'OAuth Client ID',\n description:\n 'OAuth 2.0 client ID from Google Cloud Console. Required when using refreshToken auth.',\n placeholder: '…apps.googleusercontent.com',\n }),\n clientSecret: z.object({ $secret: z.string() }).optional().meta({\n label: 'OAuth Client Secret',\n description:\n 'OAuth 2.0 client secret from Google Cloud Console. Required when using refreshToken auth.',\n secret: true,\n }),\n lookbackDays: z.number().int().positive().optional().meta({\n label: 'Lookback days (full sync)',\n description:\n 'How many calendar days to fetch on a full sync. Defaults to 90.',\n placeholder: '90',\n }),\n })\n .refine(\n (val) =>\n val.serviceAccountJson !== undefined ||\n (val.refreshToken !== undefined &&\n val.clientId !== undefined &&\n val.clientSecret !== undefined),\n {\n message:\n 'Provide either serviceAccountJson or the full OAuth tuple (refreshToken + clientId + clientSecret)',\n },\n ),\n);\n\nexport const doc: ConnectorDoc = defineConnectorDoc({\n displayName: 'Google Analytics',\n category: 'analytics',\n brandColor: '#E37400',\n tagline:\n 'Sync daily GA4 traffic, acquisition, top pages, events, conversions, and geography metrics from a Google Analytics 4 property.',\n vendor: {\n name: 'Google Analytics',\n apiDocs:\n 'https://developers.google.com/analytics/devguides/reporting/data/v1',\n website: 'https://analytics.google.com',\n },\n auth: {\n summary:\n 'Authenticate against the GA4 Data API with either a Google service account JSON key (recommended) or an OAuth 2.0 refresh-token tuple. The identity must have at least the Analytics Viewer role on the property.',\n setup: [\n 'Find your GA4 Property ID under Google Analytics -> Admin -> Property settings (numeric, e.g. 123456789).',\n 'Recommended: create a service account at Google Cloud -> IAM & Admin -> Service Accounts, generate a JSON key, and grant it the Analytics Viewer role on the property. Store the JSON as a secret and reference it as serviceAccountJson: secret(\"GA4_SERVICE_ACCOUNT_JSON\").',\n 'Alternative: provide an OAuth 2.0 refresh token with the analytics.readonly scope together with its clientId and clientSecret from the Google Cloud Console.',\n ],\n },\n rateLimit:\n 'GA4 Data API quota is 200,000 tokens/day per property (default); 429 responses are retried automatically with exponential backoff.',\n limitations: [\n 'Incremental syncs use a 30-day window because GA4 can attribute conversions up to 3 days after the session.',\n 'Report pagination is 10,000 rows per page.',\n ],\n});\n\n// ---------------------------------------------------------------------------\n// Settings / credentials\n// ---------------------------------------------------------------------------\n\nexport interface GA4Settings {\n propertyId: string;\n lookbackDays?: number;\n}\n\nconst ga4Credentials = {\n serviceAccountJson: {\n description: 'Google service account JSON key (base64 or raw JSON)',\n auth: 'optional' as const,\n },\n refreshToken: {\n description: 'Google OAuth 2.0 refresh token',\n auth: 'optional' as const,\n },\n clientId: {\n description: 'Google OAuth 2.0 client ID',\n auth: 'optional' as const,\n },\n clientSecret: {\n description: 'Google OAuth 2.0 client secret',\n auth: 'optional' as const,\n },\n} satisfies CredentialsSchema;\n\ntype GA4Credentials = typeof ga4Credentials;\n\n// ---------------------------------------------------------------------------\n// Sync phases + cursor\n// ---------------------------------------------------------------------------\n\nconst PHASE_ORDER = [\n 'traffic_by_day',\n 'traffic_by_source',\n 'top_pages',\n 'events',\n 'conversions',\n 'geo',\n] as const;\n\ntype GA4Phase = (typeof PHASE_ORDER)[number];\n\ninterface GA4DateRange {\n startDate: string;\n endDate: string;\n}\n\ninterface GA4SyncCursor {\n phase: GA4Phase;\n // dateRange always populated, even when we abort between phases, so a\n // resumed run uses the original window for every remaining phase.\n dateRange: GA4DateRange;\n}\n\nconst GA4_DATE_RE = /^\\d{4}-\\d{2}-\\d{2}$/;\n\nfunction isGA4DateString(value: unknown): value is string {\n return typeof value === 'string' && GA4_DATE_RE.test(value);\n}\n\nfunction isGA4DateRange(value: unknown): value is GA4DateRange {\n if (typeof value !== 'object' || value === null) {\n return false;\n }\n const v = value as { startDate?: unknown; endDate?: unknown };\n return isGA4DateString(v.startDate) && isGA4DateString(v.endDate);\n}\n\nfunction isGA4SyncCursor(value: unknown): value is GA4SyncCursor {\n if (typeof value !== 'object' || value === null) {\n return false;\n }\n const v = value as { phase?: unknown; dateRange?: unknown };\n if (typeof v.phase !== 'string') {\n return false;\n }\n if (!(PHASE_ORDER as readonly string[]).includes(v.phase)) {\n return false;\n }\n return isGA4DateRange(v.dateRange);\n}\n\n// ---------------------------------------------------------------------------\n// Phase configs — dimensions + metrics for each resource\n// ---------------------------------------------------------------------------\n\ninterface PhaseConfig {\n dimensions: string[];\n metrics: string[];\n metricName: string;\n}\n\nconst PHASE_CONFIGS: Record<GA4Phase, PhaseConfig> = {\n traffic_by_day: {\n dimensions: ['date'],\n metrics: [\n 'sessions',\n 'totalUsers',\n 'newUsers',\n 'screenPageViews',\n 'engagementRate',\n ],\n metricName: 'ga4_traffic_by_day',\n },\n traffic_by_source: {\n dimensions: ['date', 'sessionSource', 'sessionMedium'],\n metrics: ['sessions', 'conversions'],\n metricName: 'ga4_traffic_by_source',\n },\n top_pages: {\n dimensions: ['date', 'pagePath'],\n metrics: ['screenPageViews', 'averageSessionDuration'],\n metricName: 'ga4_top_pages',\n },\n events: {\n dimensions: ['date', 'eventName'],\n metrics: ['eventCount', 'totalUsers'],\n metricName: 'ga4_events',\n },\n conversions: {\n dimensions: ['date', 'eventName'],\n metrics: ['conversions', 'totalRevenue'],\n metricName: 'ga4_conversions',\n },\n geo: {\n dimensions: ['date', 'country'],\n metrics: ['sessions', 'totalUsers'],\n metricName: 'ga4_geo',\n },\n};\n\nconst ROWS_PER_PAGE = 10_000;\n\n// ---------------------------------------------------------------------------\n// GA4 Data API types\n// ---------------------------------------------------------------------------\n\nexport interface GA4DimensionValue {\n value: string;\n}\n\nexport interface GA4MetricValue {\n value: string;\n}\n\nexport interface GA4ReportRow {\n dimensionValues: GA4DimensionValue[];\n metricValues: GA4MetricValue[];\n}\n\ninterface GA4ReportResponse {\n rows?: GA4ReportRow[];\n rowCount?: number;\n dimensionHeaders?: Array<{ name: string }>;\n metricHeaders?: Array<{ name: string; type: string }>;\n}\n\n// ---------------------------------------------------------------------------\n// Service account / OAuth token helpers\n// ---------------------------------------------------------------------------\n\ninterface ServiceAccountKey {\n client_email: string;\n private_key: string;\n token_uri?: string;\n}\n\ninterface TokenResponse {\n access_token: string;\n expires_in?: number;\n}\n\nfunction base64urlFromBytes(bytes: Uint8Array): string {\n let binary = '';\n for (let i = 0; i < bytes.length; i++) {\n binary += String.fromCharCode(bytes[i]!);\n }\n return btoa(binary).replace(/\\+/g, '-').replace(/\\//g, '_').replace(/=/g, '');\n}\n\nfunction base64urlFromString(str: string): string {\n return base64urlFromBytes(new TextEncoder().encode(str));\n}\n\nasync function signRS256JWT(\n payload: Record<string, unknown>,\n privateKeyPem: string,\n): Promise<string> {\n const header = { alg: 'RS256', typ: 'JWT' };\n const headerB64 = base64urlFromString(JSON.stringify(header));\n const payloadB64 = base64urlFromString(JSON.stringify(payload));\n const signingInput = `${headerB64}.${payloadB64}`;\n\n const pemContent = privateKeyPem\n .replace(/-----BEGIN PRIVATE KEY-----/g, '')\n .replace(/-----END PRIVATE KEY-----/g, '')\n .replace(/\\s/g, '');\n const der = Uint8Array.from(atob(pemContent), (c) => c.charCodeAt(0));\n\n const key = await globalThis.crypto.subtle.importKey(\n 'pkcs8',\n der,\n { name: 'RSASSA-PKCS1-v1_5', hash: 'SHA-256' },\n false,\n ['sign'],\n );\n\n const signature = await globalThis.crypto.subtle.sign(\n 'RSASSA-PKCS1-v1_5',\n key,\n new TextEncoder().encode(signingInput),\n );\n\n return `${signingInput}.${base64urlFromBytes(new Uint8Array(signature))}`;\n}\n\nfunction parseServiceAccountJson(value: string): ServiceAccountKey {\n const trimmed = value.trim();\n if (trimmed.startsWith('{')) {\n return JSON.parse(trimmed) as ServiceAccountKey;\n }\n const binary = atob(trimmed);\n const bytes = new Uint8Array(binary.length);\n for (let i = 0; i < binary.length; i++) {\n bytes[i] = binary.charCodeAt(i);\n }\n const decoded = new TextDecoder().decode(bytes);\n return JSON.parse(decoded) as ServiceAccountKey;\n}\n\nasync function buildServiceAccountJwt(\n serviceAccountJson: string,\n): Promise<{ url: string; body: string }> {\n const sa = parseServiceAccountJson(serviceAccountJson);\n const now = Math.floor(Date.now() / 1000);\n const jwt = await signRS256JWT(\n {\n iss: sa.client_email,\n scope: 'https://www.googleapis.com/auth/analytics.readonly',\n aud: sa.token_uri ?? 'https://oauth2.googleapis.com/token',\n exp: now + 3600,\n iat: now,\n },\n sa.private_key,\n );\n\n const body = new URLSearchParams({\n grant_type: 'urn:ietf:params:oauth:grant-type:jwt-bearer',\n assertion: jwt,\n }).toString();\n\n return {\n url: sa.token_uri ?? 'https://oauth2.googleapis.com/token',\n body,\n };\n}\n\n// ---------------------------------------------------------------------------\n// Date helpers\n// ---------------------------------------------------------------------------\n\nfunction toGA4Date(date: Date): string {\n const y = date.getUTCFullYear();\n const m = String(date.getUTCMonth() + 1).padStart(2, '0');\n const d = String(date.getUTCDate()).padStart(2, '0');\n return `${y}-${m}-${d}`;\n}\n\nfunction ga4DateToMs(ga4Date: string): number {\n // GA4 dates arrive as 'YYYYMMDD'\n const y = ga4Date.slice(0, 4);\n const m = ga4Date.slice(4, 6);\n const d = ga4Date.slice(6, 8);\n return Date.UTC(Number(y), Number(m) - 1, Number(d));\n}\n\nconst MS_PER_DAY = 24 * 60 * 60 * 1000;\nconst INCREMENTAL_LOOKBACK_DAYS = 30;\n\nfunction getDateRange(\n options: SyncOptions,\n lookbackDays: number,\n): GA4DateRange {\n const now = Date.now();\n const endDate = toGA4Date(new Date(now));\n if (options.mode === 'latest' && options.since) {\n const startMs = now - (INCREMENTAL_LOOKBACK_DAYS - 1) * MS_PER_DAY;\n return { startDate: toGA4Date(new Date(startMs)), endDate };\n }\n if (options.since) {\n const sinceMs = new Date(options.since).getTime();\n if (Number.isFinite(sinceMs)) {\n const days = Math.max(1, Math.ceil((now - sinceMs) / MS_PER_DAY));\n const cappedDays = Math.min(days, lookbackDays);\n const startMs = now - (cappedDays - 1) * MS_PER_DAY;\n return { startDate: toGA4Date(new Date(startMs)), endDate };\n }\n }\n const startMs = now - (lookbackDays - 1) * MS_PER_DAY;\n return { startDate: toGA4Date(new Date(startMs)), endDate };\n}\n\n// ---------------------------------------------------------------------------\n// Row conversion\n// ---------------------------------------------------------------------------\n\nexport function rowToMetricSample(\n row: GA4ReportRow,\n dimensionHeaders: string[],\n metricHeaders: string[],\n metricName: string,\n): {\n name: string;\n ts: number;\n value: number;\n attributes: Record<string, string | number>;\n} {\n const dims: Record<string, string> = {};\n for (let i = 0; i < dimensionHeaders.length; i++) {\n dims[dimensionHeaders[i]!] = row.dimensionValues[i]?.value ?? '';\n }\n\n const mets: Record<string, number> = {};\n for (let i = 0; i < metricHeaders.length; i++) {\n mets[metricHeaders[i]!] =\n parseFloat(row.metricValues[i]?.value ?? '0') || 0;\n }\n\n const dateStr = dims['date'] ?? '19700101';\n const ts = ga4DateToMs(dateStr);\n const primaryValue = mets[metricHeaders[0]!] ?? 0;\n\n return {\n name: metricName,\n ts,\n value: primaryValue,\n attributes: { ...dims, ...mets },\n };\n}\n\n// ---------------------------------------------------------------------------\n// GA4Connector\n// ---------------------------------------------------------------------------\n\n// ---------------------------------------------------------------------------\n// Schemas — describe the per-resource API response shape consumed by request()\n// ---------------------------------------------------------------------------\n\nconst dateDimensionValue = z.object({\n value: z.string().regex(/^(19|20)\\d{2}(0[1-9]|1[0-2])(0[1-9]|[12]\\d|3[01])$/),\n});\n\nconst stringDimensionValue = z.object({ value: z.string() });\nconst numericMetricValue = z.object({\n value: z.string().regex(/^-?\\d+(\\.\\d+)?$/),\n});\n\nfunction reportSchema(dimensionCount: number) {\n const dims =\n dimensionCount === 1\n ? z.tuple([dateDimensionValue])\n : z.tuple([\n dateDimensionValue,\n ...Array(dimensionCount - 1).fill(stringDimensionValue),\n ] as [typeof dateDimensionValue, ...z.ZodType[]]);\n return z.object({\n rows: z\n .array(\n z.object({\n dimensionValues: dims,\n metricValues: z.array(numericMetricValue).nonempty(),\n }),\n )\n .optional(),\n });\n}\n\nconst tokenResponseSchema = z.object({\n access_token: z.string().min(1),\n expires_in: z.number().int().positive().optional(),\n});\n\nconst googleAnalyticsResources = defineResources({\n ga4_traffic_by_day: {\n shape: 'metric',\n description:\n 'Daily site traffic totals - sessions, total users, new users, page views, and engagement rate.',\n unit: 'sessions',\n granularity: 'day',\n endpoint: 'POST /v1beta/properties/{propertyId}:runReport',\n dimensions: [\n { name: 'date', description: 'Calendar day of the metric sample.' },\n ],\n responses: {\n oauth_token: tokenResponseSchema,\n traffic_by_day: reportSchema(1),\n },\n },\n ga4_traffic_by_source: {\n shape: 'metric',\n description:\n 'Daily sessions and conversions broken down by acquisition source and medium.',\n unit: 'sessions',\n granularity: 'day',\n endpoint: 'POST /v1beta/properties/{propertyId}:runReport',\n dimensions: [\n { name: 'date', description: 'Calendar day of the metric sample.' },\n {\n name: 'sessionSource',\n description: 'Origin of the session (e.g. google, direct, newsletter).',\n },\n {\n name: 'sessionMedium',\n description:\n 'Acquisition medium of the session (e.g. organic, cpc, referral).',\n },\n ],\n responses: { traffic_by_source: reportSchema(3) },\n },\n ga4_top_pages: {\n shape: 'metric',\n description:\n 'Daily page views and average session duration bucketed by page path.',\n unit: 'page_views',\n granularity: 'day',\n endpoint: 'POST /v1beta/properties/{propertyId}:runReport',\n dimensions: [\n { name: 'date', description: 'Calendar day of the metric sample.' },\n {\n name: 'pagePath',\n description: 'URL path of the page that was viewed.',\n },\n ],\n responses: { top_pages: reportSchema(2) },\n },\n ga4_events: {\n shape: 'metric',\n description:\n 'Daily event counts and the users that triggered them, bucketed by event name.',\n unit: 'events',\n granularity: 'day',\n endpoint: 'POST /v1beta/properties/{propertyId}:runReport',\n dimensions: [\n { name: 'date', description: 'Calendar day of the metric sample.' },\n {\n name: 'eventName',\n description: 'GA4 event name (e.g. page_view, scroll, click).',\n },\n ],\n responses: { events: reportSchema(2) },\n },\n ga4_conversions: {\n shape: 'metric',\n description:\n 'Daily conversion counts and total revenue bucketed by conversion event name.',\n unit: 'conversions',\n granularity: 'day',\n endpoint: 'POST /v1beta/properties/{propertyId}:runReport',\n dimensions: [\n { name: 'date', description: 'Calendar day of the metric sample.' },\n {\n name: 'eventName',\n description:\n 'GA4 conversion event name (e.g. purchase, generate_lead).',\n },\n ],\n responses: { conversions: reportSchema(2) },\n },\n ga4_geo: {\n shape: 'metric',\n description: 'Daily sessions and total users bucketed by visitor country.',\n unit: 'sessions',\n granularity: 'day',\n endpoint: 'POST /v1beta/properties/{propertyId}:runReport',\n dimensions: [\n { name: 'date', description: 'Calendar day of the metric sample.' },\n {\n name: 'country',\n description: 'Country the session originated from.',\n },\n ],\n responses: { geo: reportSchema(2) },\n },\n});\n\nexport class GA4Connector extends BaseConnector<GA4Settings, GA4Credentials> {\n static readonly id = 'google-analytics';\n\n static readonly resources = googleAnalyticsResources;\n\n static readonly schemas = schemasFromResources(googleAnalyticsResources);\n\n static create(input: unknown, ctx?: ConnectorContext): GA4Connector {\n const parsed = configFields.parse(input);\n return new GA4Connector(\n {\n propertyId: parsed.propertyId,\n lookbackDays: parsed.lookbackDays,\n },\n {\n serviceAccountJson: parsed.serviceAccountJson,\n refreshToken: parsed.refreshToken,\n clientId: parsed.clientId,\n clientSecret: parsed.clientSecret,\n },\n ctx,\n );\n }\n\n readonly id = 'google-analytics';\n override readonly credentials = ga4Credentials;\n\n private cachedToken: { token: string; expiresAt: number } | null = null;\n\n private async fetchOAuthToken(\n url: string,\n body: string,\n signal: AbortSignal | undefined,\n ): Promise<{ token: string; expiresAt: number }> {\n const res = await this.post<TokenResponse>(url, {\n resource: 'oauth_token',\n headers: { 'Content-Type': 'application/x-www-form-urlencoded' },\n body,\n signal,\n });\n const expiresIn = res.body.expires_in ?? 3600;\n return {\n token: res.body.access_token,\n expiresAt: Date.now() + (expiresIn - 60) * 1000,\n };\n }\n\n private async getAccessToken(signal?: AbortSignal): Promise<string> {\n if (this.cachedToken && Date.now() < this.cachedToken.expiresAt) {\n return this.cachedToken.token;\n }\n\n const { serviceAccountJson, refreshToken, clientId, clientSecret } =\n this.creds;\n\n if (serviceAccountJson) {\n const { url, body } = await buildServiceAccountJwt(serviceAccountJson);\n this.cachedToken = await this.fetchOAuthToken(url, body, signal);\n return this.cachedToken.token;\n }\n\n if (refreshToken && clientId && clientSecret) {\n const body = new URLSearchParams({\n grant_type: 'refresh_token',\n refresh_token: refreshToken,\n client_id: clientId,\n client_secret: clientSecret,\n }).toString();\n this.cachedToken = await this.fetchOAuthToken(\n 'https://oauth2.googleapis.com/token',\n body,\n signal,\n );\n return this.cachedToken.token;\n }\n\n throw new Error(\n 'GA4 connector: provide either serviceAccountJson or (refreshToken + clientId + clientSecret)',\n );\n }\n\n private async runReport(\n accessToken: string,\n phase: GA4Phase,\n dateRange: { startDate: string; endDate: string },\n offset: number,\n signal?: AbortSignal,\n ): Promise<GA4ReportResponse> {\n const { dimensions, metrics } = PHASE_CONFIGS[phase];\n const url = `https://analyticsdata.googleapis.com/v1beta/properties/${this.settings.propertyId}:runReport`;\n\n const body: Record<string, unknown> = {\n dimensions: dimensions.map((name) => ({ name })),\n metrics: metrics.map((name) => ({ name })),\n dateRanges: [\n { startDate: dateRange.startDate, endDate: dateRange.endDate },\n ],\n limit: ROWS_PER_PAGE,\n offset,\n };\n\n const res = await this.post<GA4ReportResponse>(url, {\n resource: phase,\n headers: {\n Authorization: `Bearer ${accessToken}`,\n 'Content-Type': 'application/json',\n 'User-Agent': connectorUserAgent('google-analytics'),\n },\n body: JSON.stringify(body),\n signal,\n });\n return res.body;\n }\n\n async sync(\n options: SyncOptions,\n storage: StorageHandle,\n signal?: AbortSignal,\n ): Promise<SyncResult> {\n const lookbackDays = this.settings.lookbackDays ?? 90;\n\n const cursor = isGA4SyncCursor(options.cursor) ? options.cursor : undefined;\n // Restore the originally-computed window on resume so phases stay aligned\n // across midnight rollovers and lookbackDays changes between runs.\n const dateRange = cursor?.dateRange ?? getDateRange(options, lookbackDays);\n\n let accessToken: string | null = null;\n const getToken = async (sig?: AbortSignal): Promise<string> => {\n if (!accessToken) {\n accessToken = await this.getAccessToken(sig);\n }\n return accessToken;\n };\n\n const runReportWithRetry = async (\n phase: GA4Phase,\n offset: number,\n sig: AbortSignal | undefined,\n ): Promise<GA4ReportResponse> => {\n const token = await getToken(sig);\n try {\n return await this.runReport(token, phase, dateRange, offset, sig);\n } catch (err) {\n console.warn(\n `[ga4] runReport failed, refreshing token and retrying once`,\n err,\n );\n accessToken = null;\n const freshToken = await getToken(sig);\n return this.runReport(freshToken, phase, dateRange, offset, sig);\n }\n };\n\n const drainPhase = async (phase: GA4Phase): Promise<GA4ReportRow[]> => {\n const allRows: GA4ReportRow[] = [];\n let offset = 0;\n for (;;) {\n const response = await runReportWithRetry(phase, offset, signal);\n const rows = response.rows ?? [];\n allRows.push(...rows);\n offset += rows.length;\n if (rows.length === 0) {\n break;\n }\n // Prefer the API's authoritative rowCount when available; fall back\n // to a short-page heuristic only when GA4 omits it, so a missing\n // field can't truncate a multi-page dataset to its first page.\n const done =\n typeof response.rowCount === 'number'\n ? offset >= response.rowCount\n : rows.length < ROWS_PER_PAGE;\n if (done) {\n break;\n }\n }\n return allRows;\n };\n\n const resumeIdx = cursor ? PHASE_ORDER.indexOf(cursor.phase) : -1;\n const startIdx = resumeIdx >= 0 ? resumeIdx : 0;\n\n for (let i = startIdx; i < PHASE_ORDER.length; i++) {\n const phase = PHASE_ORDER[i]!;\n if (signal?.aborted) {\n return { done: false, cursor: { phase, dateRange } };\n }\n\n // Drain every page of this phase in-memory before writing so the commit\n // is one atomic call. A mid-phase failure restarts this phase from\n // scratch on the next sync; the clear-and-replace below wipes partial\n // state. If the abort signal trips mid-drain, surface a resumable\n // cursor instead of throwing the AbortError up to the caller.\n let rows: GA4ReportRow[];\n try {\n rows = await drainPhase(phase);\n } catch (err) {\n if (signal?.aborted) {\n return { done: false, cursor: { phase, dateRange } };\n }\n throw err;\n }\n const cfg = PHASE_CONFIGS[phase];\n const samples = rows.map((row) =>\n rowToMetricSample(row, cfg.dimensions, cfg.metrics, cfg.metricName),\n );\n // Scoping by name ensures stale rows are wiped even when samples is empty.\n await storage.metrics(samples, { names: [cfg.metricName] });\n }\n\n return { done: true };\n }\n}\n","import { GA4Connector } from './google-analytics';\n\nexport {\n configFields,\n doc,\n GA4Connector,\n rowToMetricSample,\n} from './google-analytics';\nexport type { GA4Settings } from './google-analytics';\nexport default GA4Connector;\n"],"mappings":";AEAO,IAAM,sBAAsB;AAE5B,IAAM,qBAAqB,qBAAqB,mBAAmB;AAEnE,SAAS,mBAAmB,aAA6B;AAC9D,SAAO,qBAAqB,WAAW,IAAI,mBAAmB;AAChE;;;AOLA;AAAA,EACE;AAAA,EAOA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,SAAS;AAEX,IAAM,eAAe;AAAA,EAC1B,EACG,OAAO;AAAA,IACN,YAAY,EACT,OAAO,EACP,KAAK,EACL,MAAM,SAAS,qCAAqC,EACpD,KAAK;AAAA,MACJ,OAAO;AAAA,MACP,aACE;AAAA,MACF,aAAa;AAAA,IACf,CAAC;AAAA,IACH,oBAAoB,EAAE,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,CAAC,EAAE,SAAS,EAAE,KAAK;AAAA,MACpE,OAAO;AAAA,MACP,aACE;AAAA,MACF,QAAQ;AAAA,IACV,CAAC;AAAA,IACD,cAAc,EAAE,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,CAAC,EAAE,SAAS,EAAE,KAAK;AAAA,MAC9D,OAAO;AAAA,MACP,aACE;AAAA,MACF,QAAQ;AAAA,IACV,CAAC;AAAA,IACD,UAAU,EAAE,OAAO,EAAE,SAAS,EAAE,KAAK;AAAA,MACnC,OAAO;AAAA,MACP,aACE;AAAA,MACF,aAAa;AAAA,IACf,CAAC;AAAA,IACD,cAAc,EAAE,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,CAAC,EAAE,SAAS,EAAE,KAAK;AAAA,MAC9D,OAAO;AAAA,MACP,aACE;AAAA,MACF,QAAQ;AAAA,IACV,CAAC;AAAA,IACD,cAAc,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,SAAS,EAAE,KAAK;AAAA,MACxD,OAAO;AAAA,MACP,aACE;AAAA,MACF,aAAa;AAAA,IACf,CAAC;AAAA,EACH,CAAC,EACA;AAAA,IACC,CAAC,QACC,IAAI,uBAAuB,UAC1B,IAAI,iBAAiB,UACpB,IAAI,aAAa,UACjB,IAAI,iBAAiB;AAAA,IACzB;AAAA,MACE,SACE;AAAA,IACJ;AAAA,EACF;AACJ;AAEO,IAAM,MAAoB,mBAAmB;AAAA,EAClD,aAAa;AAAA,EACb,UAAU;AAAA,EACV,YAAY;AAAA,EACZ,SACE;AAAA,EACF,QAAQ;AAAA,IACN,MAAM;AAAA,IACN,SACE;AAAA,IACF,SAAS;AAAA,EACX;AAAA,EACA,MAAM;AAAA,IACJ,SACE;AAAA,IACF,OAAO;AAAA,MACL;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAAA,EACF;AAAA,EACA,WACE;AAAA,EACF,aAAa;AAAA,IACX;AAAA,IACA;AAAA,EACF;AACF,CAAC;AAWD,IAAM,iBAAiB;AAAA,EACrB,oBAAoB;AAAA,IAClB,aAAa;AAAA,IACb,MAAM;AAAA,EACR;AAAA,EACA,cAAc;AAAA,IACZ,aAAa;AAAA,IACb,MAAM;AAAA,EACR;AAAA,EACA,UAAU;AAAA,IACR,aAAa;AAAA,IACb,MAAM;AAAA,EACR;AAAA,EACA,cAAc;AAAA,IACZ,aAAa;AAAA,IACb,MAAM;AAAA,EACR;AACF;AAQA,IAAM,cAAc;AAAA,EAClB;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF;AAgBA,IAAM,cAAc;AAEpB,SAAS,gBAAgB,OAAiC;AACxD,SAAO,OAAO,UAAU,YAAY,YAAY,KAAK,KAAK;AAC5D;AAEA,SAAS,eAAe,OAAuC;AAC7D,MAAI,OAAO,UAAU,YAAY,UAAU,MAAM;AAC/C,WAAO;AAAA,EACT;AACA,QAAM,IAAI;AACV,SAAO,gBAAgB,EAAE,SAAS,KAAK,gBAAgB,EAAE,OAAO;AAClE;AAEA,SAAS,gBAAgB,OAAwC;AAC/D,MAAI,OAAO,UAAU,YAAY,UAAU,MAAM;AAC/C,WAAO;AAAA,EACT;AACA,QAAM,IAAI;AACV,MAAI,OAAO,EAAE,UAAU,UAAU;AAC/B,WAAO;AAAA,EACT;AACA,MAAI,CAAE,YAAkC,SAAS,EAAE,KAAK,GAAG;AACzD,WAAO;AAAA,EACT;AACA,SAAO,eAAe,EAAE,SAAS;AACnC;AAYA,IAAM,gBAA+C;AAAA,EACnD,gBAAgB;AAAA,IACd,YAAY,CAAC,MAAM;AAAA,IACnB,SAAS;AAAA,MACP;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAAA,IACA,YAAY;AAAA,EACd;AAAA,EACA,mBAAmB;AAAA,IACjB,YAAY,CAAC,QAAQ,iBAAiB,eAAe;AAAA,IACrD,SAAS,CAAC,YAAY,aAAa;AAAA,IACnC,YAAY;AAAA,EACd;AAAA,EACA,WAAW;AAAA,IACT,YAAY,CAAC,QAAQ,UAAU;AAAA,IAC/B,SAAS,CAAC,mBAAmB,wBAAwB;AAAA,IACrD,YAAY;AAAA,EACd;AAAA,EACA,QAAQ;AAAA,IACN,YAAY,CAAC,QAAQ,WAAW;AAAA,IAChC,SAAS,CAAC,cAAc,YAAY;AAAA,IACpC,YAAY;AAAA,EACd;AAAA,EACA,aAAa;AAAA,IACX,YAAY,CAAC,QAAQ,WAAW;AAAA,IAChC,SAAS,CAAC,eAAe,cAAc;AAAA,IACvC,YAAY;AAAA,EACd;AAAA,EACA,KAAK;AAAA,IACH,YAAY,CAAC,QAAQ,SAAS;AAAA,IAC9B,SAAS,CAAC,YAAY,YAAY;AAAA,IAClC,YAAY;AAAA,EACd;AACF;AAEA,IAAM,gBAAgB;AAyCtB,SAAS,mBAAmB,OAA2B;AACrD,MAAI,SAAS;AACb,WAAS,IAAI,GAAG,IAAI,MAAM,QAAQ,KAAK;AACrC,cAAU,OAAO,aAAa,MAAM,CAAC,CAAE;AAAA,EACzC;AACA,SAAO,KAAK,MAAM,EAAE,QAAQ,OAAO,GAAG,EAAE,QAAQ,OAAO,GAAG,EAAE,QAAQ,MAAM,EAAE;AAC9E;AAEA,SAAS,oBAAoB,KAAqB;AAChD,SAAO,mBAAmB,IAAI,YAAY,EAAE,OAAO,GAAG,CAAC;AACzD;AAEA,eAAe,aACb,SACA,eACiB;AACjB,QAAM,SAAS,EAAE,KAAK,SAAS,KAAK,MAAM;AAC1C,QAAM,YAAY,oBAAoB,KAAK,UAAU,MAAM,CAAC;AAC5D,QAAM,aAAa,oBAAoB,KAAK,UAAU,OAAO,CAAC;AAC9D,QAAM,eAAe,GAAG,SAAS,IAAI,UAAU;AAE/C,QAAM,aAAa,cAChB,QAAQ,gCAAgC,EAAE,EAC1C,QAAQ,8BAA8B,EAAE,EACxC,QAAQ,OAAO,EAAE;AACpB,QAAM,MAAM,WAAW,KAAK,KAAK,UAAU,GAAG,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC;AAEpE,QAAM,MAAM,MAAM,WAAW,OAAO,OAAO;AAAA,IACzC;AAAA,IACA;AAAA,IACA,EAAE,MAAM,qBAAqB,MAAM,UAAU;AAAA,IAC7C;AAAA,IACA,CAAC,MAAM;AAAA,EACT;AAEA,QAAM,YAAY,MAAM,WAAW,OAAO,OAAO;AAAA,IAC/C;AAAA,IACA;AAAA,IACA,IAAI,YAAY,EAAE,OAAO,YAAY;AAAA,EACvC;AAEA,SAAO,GAAG,YAAY,IAAI,mBAAmB,IAAI,WAAW,SAAS,CAAC,CAAC;AACzE;AAEA,SAAS,wBAAwB,OAAkC;AACjE,QAAM,UAAU,MAAM,KAAK;AAC3B,MAAI,QAAQ,WAAW,GAAG,GAAG;AAC3B,WAAO,KAAK,MAAM,OAAO;AAAA,EAC3B;AACA,QAAM,SAAS,KAAK,OAAO;AAC3B,QAAM,QAAQ,IAAI,WAAW,OAAO,MAAM;AAC1C,WAAS,IAAI,GAAG,IAAI,OAAO,QAAQ,KAAK;AACtC,UAAM,CAAC,IAAI,OAAO,WAAW,CAAC;AAAA,EAChC;AACA,QAAM,UAAU,IAAI,YAAY,EAAE,OAAO,KAAK;AAC9C,SAAO,KAAK,MAAM,OAAO;AAC3B;AAEA,eAAe,uBACb,oBACwC;AACxC,QAAM,KAAK,wBAAwB,kBAAkB;AACrD,QAAM,MAAM,KAAK,MAAM,KAAK,IAAI,IAAI,GAAI;AACxC,QAAM,MAAM,MAAM;AAAA,IAChB;AAAA,MACE,KAAK,GAAG;AAAA,MACR,OAAO;AAAA,MACP,KAAK,GAAG,aAAa;AAAA,MACrB,KAAK,MAAM;AAAA,MACX,KAAK;AAAA,IACP;AAAA,IACA,GAAG;AAAA,EACL;AAEA,QAAM,OAAO,IAAI,gBAAgB;AAAA,IAC/B,YAAY;AAAA,IACZ,WAAW;AAAA,EACb,CAAC,EAAE,SAAS;AAEZ,SAAO;AAAA,IACL,KAAK,GAAG,aAAa;AAAA,IACrB;AAAA,EACF;AACF;AAMA,SAAS,UAAU,MAAoB;AACrC,QAAM,IAAI,KAAK,eAAe;AAC9B,QAAM,IAAI,OAAO,KAAK,YAAY,IAAI,CAAC,EAAE,SAAS,GAAG,GAAG;AACxD,QAAM,IAAI,OAAO,KAAK,WAAW,CAAC,EAAE,SAAS,GAAG,GAAG;AACnD,SAAO,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC;AACvB;AAEA,SAAS,YAAY,SAAyB;AAE5C,QAAM,IAAI,QAAQ,MAAM,GAAG,CAAC;AAC5B,QAAM,IAAI,QAAQ,MAAM,GAAG,CAAC;AAC5B,QAAM,IAAI,QAAQ,MAAM,GAAG,CAAC;AAC5B,SAAO,KAAK,IAAI,OAAO,CAAC,GAAG,OAAO,CAAC,IAAI,GAAG,OAAO,CAAC,CAAC;AACrD;AAEA,IAAM,aAAa,KAAK,KAAK,KAAK;AAClC,IAAM,4BAA4B;AAElC,SAAS,aACP,SACA,cACc;AACd,QAAM,MAAM,KAAK,IAAI;AACrB,QAAM,UAAU,UAAU,IAAI,KAAK,GAAG,CAAC;AACvC,MAAI,QAAQ,SAAS,YAAY,QAAQ,OAAO;AAC9C,UAAMA,WAAU,OAAO,4BAA4B,KAAK;AACxD,WAAO,EAAE,WAAW,UAAU,IAAI,KAAKA,QAAO,CAAC,GAAG,QAAQ;AAAA,EAC5D;AACA,MAAI,QAAQ,OAAO;AACjB,UAAM,UAAU,IAAI,KAAK,QAAQ,KAAK,EAAE,QAAQ;AAChD,QAAI,OAAO,SAAS,OAAO,GAAG;AAC5B,YAAM,OAAO,KAAK,IAAI,GAAG,KAAK,MAAM,MAAM,WAAW,UAAU,CAAC;AAChE,YAAM,aAAa,KAAK,IAAI,MAAM,YAAY;AAC9C,YAAMA,WAAU,OAAO,aAAa,KAAK;AACzC,aAAO,EAAE,WAAW,UAAU,IAAI,KAAKA,QAAO,CAAC,GAAG,QAAQ;AAAA,IAC5D;AAAA,EACF;AACA,QAAM,UAAU,OAAO,eAAe,KAAK;AAC3C,SAAO,EAAE,WAAW,UAAU,IAAI,KAAK,OAAO,CAAC,GAAG,QAAQ;AAC5D;AAMO,SAAS,kBACd,KACA,kBACA,eACA,YAMA;AACA,QAAM,OAA+B,CAAC;AACtC,WAAS,IAAI,GAAG,IAAI,iBAAiB,QAAQ,KAAK;AAChD,SAAK,iBAAiB,CAAC,CAAE,IAAI,IAAI,gBAAgB,CAAC,GAAG,SAAS;AAAA,EAChE;AAEA,QAAM,OAA+B,CAAC;AACtC,WAAS,IAAI,GAAG,IAAI,cAAc,QAAQ,KAAK;AAC7C,SAAK,cAAc,CAAC,CAAE,IACpB,WAAW,IAAI,aAAa,CAAC,GAAG,SAAS,GAAG,KAAK;AAAA,EACrD;AAEA,QAAM,UAAU,KAAK,MAAM,KAAK;AAChC,QAAM,KAAK,YAAY,OAAO;AAC9B,QAAM,eAAe,KAAK,cAAc,CAAC,CAAE,KAAK;AAEhD,SAAO;AAAA,IACL,MAAM;AAAA,IACN;AAAA,IACA,OAAO;AAAA,IACP,YAAY,EAAE,GAAG,MAAM,GAAG,KAAK;AAAA,EACjC;AACF;AAUA,IAAM,qBAAqB,EAAE,OAAO;AAAA,EAClC,OAAO,EAAE,OAAO,EAAE,MAAM,oDAAoD;AAC9E,CAAC;AAED,IAAM,uBAAuB,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC;AAC3D,IAAM,qBAAqB,EAAE,OAAO;AAAA,EAClC,OAAO,EAAE,OAAO,EAAE,MAAM,iBAAiB;AAC3C,CAAC;AAED,SAAS,aAAa,gBAAwB;AAC5C,QAAM,OACJ,mBAAmB,IACf,EAAE,MAAM,CAAC,kBAAkB,CAAC,IAC5B,EAAE,MAAM;AAAA,IACN;AAAA,IACA,GAAG,MAAM,iBAAiB,CAAC,EAAE,KAAK,oBAAoB;AAAA,EACxD,CAAgD;AACtD,SAAO,EAAE,OAAO;AAAA,IACd,MAAM,EACH;AAAA,MACC,EAAE,OAAO;AAAA,QACP,iBAAiB;AAAA,QACjB,cAAc,EAAE,MAAM,kBAAkB,EAAE,SAAS;AAAA,MACrD,CAAC;AAAA,IACH,EACC,SAAS;AAAA,EACd,CAAC;AACH;AAEA,IAAM,sBAAsB,EAAE,OAAO;AAAA,EACnC,cAAc,EAAE,OAAO,EAAE,IAAI,CAAC;AAAA,EAC9B,YAAY,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,SAAS;AACnD,CAAC;AAED,IAAM,2BAA2B,gBAAgB;AAAA,EAC/C,oBAAoB;AAAA,IAClB,OAAO;AAAA,IACP,aACE;AAAA,IACF,MAAM;AAAA,IACN,aAAa;AAAA,IACb,UAAU;AAAA,IACV,YAAY;AAAA,MACV,EAAE,MAAM,QAAQ,aAAa,qCAAqC;AAAA,IACpE;AAAA,IACA,WAAW;AAAA,MACT,aAAa;AAAA,MACb,gBAAgB,aAAa,CAAC;AAAA,IAChC;AAAA,EACF;AAAA,EACA,uBAAuB;AAAA,IACrB,OAAO;AAAA,IACP,aACE;AAAA,IACF,MAAM;AAAA,IACN,aAAa;AAAA,IACb,UAAU;AAAA,IACV,YAAY;AAAA,MACV,EAAE,MAAM,QAAQ,aAAa,qCAAqC;AAAA,MAClE;AAAA,QACE,MAAM;AAAA,QACN,aAAa;AAAA,MACf;AAAA,MACA;AAAA,QACE,MAAM;AAAA,QACN,aACE;AAAA,MACJ;AAAA,IACF;AAAA,IACA,WAAW,EAAE,mBAAmB,aAAa,CAAC,EAAE;AAAA,EAClD;AAAA,EACA,eAAe;AAAA,IACb,OAAO;AAAA,IACP,aACE;AAAA,IACF,MAAM;AAAA,IACN,aAAa;AAAA,IACb,UAAU;AAAA,IACV,YAAY;AAAA,MACV,EAAE,MAAM,QAAQ,aAAa,qCAAqC;AAAA,MAClE;AAAA,QACE,MAAM;AAAA,QACN,aAAa;AAAA,MACf;AAAA,IACF;AAAA,IACA,WAAW,EAAE,WAAW,aAAa,CAAC,EAAE;AAAA,EAC1C;AAAA,EACA,YAAY;AAAA,IACV,OAAO;AAAA,IACP,aACE;AAAA,IACF,MAAM;AAAA,IACN,aAAa;AAAA,IACb,UAAU;AAAA,IACV,YAAY;AAAA,MACV,EAAE,MAAM,QAAQ,aAAa,qCAAqC;AAAA,MAClE;AAAA,QACE,MAAM;AAAA,QACN,aAAa;AAAA,MACf;AAAA,IACF;AAAA,IACA,WAAW,EAAE,QAAQ,aAAa,CAAC,EAAE;AAAA,EACvC;AAAA,EACA,iBAAiB;AAAA,IACf,OAAO;AAAA,IACP,aACE;AAAA,IACF,MAAM;AAAA,IACN,aAAa;AAAA,IACb,UAAU;AAAA,IACV,YAAY;AAAA,MACV,EAAE,MAAM,QAAQ,aAAa,qCAAqC;AAAA,MAClE;AAAA,QACE,MAAM;AAAA,QACN,aACE;AAAA,MACJ;AAAA,IACF;AAAA,IACA,WAAW,EAAE,aAAa,aAAa,CAAC,EAAE;AAAA,EAC5C;AAAA,EACA,SAAS;AAAA,IACP,OAAO;AAAA,IACP,aAAa;AAAA,IACb,MAAM;AAAA,IACN,aAAa;AAAA,IACb,UAAU;AAAA,IACV,YAAY;AAAA,MACV,EAAE,MAAM,QAAQ,aAAa,qCAAqC;AAAA,MAClE;AAAA,QACE,MAAM;AAAA,QACN,aAAa;AAAA,MACf;AAAA,IACF;AAAA,IACA,WAAW,EAAE,KAAK,aAAa,CAAC,EAAE;AAAA,EACpC;AACF,CAAC;AAEM,IAAM,eAAN,MAAM,sBAAqB,cAA2C;AAAA,EAC3E,OAAgB,KAAK;AAAA,EAErB,OAAgB,YAAY;AAAA,EAE5B,OAAgB,UAAU,qBAAqB,wBAAwB;AAAA,EAEvE,OAAO,OAAO,OAAgB,KAAsC;AAClE,UAAM,SAAS,aAAa,MAAM,KAAK;AACvC,WAAO,IAAI;AAAA,MACT;AAAA,QACE,YAAY,OAAO;AAAA,QACnB,cAAc,OAAO;AAAA,MACvB;AAAA,MACA;AAAA,QACE,oBAAoB,OAAO;AAAA,QAC3B,cAAc,OAAO;AAAA,QACrB,UAAU,OAAO;AAAA,QACjB,cAAc,OAAO;AAAA,MACvB;AAAA,MACA;AAAA,IACF;AAAA,EACF;AAAA,EAES,KAAK;AAAA,EACI,cAAc;AAAA,EAExB,cAA2D;AAAA,EAEnE,MAAc,gBACZ,KACA,MACA,QAC+C;AAC/C,UAAM,MAAM,MAAM,KAAK,KAAoB,KAAK;AAAA,MAC9C,UAAU;AAAA,MACV,SAAS,EAAE,gBAAgB,oCAAoC;AAAA,MAC/D;AAAA,MACA;AAAA,IACF,CAAC;AACD,UAAM,YAAY,IAAI,KAAK,cAAc;AACzC,WAAO;AAAA,MACL,OAAO,IAAI,KAAK;AAAA,MAChB,WAAW,KAAK,IAAI,KAAK,YAAY,MAAM;AAAA,IAC7C;AAAA,EACF;AAAA,EAEA,MAAc,eAAe,QAAuC;AAClE,QAAI,KAAK,eAAe,KAAK,IAAI,IAAI,KAAK,YAAY,WAAW;AAC/D,aAAO,KAAK,YAAY;AAAA,IAC1B;AAEA,UAAM,EAAE,oBAAoB,cAAc,UAAU,aAAa,IAC/D,KAAK;AAEP,QAAI,oBAAoB;AACtB,YAAM,EAAE,KAAK,KAAK,IAAI,MAAM,uBAAuB,kBAAkB;AACrE,WAAK,cAAc,MAAM,KAAK,gBAAgB,KAAK,MAAM,MAAM;AAC/D,aAAO,KAAK,YAAY;AAAA,IAC1B;AAEA,QAAI,gBAAgB,YAAY,cAAc;AAC5C,YAAM,OAAO,IAAI,gBAAgB;AAAA,QAC/B,YAAY;AAAA,QACZ,eAAe;AAAA,QACf,WAAW;AAAA,QACX,eAAe;AAAA,MACjB,CAAC,EAAE,SAAS;AACZ,WAAK,cAAc,MAAM,KAAK;AAAA,QAC5B;AAAA,QACA;AAAA,QACA;AAAA,MACF;AACA,aAAO,KAAK,YAAY;AAAA,IAC1B;AAEA,UAAM,IAAI;AAAA,MACR;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAc,UACZ,aACA,OACA,WACA,QACA,QAC4B;AAC5B,UAAM,EAAE,YAAY,QAAQ,IAAI,cAAc,KAAK;AACnD,UAAM,MAAM,0DAA0D,KAAK,SAAS,UAAU;AAE9F,UAAM,OAAgC;AAAA,MACpC,YAAY,WAAW,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE;AAAA,MAC/C,SAAS,QAAQ,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE;AAAA,MACzC,YAAY;AAAA,QACV,EAAE,WAAW,UAAU,WAAW,SAAS,UAAU,QAAQ;AAAA,MAC/D;AAAA,MACA,OAAO;AAAA,MACP;AAAA,IACF;AAEA,UAAM,MAAM,MAAM,KAAK,KAAwB,KAAK;AAAA,MAClD,UAAU;AAAA,MACV,SAAS;AAAA,QACP,eAAe,UAAU,WAAW;AAAA,QACpC,gBAAgB;AAAA,QAChB,cAAc,mBAAmB,kBAAkB;AAAA,MACrD;AAAA,MACA,MAAM,KAAK,UAAU,IAAI;AAAA,MACzB;AAAA,IACF,CAAC;AACD,WAAO,IAAI;AAAA,EACb;AAAA,EAEA,MAAM,KACJ,SACA,SACA,QACqB;AACrB,UAAM,eAAe,KAAK,SAAS,gBAAgB;AAEnD,UAAM,SAAS,gBAAgB,QAAQ,MAAM,IAAI,QAAQ,SAAS;AAGlE,UAAM,YAAY,QAAQ,aAAa,aAAa,SAAS,YAAY;AAEzE,QAAI,cAA6B;AACjC,UAAM,WAAW,OAAO,QAAuC;AAC7D,UAAI,CAAC,aAAa;AAChB,sBAAc,MAAM,KAAK,eAAe,GAAG;AAAA,MAC7C;AACA,aAAO;AAAA,IACT;AAEA,UAAM,qBAAqB,OACzB,OACA,QACA,QAC+B;AAC/B,YAAM,QAAQ,MAAM,SAAS,GAAG;AAChC,UAAI;AACF,eAAO,MAAM,KAAK,UAAU,OAAO,OAAO,WAAW,QAAQ,GAAG;AAAA,MAClE,SAAS,KAAK;AACZ,gBAAQ;AAAA,UACN;AAAA,UACA;AAAA,QACF;AACA,sBAAc;AACd,cAAM,aAAa,MAAM,SAAS,GAAG;AACrC,eAAO,KAAK,UAAU,YAAY,OAAO,WAAW,QAAQ,GAAG;AAAA,MACjE;AAAA,IACF;AAEA,UAAM,aAAa,OAAO,UAA6C;AACrE,YAAM,UAA0B,CAAC;AACjC,UAAI,SAAS;AACb,iBAAS;AACP,cAAM,WAAW,MAAM,mBAAmB,OAAO,QAAQ,MAAM;AAC/D,cAAM,OAAO,SAAS,QAAQ,CAAC;AAC/B,gBAAQ,KAAK,GAAG,IAAI;AACpB,kBAAU,KAAK;AACf,YAAI,KAAK,WAAW,GAAG;AACrB;AAAA,QACF;AAIA,cAAM,OACJ,OAAO,SAAS,aAAa,WACzB,UAAU,SAAS,WACnB,KAAK,SAAS;AACpB,YAAI,MAAM;AACR;AAAA,QACF;AAAA,MACF;AACA,aAAO;AAAA,IACT;AAEA,UAAM,YAAY,SAAS,YAAY,QAAQ,OAAO,KAAK,IAAI;AAC/D,UAAM,WAAW,aAAa,IAAI,YAAY;AAE9C,aAAS,IAAI,UAAU,IAAI,YAAY,QAAQ,KAAK;AAClD,YAAM,QAAQ,YAAY,CAAC;AAC3B,UAAI,QAAQ,SAAS;AACnB,eAAO,EAAE,MAAM,OAAO,QAAQ,EAAE,OAAO,UAAU,EAAE;AAAA,MACrD;AAOA,UAAI;AACJ,UAAI;AACF,eAAO,MAAM,WAAW,KAAK;AAAA,MAC/B,SAAS,KAAK;AACZ,YAAI,QAAQ,SAAS;AACnB,iBAAO,EAAE,MAAM,OAAO,QAAQ,EAAE,OAAO,UAAU,EAAE;AAAA,QACrD;AACA,cAAM;AAAA,MACR;AACA,YAAM,MAAM,cAAc,KAAK;AAC/B,YAAM,UAAU,KAAK;AAAA,QAAI,CAAC,QACxB,kBAAkB,KAAK,IAAI,YAAY,IAAI,SAAS,IAAI,UAAU;AAAA,MACpE;AAEA,YAAM,QAAQ,QAAQ,SAAS,EAAE,OAAO,CAAC,IAAI,UAAU,EAAE,CAAC;AAAA,IAC5D;AAEA,WAAO,EAAE,MAAM,KAAK;AAAA,EACtB;AACF;;;ACxxBA,IAAO,gBAAQ;","names":["startMs"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@rawdash/connector-google-analytics",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.16.0",
|
|
4
4
|
"description": "Rawdash connector for Google Analytics 4 — syncs traffic, sources, pages, events, conversions, and geo data into the six-shape storage model",
|
|
5
5
|
"license": "Apache-2.0",
|
|
6
6
|
"type": "module",
|
|
@@ -23,15 +23,15 @@
|
|
|
23
23
|
},
|
|
24
24
|
"dependencies": {
|
|
25
25
|
"zod": "^4.4.3",
|
|
26
|
-
"@rawdash/core": "0.
|
|
26
|
+
"@rawdash/core": "0.16.0"
|
|
27
27
|
},
|
|
28
28
|
"devDependencies": {
|
|
29
29
|
"fast-check": "^4.8.0",
|
|
30
30
|
"tsup": "^8.0.0",
|
|
31
31
|
"typescript": "^5.7.2",
|
|
32
32
|
"vitest": "^4.1.4",
|
|
33
|
-
"@rawdash/connector-shared": "0.
|
|
34
|
-
"@rawdash/connector-test-utils": "0.0.
|
|
33
|
+
"@rawdash/connector-shared": "0.3.0",
|
|
34
|
+
"@rawdash/connector-test-utils": "0.0.3"
|
|
35
35
|
},
|
|
36
36
|
"scripts": {
|
|
37
37
|
"build": "tsup",
|