- Add full Telegram bot functionality with Z.AI API integration
- Implement 4 tools: Bash, FileEdit, WebSearch, Git
- Add 3 agents: Code Reviewer, Architect, DevOps Engineer
- Add 6 skills for common coding tasks
- Add systemd service file for 24/7 operation
- Add nginx configuration for HTTPS webhook
- Add comprehensive documentation
- Implement WebSocket server for real-time updates
- Add logging system with Winston
- Add environment validation
🤖 zCode CLI X - Agentic coder with Z.AI + Telegram integration
240 lines
10 KiB
JavaScript
240 lines
10 KiB
JavaScript
"use strict";
|
|
// Copyright 2023 Google LLC
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
exports.ExternalAccountAuthorizedUserClient = exports.EXTERNAL_ACCOUNT_AUTHORIZED_USER_TYPE = void 0;
|
|
const authclient_1 = require("./authclient");
|
|
const oauth2common_1 = require("./oauth2common");
|
|
const gaxios_1 = require("gaxios");
|
|
const stream = require("stream");
|
|
const baseexternalclient_1 = require("./baseexternalclient");
|
|
/**
|
|
* The credentials JSON file type for external account authorized user clients.
|
|
*/
|
|
exports.EXTERNAL_ACCOUNT_AUTHORIZED_USER_TYPE = 'external_account_authorized_user';
|
|
const DEFAULT_TOKEN_URL = 'https://sts.{universeDomain}/v1/oauthtoken';
|
|
/**
|
|
* Handler for token refresh requests sent to the token_url endpoint for external
|
|
* authorized user credentials.
|
|
*/
|
|
class ExternalAccountAuthorizedUserHandler extends oauth2common_1.OAuthClientAuthHandler {
|
|
/**
|
|
* Initializes an ExternalAccountAuthorizedUserHandler instance.
|
|
* @param url The URL of the token refresh endpoint.
|
|
* @param transporter The transporter to use for the refresh request.
|
|
* @param clientAuthentication The client authentication credentials to use
|
|
* for the refresh request.
|
|
*/
|
|
constructor(url, transporter, clientAuthentication) {
|
|
super(clientAuthentication);
|
|
this.url = url;
|
|
this.transporter = transporter;
|
|
}
|
|
/**
|
|
* Requests a new access token from the token_url endpoint using the provided
|
|
* refresh token.
|
|
* @param refreshToken The refresh token to use to generate a new access token.
|
|
* @param additionalHeaders Optional additional headers to pass along the
|
|
* request.
|
|
* @return A promise that resolves with the token refresh response containing
|
|
* the requested access token and its expiration time.
|
|
*/
|
|
async refreshToken(refreshToken, additionalHeaders) {
|
|
const values = new URLSearchParams({
|
|
grant_type: 'refresh_token',
|
|
refresh_token: refreshToken,
|
|
});
|
|
const headers = {
|
|
'Content-Type': 'application/x-www-form-urlencoded',
|
|
...additionalHeaders,
|
|
};
|
|
const opts = {
|
|
...ExternalAccountAuthorizedUserHandler.RETRY_CONFIG,
|
|
url: this.url,
|
|
method: 'POST',
|
|
headers,
|
|
data: values.toString(),
|
|
responseType: 'json',
|
|
};
|
|
// Apply OAuth client authentication.
|
|
this.applyClientAuthenticationOptions(opts);
|
|
try {
|
|
const response = await this.transporter.request(opts);
|
|
// Successful response.
|
|
const tokenRefreshResponse = response.data;
|
|
tokenRefreshResponse.res = response;
|
|
return tokenRefreshResponse;
|
|
}
|
|
catch (error) {
|
|
// Translate error to OAuthError.
|
|
if (error instanceof gaxios_1.GaxiosError && error.response) {
|
|
throw (0, oauth2common_1.getErrorFromOAuthErrorResponse)(error.response.data,
|
|
// Preserve other fields from the original error.
|
|
error);
|
|
}
|
|
// Request could fail before the server responds.
|
|
throw error;
|
|
}
|
|
}
|
|
}
|
|
/**
|
|
* External Account Authorized User Client. This is used for OAuth2 credentials
|
|
* sourced using external identities through Workforce Identity Federation.
|
|
* Obtaining the initial access and refresh token can be done through the
|
|
* Google Cloud CLI.
|
|
*/
|
|
class ExternalAccountAuthorizedUserClient extends authclient_1.AuthClient {
|
|
/**
|
|
* Instantiates an ExternalAccountAuthorizedUserClient instances using the
|
|
* provided JSON object loaded from a credentials files.
|
|
* An error is throws if the credential is not valid.
|
|
* @param options The external account authorized user option object typically
|
|
* from the external accoutn authorized user JSON credential file.
|
|
* @param additionalOptions **DEPRECATED, all options are available in the
|
|
* `options` parameter.** Optional additional behavior customization options.
|
|
* These currently customize expiration threshold time and whether to retry
|
|
* on 401/403 API request errors.
|
|
*/
|
|
constructor(options, additionalOptions) {
|
|
var _a;
|
|
super({ ...options, ...additionalOptions });
|
|
if (options.universe_domain) {
|
|
this.universeDomain = options.universe_domain;
|
|
}
|
|
this.refreshToken = options.refresh_token;
|
|
const clientAuth = {
|
|
confidentialClientType: 'basic',
|
|
clientId: options.client_id,
|
|
clientSecret: options.client_secret,
|
|
};
|
|
this.externalAccountAuthorizedUserHandler =
|
|
new ExternalAccountAuthorizedUserHandler((_a = options.token_url) !== null && _a !== void 0 ? _a : DEFAULT_TOKEN_URL.replace('{universeDomain}', this.universeDomain), this.transporter, clientAuth);
|
|
this.cachedAccessToken = null;
|
|
this.quotaProjectId = options.quota_project_id;
|
|
// As threshold could be zero,
|
|
// eagerRefreshThresholdMillis || EXPIRATION_TIME_OFFSET will override the
|
|
// zero value.
|
|
if (typeof (additionalOptions === null || additionalOptions === void 0 ? void 0 : additionalOptions.eagerRefreshThresholdMillis) !== 'number') {
|
|
this.eagerRefreshThresholdMillis = baseexternalclient_1.EXPIRATION_TIME_OFFSET;
|
|
}
|
|
else {
|
|
this.eagerRefreshThresholdMillis = additionalOptions
|
|
.eagerRefreshThresholdMillis;
|
|
}
|
|
this.forceRefreshOnFailure = !!(additionalOptions === null || additionalOptions === void 0 ? void 0 : additionalOptions.forceRefreshOnFailure);
|
|
}
|
|
async getAccessToken() {
|
|
// If cached access token is unavailable or expired, force refresh.
|
|
if (!this.cachedAccessToken || this.isExpired(this.cachedAccessToken)) {
|
|
await this.refreshAccessTokenAsync();
|
|
}
|
|
// Return GCP access token in GetAccessTokenResponse format.
|
|
return {
|
|
token: this.cachedAccessToken.access_token,
|
|
res: this.cachedAccessToken.res,
|
|
};
|
|
}
|
|
async getRequestHeaders() {
|
|
const accessTokenResponse = await this.getAccessToken();
|
|
const headers = {
|
|
Authorization: `Bearer ${accessTokenResponse.token}`,
|
|
};
|
|
return this.addSharedMetadataHeaders(headers);
|
|
}
|
|
request(opts, callback) {
|
|
if (callback) {
|
|
this.requestAsync(opts).then(r => callback(null, r), e => {
|
|
return callback(e, e.response);
|
|
});
|
|
}
|
|
else {
|
|
return this.requestAsync(opts);
|
|
}
|
|
}
|
|
/**
|
|
* Authenticates the provided HTTP request, processes it and resolves with the
|
|
* returned response.
|
|
* @param opts The HTTP request options.
|
|
* @param reAuthRetried Whether the current attempt is a retry after a failed attempt due to an auth failure.
|
|
* @return A promise that resolves with the successful response.
|
|
*/
|
|
async requestAsync(opts, reAuthRetried = false) {
|
|
let response;
|
|
try {
|
|
const requestHeaders = await this.getRequestHeaders();
|
|
opts.headers = opts.headers || {};
|
|
if (requestHeaders && requestHeaders['x-goog-user-project']) {
|
|
opts.headers['x-goog-user-project'] =
|
|
requestHeaders['x-goog-user-project'];
|
|
}
|
|
if (requestHeaders && requestHeaders.Authorization) {
|
|
opts.headers.Authorization = requestHeaders.Authorization;
|
|
}
|
|
response = await this.transporter.request(opts);
|
|
}
|
|
catch (e) {
|
|
const res = e.response;
|
|
if (res) {
|
|
const statusCode = res.status;
|
|
// Retry the request for metadata if the following criteria are true:
|
|
// - We haven't already retried. It only makes sense to retry once.
|
|
// - The response was a 401 or a 403
|
|
// - The request didn't send a readableStream
|
|
// - forceRefreshOnFailure is true
|
|
const isReadableStream = res.config.data instanceof stream.Readable;
|
|
const isAuthErr = statusCode === 401 || statusCode === 403;
|
|
if (!reAuthRetried &&
|
|
isAuthErr &&
|
|
!isReadableStream &&
|
|
this.forceRefreshOnFailure) {
|
|
await this.refreshAccessTokenAsync();
|
|
return await this.requestAsync(opts, true);
|
|
}
|
|
}
|
|
throw e;
|
|
}
|
|
return response;
|
|
}
|
|
/**
|
|
* Forces token refresh, even if unexpired tokens are currently cached.
|
|
* @return A promise that resolves with the refreshed credential.
|
|
*/
|
|
async refreshAccessTokenAsync() {
|
|
// Refresh the access token using the refresh token.
|
|
const refreshResponse = await this.externalAccountAuthorizedUserHandler.refreshToken(this.refreshToken);
|
|
this.cachedAccessToken = {
|
|
access_token: refreshResponse.access_token,
|
|
expiry_date: new Date().getTime() + refreshResponse.expires_in * 1000,
|
|
res: refreshResponse.res,
|
|
};
|
|
if (refreshResponse.refresh_token !== undefined) {
|
|
this.refreshToken = refreshResponse.refresh_token;
|
|
}
|
|
return this.cachedAccessToken;
|
|
}
|
|
/**
|
|
* Returns whether the provided credentials are expired or not.
|
|
* If there is no expiry time, assumes the token is not expired or expiring.
|
|
* @param credentials The credentials to check for expiration.
|
|
* @return Whether the credentials are expired or not.
|
|
*/
|
|
isExpired(credentials) {
|
|
const now = new Date().getTime();
|
|
return credentials.expiry_date
|
|
? now >= credentials.expiry_date - this.eagerRefreshThresholdMillis
|
|
: false;
|
|
}
|
|
}
|
|
exports.ExternalAccountAuthorizedUserClient = ExternalAccountAuthorizedUserClient;
|