n8n-nodes-jygse-vw-weconnect 0.1.15 → 0.2.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.
@@ -141,17 +141,15 @@ class VwWeConnect {
141
141
  }
142
142
  }
143
143
  exports.VwWeConnect = VwWeConnect;
144
- // VW OAuth2 Configuration (Updated January 2026 - from WeConnect-python)
144
+ // VW OAuth2 Configuration (Updated January 2026 - from ioBroker.vw-connect)
145
145
  const VW_CLIENT_ID = 'a24fba63-34b3-4d43-b181-942111e6bda8@apps_vw-dilab_com';
146
- const VW_SCOPE = 'openid profile badge cars dealers vin';
146
+ const VW_SCOPE = 'openid profile badge cars dealers birthdate vin';
147
147
  const VW_REDIRECT_URI = 'weconnect://authenticated';
148
148
  const VW_RESPONSE_TYPE = 'code id_token token';
149
- // WeConnect App User-Agent and headers (critical for authentication)
150
- // Using exact version from WeConnect-python v0.60.9
151
- const VW_USER_AGENT = 'Volkswagen/3.51.1-android/14';
149
+ // WeConnect App headers (from ioBroker.vw-connect v0.7.15)
150
+ const VW_USER_AGENT = 'Mozilla/5.0 (Linux; Android 14; SM-G960F) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Mobile Safari/537.36';
152
151
  const VW_APP_PACKAGE = 'com.volkswagen.weconnect';
153
- const VW_ACCEPT_LANGUAGE = 'de-de';
154
- const VW_NEWRELIC_ID = 'VgAEWV9QDRAEXFlRAAYPUA==';
152
+ const VW_ACCEPT_LANGUAGE = 'en-US,en;q=0.9';
155
153
  function generateTraceId() {
156
154
  // Generate UUID v4 format
157
155
  return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {
@@ -177,33 +175,34 @@ async function generateCodeChallenge(verifier) {
177
175
  }
178
176
  async function vwLogin(context, email, password) {
179
177
  try {
180
- const traceId = generateTraceId();
181
178
  const nonce = generateNonce();
182
- // API headers matching WeConnect-python v0.60.11 exactly
183
- const apiHeaders = {
179
+ const state = generateTraceId(); // Use UUID as state
180
+ const traceId = generateTraceId(); // For weconnect-trace-id header
181
+ const codeVerifier = generateCodeVerifier();
182
+ // Browser-like headers (from ioBroker.vw-connect v0.7.15)
183
+ const browserHeaders = {
184
184
  'User-Agent': VW_USER_AGENT,
185
- 'Accept': '*/*',
185
+ 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8',
186
186
  'Accept-Language': VW_ACCEPT_LANGUAGE,
187
- 'Content-Type': 'application/json',
188
- 'content-version': '1',
189
- 'Cache-Control': 'no-cache',
190
- 'Pragma': 'no-cache',
191
- 'x-newrelic-id': VW_NEWRELIC_ID,
192
- 'x-android-package-name': VW_APP_PACKAGE,
193
- 'weconnect-trace-id': traceId,
187
+ 'Accept-Encoding': 'gzip, deflate, br',
188
+ 'x-requested-with': VW_APP_PACKAGE,
189
+ 'upgrade-insecure-requests': '1',
194
190
  };
195
- // Step 1: Get authorization page via CARIAD BFF
196
- // WeConnect-python only sends redirect_uri and nonce to this endpoint
197
- const authorizeUrl = 'https://emea.bff.cariad.digital/user-login/v1/authorize';
191
+ // Step 1: Go directly to identity.vwgroup.io (like ioBroker adapter)
192
+ const authorizeUrl = 'https://identity.vwgroup.io/oidc/v1/authorize';
198
193
  const authorizeParams = new URLSearchParams({
194
+ client_id: VW_CLIENT_ID,
195
+ scope: VW_SCOPE,
196
+ response_type: VW_RESPONSE_TYPE,
199
197
  redirect_uri: VW_REDIRECT_URI,
200
198
  nonce: nonce,
199
+ state: state,
201
200
  });
202
- // Use WeConnect-python API headers for the initial CARIAD BFF request
201
+ // Use browser-like headers for initial request
203
202
  const authorizeResponse = await context.helpers.httpRequest({
204
203
  method: 'GET',
205
204
  url: `${authorizeUrl}?${authorizeParams.toString()}`,
206
- headers: apiHeaders,
205
+ headers: browserHeaders,
207
206
  encoding: 'text',
208
207
  returnFullResponse: true,
209
208
  ignoreHttpStatusErrors: true,
@@ -243,8 +242,8 @@ async function vwLogin(context, email, password) {
243
242
  }
244
243
  // Debug: If we got an error page (status 400/500 or "Oops" in content), throw with details
245
244
  if (httpStatusCode >= 400 || htmlContent.includes('Oops!, something went wrong')) {
246
- const preview = htmlContent.substring(0, 1000);
247
- throw new Error(`CARIAD BFF Error (HTTP ${httpStatusCode}): ${preview}`);
245
+ const preview = htmlContent.substring(0, 600);
246
+ throw new Error(`VW Identity Error (HTTP ${httpStatusCode}). URL: ${authorizeUrl}. Body: ${preview}`);
248
247
  }
249
248
  // Follow redirects manually to capture the state parameter
250
249
  let maxInitialRedirects = 5;
@@ -263,7 +262,7 @@ async function vwLogin(context, email, password) {
263
262
  const followResponse = await context.helpers.httpRequest({
264
263
  method: 'GET',
265
264
  url: followUrl,
266
- headers: apiHeaders,
265
+ headers: browserHeaders,
267
266
  encoding: 'text',
268
267
  returnFullResponse: true,
269
268
  ignoreHttpStatusErrors: true,
@@ -546,7 +545,7 @@ async function vwLogin(context, email, password) {
546
545
  const followResponse = await context.helpers.httpRequest({
547
546
  method: 'GET',
548
547
  url: redirectUrl.startsWith('http') ? redirectUrl : `https://identity.vwgroup.io${redirectUrl}`,
549
- headers: apiHeaders,
548
+ headers: browserHeaders,
550
549
  encoding: 'text',
551
550
  returnFullResponse: true,
552
551
  ignoreHttpStatusErrors: true,
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "n8n-nodes-jygse-vw-weconnect",
3
- "version": "0.1.15",
3
+ "version": "0.2.0",
4
4
  "description": "n8n community node for VW We Connect - Control your Volkswagen T6.1 and other VW vehicles",
5
5
  "keywords": [
6
6
  "n8n-community-node-package",