mirror of
https://github.com/n8n-io/n8n.git
synced 2024-11-14 16:44:07 -08:00
42721dba80
* First node set up. * Progress: all Resources and Operations updated * Upsates to all resources. * Updated tooltip for Tweet > Search > Tweet fields. * Upodate to resource locator items in User > Search. * Added e.g. to placeholders and minor copy tweaks. * Fixed Operations sorting. * Added a couple of operations back. * Removed 'Authorized API Call'. * Remove authorization header when empty * Import pkce * Add OAuth2 with new grant type to Twitter * Add pkce logic auto assign authorization code if pkce not defined * Add pkce to ui and interfaces * Fix scopes for Oauth2 twitter * Deubg + pass it through header * Add debug console, add airtable cred * Remove all console.logs, make PKCE in th body only when it exists * Remove invalid character ~ * Remove more console.logs * remove body inside query * Remove useless grantype check * Hide oauth2 twitter waiting for overhaul * Remove redundant header removal * Remove more console.logs * Add V2 twitter * Add V1 * Fix description V1, V2 * Fix description for V1 * Add Oauth2 request * Add user lookup * Add search username by ID * Search tweet feat * Wip create tweet * Generic function for returning ID * Add like and retweet feat * Add delete tweet feat * Fix Location tweets * Fix type * Feat List add members * Add scopes for dm and list * Add direct message feature * Improve response data * Fix regex * Add unit test to Twitter v2 * Fix unit test * Remove console.logs * Remove more console.logs * Handle @ in username * Minor copy tweaks. * Add return all logic * Add error for api permission error * Update message api error * Add error for date error * Add notice for TwitterOAuth2 api link * Fix display names location * fix List RLC * Fix like endpoint * Fix error message check * fix(core): Fix OAuth2 callback for grantType=clientCredentials * Improve fix for callback * update pnpm * Fix iso time for end time * sync oauth2Credential * remove unused codeVerifer in Server.ts * Add location and attachments notice * Add notice to oauth1 * Improve notice for twitter * moved credentials notice to TwitterOAuth1Api.credentials.ts --------- Co-authored-by: agobrech <ael.gobrecht@gmail.com> Co-authored-by: कारतोफ्फेलस्क्रिप्ट™ <aditya@netroy.in> Co-authored-by: Marcus <marcus@n8n.io>
126 lines
3.5 KiB
TypeScript
126 lines
3.5 KiB
TypeScript
import type { OptionsWithUrl } from 'request';
|
|
|
|
import type {
|
|
IDataObject,
|
|
IExecuteFunctions,
|
|
IExecuteSingleFunctions,
|
|
IHookFunctions,
|
|
ILoadOptionsFunctions,
|
|
INodeParameterResourceLocator,
|
|
JsonObject,
|
|
} from 'n8n-workflow';
|
|
import { NodeApiError, NodeOperationError } from 'n8n-workflow';
|
|
|
|
export async function twitterApiRequest(
|
|
this: IExecuteFunctions | IExecuteSingleFunctions | ILoadOptionsFunctions | IHookFunctions,
|
|
method: string,
|
|
resource: string,
|
|
body: IDataObject = {},
|
|
qs: IDataObject = {},
|
|
fullOutput?: boolean,
|
|
uri?: string,
|
|
option: IDataObject = {},
|
|
) {
|
|
let options: OptionsWithUrl = {
|
|
method,
|
|
body,
|
|
qs,
|
|
url: uri || `https://api.twitter.com/2${resource}`,
|
|
json: true,
|
|
};
|
|
try {
|
|
if (Object.keys(option).length !== 0) {
|
|
options = Object.assign({}, options, option);
|
|
}
|
|
if (Object.keys(body).length === 0) {
|
|
delete options.body;
|
|
}
|
|
if (Object.keys(qs).length === 0) {
|
|
delete options.qs;
|
|
}
|
|
if (fullOutput) {
|
|
return await this.helpers.requestOAuth2.call(this, 'twitterOAuth2Api', options);
|
|
} else {
|
|
const { data } = await this.helpers.requestOAuth2.call(this, 'twitterOAuth2Api', options);
|
|
return data;
|
|
}
|
|
} catch (error) {
|
|
if (error.error?.required_enrollment === 'Appropriate Level of API Access') {
|
|
throw new NodeOperationError(
|
|
this.getNode(),
|
|
'The operation requires Twitter Api to be either Basic or Pro.',
|
|
);
|
|
} else if (error.errors && error.error?.errors[0].message.includes('must be ')) {
|
|
throw new NodeOperationError(this.getNode(), error.error.errors[0].message as string);
|
|
}
|
|
throw new NodeApiError(this.getNode(), error as JsonObject);
|
|
}
|
|
}
|
|
|
|
export async function twitterApiRequestAllItems(
|
|
this: IExecuteFunctions | ILoadOptionsFunctions,
|
|
propertyName: string,
|
|
method: string,
|
|
endpoint: string,
|
|
body: IDataObject = {},
|
|
query: IDataObject = {},
|
|
) {
|
|
const returnData: IDataObject[] = [];
|
|
|
|
let responseData;
|
|
|
|
query.max_results = 10;
|
|
|
|
do {
|
|
responseData = await twitterApiRequest.call(this, method, endpoint, body, query, true);
|
|
query.next_token = responseData.meta.next_token as string;
|
|
returnData.push.apply(returnData, responseData[propertyName] as IDataObject[]);
|
|
} while (responseData.meta.next_token);
|
|
|
|
return returnData;
|
|
}
|
|
|
|
export function returnId(tweetId: INodeParameterResourceLocator) {
|
|
if (tweetId.mode === 'id') {
|
|
return tweetId.value as string;
|
|
} else if (tweetId.mode === 'url') {
|
|
const value = tweetId.value as string;
|
|
const tweetIdMatch = value.includes('lists')
|
|
? value.match(/^https?:\/\/twitter\.com\/(?:#!\/)?(\w+)\/list(s)?\/(\d+)$/)
|
|
: value.match(/^https?:\/\/twitter\.com\/(?:#!\/)?(\w+)\/status(es)?\/(\d+)$/);
|
|
|
|
return tweetIdMatch?.[3] as string;
|
|
} else {
|
|
throw new Error(`The mode ${tweetId.mode} is not valid!`);
|
|
}
|
|
}
|
|
|
|
export async function returnIdFromUsername(
|
|
this: IExecuteFunctions,
|
|
usernameRlc: INodeParameterResourceLocator,
|
|
) {
|
|
usernameRlc.value = (usernameRlc.value as string).includes('@')
|
|
? (usernameRlc.value as string).replace('@', '')
|
|
: usernameRlc.value;
|
|
if (
|
|
usernameRlc.mode === 'username' ||
|
|
(usernameRlc.mode === 'name' && this.getNode().parameters.list !== undefined)
|
|
) {
|
|
const user = (await twitterApiRequest.call(
|
|
this,
|
|
'GET',
|
|
`/users/by/username/${usernameRlc.value}`,
|
|
{},
|
|
)) as { id: string };
|
|
return user.id;
|
|
} else if (this.getNode().parameters.list === undefined) {
|
|
const list = (await twitterApiRequest.call(
|
|
this,
|
|
'GET',
|
|
`/list/by/name/${usernameRlc.value}`,
|
|
{},
|
|
)) as { id: string };
|
|
return list.id;
|
|
} else throw new Error(`The username mode ${usernameRlc.mode} is not valid!`);
|
|
}
|