mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-17 10:02:05 +00:00
feat(Twitter Node): Node overhaul (#4788)
* 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>
This commit is contained in:
125
packages/nodes-base/nodes/Twitter/V2/GenericFunctions.ts
Normal file
125
packages/nodes-base/nodes/Twitter/V2/GenericFunctions.ts
Normal file
@@ -0,0 +1,125 @@
|
||||
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!`);
|
||||
}
|
||||
Reference in New Issue
Block a user