2023-07-31 01:26:38 -07:00
|
|
|
import type {
|
|
|
|
IAuthenticateGeneric,
|
|
|
|
ICredentialDataDecryptedObject,
|
|
|
|
ICredentialTestRequest,
|
|
|
|
ICredentialType,
|
|
|
|
IHttpRequestHelper,
|
|
|
|
INodeProperties,
|
|
|
|
} from 'n8n-workflow';
|
|
|
|
|
|
|
|
export class ZscalerZiaApi implements ICredentialType {
|
|
|
|
name = 'zscalerZiaApi';
|
|
|
|
|
|
|
|
displayName = 'Zscaler ZIA API';
|
|
|
|
|
2023-08-09 06:16:11 -07:00
|
|
|
documentationUrl = 'zscalerzia';
|
2023-07-31 01:26:38 -07:00
|
|
|
|
|
|
|
icon = 'file:icons/Zscaler.svg';
|
|
|
|
|
|
|
|
properties: INodeProperties[] = [
|
|
|
|
{
|
|
|
|
displayName: 'Cookie',
|
|
|
|
name: 'cookie',
|
|
|
|
type: 'hidden',
|
|
|
|
typeOptions: {
|
|
|
|
expirable: true,
|
|
|
|
},
|
|
|
|
default: '',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Base URL',
|
|
|
|
name: 'baseUrl',
|
|
|
|
type: 'string',
|
|
|
|
default: '',
|
|
|
|
placeholder: 'e.g. zsapi.zscalerthree.net',
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Username',
|
|
|
|
name: 'username',
|
|
|
|
type: 'string',
|
|
|
|
default: '',
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Password',
|
|
|
|
name: 'password',
|
|
|
|
type: 'string',
|
|
|
|
typeOptions: { password: true },
|
|
|
|
default: '',
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
displayName: 'Api Key',
|
|
|
|
name: 'apiKey',
|
|
|
|
type: 'string',
|
|
|
|
typeOptions: {
|
|
|
|
password: true,
|
|
|
|
},
|
|
|
|
default: '',
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
async preAuthentication(this: IHttpRequestHelper, credentials: ICredentialDataDecryptedObject) {
|
|
|
|
const { baseUrl, username, password, apiKey } = credentials;
|
|
|
|
|
|
|
|
const url = (baseUrl as string).endsWith('/')
|
|
|
|
? (baseUrl as string).slice(0, -1)
|
|
|
|
: (baseUrl as string);
|
|
|
|
|
|
|
|
const now = Date.now().toString();
|
|
|
|
|
|
|
|
const obfuscate = (key: string, timestamp: string) => {
|
|
|
|
const high = timestamp.substring(timestamp.length - 6);
|
|
|
|
let low = (parseInt(high) >> 1).toString();
|
|
|
|
|
|
|
|
let obfuscatedApiKey = '';
|
|
|
|
while (low.length < 6) {
|
|
|
|
low = '0' + low;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = 0; i < high.length; i++) {
|
|
|
|
obfuscatedApiKey += key.charAt(parseInt(high.charAt(i)));
|
|
|
|
}
|
|
|
|
for (let j = 0; j < low.length; j++) {
|
|
|
|
obfuscatedApiKey += key.charAt(parseInt(low.charAt(j)) + 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
return obfuscatedApiKey;
|
|
|
|
};
|
|
|
|
|
|
|
|
const response = await this.helpers.httpRequest({
|
|
|
|
method: 'POST',
|
|
|
|
baseURL: `https://${url}`,
|
|
|
|
url: '/api/v1/authenticatedSession',
|
|
|
|
headers: {
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
'Cache-Control': 'no-cache',
|
|
|
|
},
|
|
|
|
body: {
|
|
|
|
apiKey: obfuscate(apiKey as string, now),
|
|
|
|
username,
|
|
|
|
password,
|
|
|
|
timestamp: now,
|
|
|
|
},
|
|
|
|
returnFullResponse: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
const headers = response.headers;
|
|
|
|
|
|
|
|
const cookie = (headers['set-cookie'] as string[])
|
|
|
|
?.find((entrt) => entrt.includes('JSESSIONID'))
|
|
|
|
?.split(';')
|
|
|
|
?.find((entry) => entry.includes('JSESSIONID'));
|
|
|
|
|
|
|
|
if (!cookie) {
|
|
|
|
throw new Error('No cookie returned. Please check your credentials.');
|
|
|
|
}
|
|
|
|
|
|
|
|
return { cookie };
|
|
|
|
}
|
|
|
|
|
|
|
|
authenticate: IAuthenticateGeneric = {
|
|
|
|
type: 'generic',
|
|
|
|
properties: {
|
|
|
|
headers: {
|
|
|
|
Cookie: '={{$credentials.cookie}}',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
test: ICredentialTestRequest = {
|
|
|
|
request: {
|
|
|
|
url: '=https://{{$credentials.baseUrl}}/api/v1/authSettings/exemptedUrls',
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|