feat(editor): Select credentials in template setup if theres only one (#7879)

Automatically select credentials in the template credential setup if
there's only one available.

NOTE! This feature is still behind a feature flag. To test, set:

```js
localStorage.setItem('template-credentials-setup', 'true')
```


https://github.com/n8n-io/n8n/assets/10324676/edc1f586-214f-4c37-b7ec-dd1786433dc1
This commit is contained in:
Tomi Turtiainen 2023-11-30 15:46:14 +02:00 committed by GitHub
parent e0b7f89035
commit fe3417a615
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 466 additions and 0 deletions

View file

@ -817,10 +817,20 @@ export interface ITemplatesWorkflow {
};
}
export interface ITemplatesWorkflowInfo {
nodeCount: number;
nodeTypes: {
[key: string]: {
count: number;
};
};
}
export interface ITemplatesWorkflowResponse extends ITemplatesWorkflow, IWorkflowTemplate {
description: string | null;
image: ITemplatesImage[];
categories: ITemplatesCategory[];
workflowInfo: ITemplatesWorkflowInfo;
}
/**
@ -1010,6 +1020,7 @@ export interface IVersionNode {
}
export interface ITemplatesNode extends IVersionNode {
id: number;
categories?: ITemplatesCategory[];
}

View file

@ -1,10 +1,16 @@
import { useTemplatesStore } from '@/stores/templates.store';
import type { IWorkflowTemplateNodeWithCredentials } from '@/utils/templates/templateTransforms';
import type { CredentialUsages } from '@/views/SetupWorkflowFromTemplateView/setupTemplate.store';
import {
getAppCredentials,
getAppsRequiringCredentials,
groupNodeCredentialsByName,
useSetupTemplateStore,
} from '@/views/SetupWorkflowFromTemplateView/setupTemplate.store';
import { setActivePinia } from 'pinia';
import * as testData from './setupTemplate.store.testData';
import { createTestingPinia } from '@pinia/testing';
import { useCredentialsStore } from '@/stores/credentials.store';
const objToMap = <T>(obj: Record<string, T>) => {
return new Map<string, T>(Object.entries(obj));
@ -145,4 +151,111 @@ describe('SetupWorkflowFromTemplateView store', () => {
]);
});
});
describe('setInitialCredentialsSelection', () => {
beforeEach(() => {
setActivePinia(
createTestingPinia({
stubActions: false,
}),
);
});
it("selects no credentials when there isn't any available", () => {
// Setup
const credentialsStore = useCredentialsStore();
credentialsStore.setCredentialTypes([testData.credentialTypeTelegram]);
const templatesStore = useTemplatesStore();
templatesStore.addWorkflows([testData.fullShopifyTelegramTwitterTemplate]);
const setupTemplateStore = useSetupTemplateStore();
setupTemplateStore.setTemplateId(testData.fullShopifyTelegramTwitterTemplate.id.toString());
// Execute
setupTemplateStore.setInitialCredentialSelection();
expect(setupTemplateStore.selectedCredentialIdByName).toEqual({});
});
it("selects credential when there's only one", () => {
// Setup
const credentialsStore = useCredentialsStore();
credentialsStore.setCredentialTypes([testData.credentialTypeTelegram]);
credentialsStore.setCredentials([testData.credentialsTelegram1]);
const templatesStore = useTemplatesStore();
templatesStore.addWorkflows([testData.fullShopifyTelegramTwitterTemplate]);
const setupTemplateStore = useSetupTemplateStore();
setupTemplateStore.setTemplateId(testData.fullShopifyTelegramTwitterTemplate.id.toString());
// Execute
setupTemplateStore.setInitialCredentialSelection();
expect(setupTemplateStore.selectedCredentialIdByName).toEqual({
telegram_habot: 'YaSKdvEcT1TSFrrr1',
});
});
it('selects no credentials when there are more than 1 available', () => {
// Setup
const credentialsStore = useCredentialsStore();
credentialsStore.setCredentialTypes([testData.credentialTypeTelegram]);
credentialsStore.setCredentials([
testData.credentialsTelegram1,
testData.credentialsTelegram2,
]);
const templatesStore = useTemplatesStore();
templatesStore.addWorkflows([testData.fullShopifyTelegramTwitterTemplate]);
const setupTemplateStore = useSetupTemplateStore();
setupTemplateStore.setTemplateId(testData.fullShopifyTelegramTwitterTemplate.id.toString());
// Execute
setupTemplateStore.setInitialCredentialSelection();
expect(setupTemplateStore.selectedCredentialIdByName).toEqual({});
});
test.each([
['httpBasicAuth'],
['httpCustomAuth'],
['httpDigestAuth'],
['httpHeaderAuth'],
['oAuth1Api'],
['oAuth2Api'],
['httpQueryAuth'],
])('does not auto-select credentials for %s', (credentialType) => {
// Setup
const credentialsStore = useCredentialsStore();
credentialsStore.setCredentialTypes([testData.newCredentialType(credentialType)]);
credentialsStore.setCredentials([
testData.newCredential({
name: `${credentialType}Credential`,
type: credentialType,
}),
]);
const templatesStore = useTemplatesStore();
const workflow = testData.newFullOneNodeTemplate({
name: 'Test',
type: 'n8n-nodes-base.httpRequest',
typeVersion: 1,
credentials: {
[credentialType]: 'Test',
},
parameters: {},
position: [250, 300],
});
templatesStore.addWorkflows([workflow]);
const setupTemplateStore = useSetupTemplateStore();
setupTemplateStore.setTemplateId(workflow.id.toString());
// Execute
setupTemplateStore.setInitialCredentialSelection();
expect(setupTemplateStore.credentialUsages.length).toBe(1);
expect(setupTemplateStore.selectedCredentialIdByName).toEqual({});
});
});
});

View file

@ -0,0 +1,309 @@
import { faker } from '@faker-js/faker/locale/en';
import type {
ICredentialsResponse,
ITemplatesWorkflowFull,
IWorkflowTemplateNode,
} from '@/Interface';
import type { ICredentialType } from 'n8n-workflow';
export const newFullOneNodeTemplate = (node: IWorkflowTemplateNode): ITemplatesWorkflowFull => ({
full: true,
id: faker.number.int(),
name: faker.commerce.productName(),
totalViews: 1,
createdAt: '2021-08-24T10:40:50.007Z',
description: faker.lorem.paragraph(),
workflow: {
nodes: [node],
connections: {},
},
nodes: [
{
defaults: {},
displayName: faker.commerce.productName(),
icon: 'file:telegram.svg',
iconData: {
fileBuffer: '',
type: 'file',
},
id: faker.number.int(),
name: node.type,
},
],
image: [],
categories: [],
user: {
username: faker.internet.userName(),
},
workflowInfo: {
nodeCount: 1,
nodeTypes: {
[node.type]: {
count: 1,
},
},
},
});
export const fullShopifyTelegramTwitterTemplate: ITemplatesWorkflowFull = {
full: true,
id: 1205,
name: 'Promote new Shopify products on Twitter and Telegram',
totalViews: 485,
createdAt: '2021-08-24T10:40:50.007Z',
description:
'This workflow automatically promotes your new Shopify products on Twitter and Telegram. This workflow is also featured in the blog post [*6 e-commerce workflows to power up your Shopify store*](https://n8n.io/blog/no-code-ecommerce-workflow-automations/#promote-your-new-products-on-social-media).\n\n## Prerequisites\n\n- A Shopify account and [credentials](https://docs.n8n.io/integrations/credentials/shopify/)\n- A Twitter account and [credentials](https://docs.n8n.io/integrations/credentials/twitter/)\n- A Telegram account and [credentials](https://docs.n8n.io/integrations/credentials/telegram/) for the channel you want to send messages to.\n\n## Nodes\n\n- [Shopify Trigger node](https://docs.n8n.io/integrations/trigger-nodes/n8n-nodes-base.shopifytrigger/) triggers the workflow when you create a new product in Shopify.\n- [Twitter node](https://docs.n8n.io/integrations/nodes/n8n-nodes-base.twitter/) posts a tweet with the text "Hey there, my design is now on a new product! Visit my {shop name} to get this cool {product title} (and check out more {product type})".\n- [Telegram node](https://docs.n8n.io/integrations/nodes/n8n-nodes-base.telegram/) posts a message with the same text as above in a Telegram channel.',
workflow: {
nodes: [
{
name: 'Twitter',
type: 'n8n-nodes-base.twitter',
position: [720, -220],
parameters: {
text: '=Hey there, my design is now on a new product ✨\nVisit my {{$json["vendor"]}} shop to get this cool{{$json["title"]}} (and check out more {{$json["product_type"]}}) 🛍️',
additionalFields: {},
},
credentials: {
twitterOAuth1Api: 'twitter',
},
typeVersion: 1,
},
{
name: 'Telegram',
type: 'n8n-nodes-base.telegram',
position: [720, -20],
parameters: {
text: '=Hey there, my design is now on a new product!\nVisit my {{$json["vendor"]}} shop to get this cool{{$json["title"]}} (and check out more {{$json["product_type"]}})',
chatId: '123456',
additionalFields: {},
},
credentials: {
telegramApi: 'telegram_habot',
},
typeVersion: 1,
},
{
name: 'product created',
type: 'n8n-nodes-base.shopifyTrigger',
position: [540, -110],
webhookId: '2a7e0e50-8f09-4a2b-bf54-a849a6ac4fe0',
parameters: {
topic: 'products/create',
},
credentials: {
shopifyApi: 'shopify_nodeqa',
},
typeVersion: 1,
},
],
connections: {
'product created': {
main: [
[
{
node: 'Twitter',
type: 'main',
index: 0,
},
{
node: 'Telegram',
type: 'main',
index: 0,
},
],
],
},
},
},
workflowInfo: {
nodeCount: 3,
nodeTypes: {
'n8n-nodes-base.twitter': {
count: 1,
},
'n8n-nodes-base.telegram': {
count: 1,
},
'n8n-nodes-base.shopifyTrigger': {
count: 1,
},
},
},
user: {
username: 'lorenanda',
},
nodes: [
{
id: 49,
icon: 'file:telegram.svg',
name: 'n8n-nodes-base.telegram',
defaults: {
name: 'Telegram',
},
iconData: {
type: 'file',
fileBuffer:
'',
},
categories: [
{
id: 6,
name: 'Communication',
},
],
displayName: 'Telegram',
typeVersion: 1,
},
{
id: 107,
icon: 'file:shopify.svg',
name: 'n8n-nodes-base.shopifyTrigger',
defaults: {
name: 'Shopify Trigger',
},
iconData: {
type: 'file',
fileBuffer:
'',
},
categories: [
{
id: 2,
name: 'Sales',
},
],
displayName: 'Shopify Trigger',
typeVersion: 1,
},
{
id: 325,
icon: 'file:x.svg',
name: 'n8n-nodes-base.twitter',
defaults: {
name: 'X',
},
iconData: {
type: 'file',
fileBuffer:
'',
},
categories: [
{
id: 1,
name: 'Marketing & Content',
},
],
displayName: 'X (Formerly Twitter)',
typeVersion: 2,
},
],
categories: [
{
id: 2,
name: 'Sales',
},
{
id: 19,
name: 'Marketing & Growth',
},
],
image: [
{
id: 527,
url: 'https://n8niostorageaccount.blob.core.windows.net/n8nio-strapi-blobs-prod/assets/89a078b208fe4c6181902608b1cd1332.png',
},
],
};
export const newCredentialType = (name: string): ICredentialType => ({
name,
displayName: name,
documentationUrl: name,
properties: [],
});
export const newCredential = (
opts: Pick<ICredentialsResponse, 'type'> & Partial<ICredentialsResponse>,
): ICredentialsResponse => ({
createdAt: faker.date.past().toISOString(),
updatedAt: faker.date.past().toISOString(),
id: faker.string.alphanumeric({ length: 16 }),
name: faker.commerce.productName(),
nodesAccess: [],
...opts,
});
export const credentialTypeTelegram: ICredentialType = {
name: 'telegramApi',
displayName: 'Telegram API',
documentationUrl: 'telegram',
properties: [
{
displayName: 'Access Token',
name: 'accessToken',
type: 'string',
typeOptions: {
password: true,
},
default: '',
description:
'Chat with the <a href="https://telegram.me/botfather">bot father</a> to obtain the access token',
},
],
test: {
request: {
baseURL: '=https://api.telegram.org/bot{{$credentials.accessToken}}',
url: '/getMe',
},
},
};
export const credentialsTelegram1: ICredentialsResponse = {
createdAt: '2023-11-23T14:26:07.969Z',
updatedAt: '2023-11-23T14:26:07.964Z',
id: 'YaSKdvEcT1TSFrrr1',
name: 'Telegram account',
type: 'telegramApi',
nodesAccess: [
{
nodeType: 'n8n-nodes-base.telegram',
date: new Date('2023-11-23T14:26:07.962Z'),
},
{
nodeType: 'n8n-nodes-base.telegramTrigger',
date: new Date('2023-11-23T14:26:07.962Z'),
},
],
ownedBy: {
id: '713ef3e7-9e65-4b0a-893c-8a653cbb2c4f',
email: 'user@n8n.io',
firstName: 'Player',
lastName: 'One',
},
sharedWith: [],
};
export const credentialsTelegram2: ICredentialsResponse = {
createdAt: '2023-11-23T14:26:07.969Z',
updatedAt: '2023-11-23T14:26:07.964Z',
id: 'YaSKdvEcT1TSFrrr2',
name: 'Telegram account',
type: 'telegramApi',
nodesAccess: [
{
nodeType: 'n8n-nodes-base.telegram',
date: new Date('2023-11-23T14:26:07.962Z'),
},
{
nodeType: 'n8n-nodes-base.telegramTrigger',
date: new Date('2023-11-23T14:26:07.962Z'),
},
],
ownedBy: {
id: '713ef3e7-9e65-4b0a-893c-8a653cbb2c4f',
email: 'user@n8n.io',
firstName: 'Player',
lastName: 'One',
},
sharedWith: [],
};

View file

@ -232,6 +232,34 @@ export const useSetupTemplateStore = defineStore('setupTemplate', () => {
templateId.value = id;
};
const ignoredAutoFillCredentialTypes = new Set([
'httpBasicAuth',
'httpCustomAuth',
'httpDigestAuth',
'httpHeaderAuth',
'oAuth1Api',
'oAuth2Api',
'httpQueryAuth',
]);
/**
* Selects initial credentials for the template. Credentials
* need to be loaded before this.
*/
const setInitialCredentialSelection = () => {
for (const credUsage of credentialUsages.value) {
if (ignoredAutoFillCredentialTypes.has(credUsage.credentialType)) {
continue;
}
const availableCreds = credentialsStore.getCredentialsByType(credUsage.credentialType);
if (availableCreds.length === 1) {
selectedCredentialIdByName.value[credUsage.credentialName] = availableCreds[0].id;
}
}
};
/**
* Loads the template if it hasn't been loaded yet.
*/
@ -241,6 +269,8 @@ export const useSetupTemplateStore = defineStore('setupTemplate', () => {
}
await templatesStore.fetchTemplateById(templateId.value);
setInitialCredentialSelection();
};
/**
@ -257,6 +287,8 @@ export const useSetupTemplateStore = defineStore('setupTemplate', () => {
nodeTypesStore.loadNodeTypesIfNotLoaded(),
loadTemplateIfNeeded(),
]);
setInitialCredentialSelection();
} finally {
isLoading.value = false;
}
@ -341,6 +373,7 @@ export const useSetupTemplateStore = defineStore('setupTemplate', () => {
skipSetup,
init,
loadTemplateIfNeeded,
setInitialCredentialSelection,
setTemplateId,
setSelectedCredentialId,
unsetSelectedCredential,