From 33e35b519ef2a922d13f1df5721382e0294baf5e Mon Sep 17 00:00:00 2001 From: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> Date: Mon, 16 Sep 2024 12:27:44 +0200 Subject: [PATCH] fix: Prevent copying workflow when copying outside of canvas (#10813) --- cypress/e2e/45-ai-assistant.cy.ts | 55 ++++ .../aiAssistant/code_snippet_response.json | 28 ++ cypress/pages/features/ai-assistant.ts | 1 + .../AskAssistantChat/AskAssistantChat.vue | 1 + .../__tests__/AskAssistantChat.spec.ts | 20 ++ .../AskAssistantChat.spec.ts.snap | 242 ++++++++++++++++++ .../AskAssistant/AskAssistantChat.vue | 2 + 7 files changed, 349 insertions(+) create mode 100644 cypress/fixtures/aiAssistant/code_snippet_response.json diff --git a/cypress/e2e/45-ai-assistant.cy.ts b/cypress/e2e/45-ai-assistant.cy.ts index 01d07cbd2ba64b..7bf97eeaf9c20b 100644 --- a/cypress/e2e/45-ai-assistant.cy.ts +++ b/cypress/e2e/45-ai-assistant.cy.ts @@ -374,3 +374,58 @@ describe('AI Assistant Credential Help', () => { aiAssistant.getters.credentialEditAssistantButton().should('be.disabled'); }); }); + +describe('General help', () => { + beforeEach(() => { + aiAssistant.actions.enableAssistant(); + wf.actions.visit(); + }); + + it('assistant returns code snippet', () => { + cy.intercept('POST', '/rest/ai-assistant/chat', { + statusCode: 200, + fixture: 'aiAssistant/code_snippet_response.json', + }).as('chatRequest'); + + aiAssistant.getters.askAssistantFloatingButton().should('be.visible'); + aiAssistant.getters.askAssistantFloatingButton().click(); + aiAssistant.getters.askAssistantChat().should('be.visible'); + aiAssistant.getters.placeholderMessage().should('be.visible'); + aiAssistant.getters.chatInput().should('be.visible'); + + aiAssistant.getters.chatInput().type('Show me an expression'); + aiAssistant.getters.sendMessageButton().click(); + + aiAssistant.getters.chatMessagesAll().should('have.length', 3); + aiAssistant.getters.chatMessagesUser().eq(0).should('contain.text', 'Show me an expression'); + + aiAssistant.getters + .chatMessagesAssistant() + .eq(0) + .should('contain.text', 'To use expressions in n8n, follow these steps:'); + + aiAssistant.getters + .chatMessagesAssistant() + .eq(0) + .should( + 'include.html', + `
[
+ {
+ "headers": {
+ "host": "n8n.instance.address",
+ ...
+ },
+ "params": {},
+ "query": {},
+ "body": {
+ "name": "Jim",
+ "age": 30,
+ "city": "New York"
+ }
+ }
+]
+
`,
+ );
+ aiAssistant.getters.codeSnippet().should('have.text', '{{$json.body.city}}');
+ });
+});
diff --git a/cypress/fixtures/aiAssistant/code_snippet_response.json b/cypress/fixtures/aiAssistant/code_snippet_response.json
new file mode 100644
index 00000000000000..b05f212de13d6a
--- /dev/null
+++ b/cypress/fixtures/aiAssistant/code_snippet_response.json
@@ -0,0 +1,28 @@
+{
+ "sessionId": "f1d19ed5-0d55-4bad-b49a-f0c56bd6f76f-705b5dbf-12d4-4805-87a3-1e5b3c716d29-W1JgVNrpfitpSNF9rAjB4",
+ "messages": [
+ {
+ "role": "assistant",
+ "type": "message",
+ "text": "To use expressions in n8n, follow these steps:\n\n1. Hover over the parameter where you want to use an expression.\n2. Select **Expressions** in the **Fixed/Expression** toggle.\n3. Write your expression in the parameter, or select **Open expression editor** to open the expressions editor. You can browse the available data in the **Variable selector**. All expressions have the format `{{ your expression here }}`.\n\n### Example: Get data from webhook body\n\nIf your webhook data looks like this:\n\n```json\n[\n {\n \"headers\": {\n \"host\": \"n8n.instance.address\",\n ...\n },\n \"params\": {},\n \"query\": {},\n \"body\": {\n \"name\": \"Jim\",\n \"age\": 30,\n \"city\": \"New York\"\n }\n }\n]\n```\n\nYou can use the following expression to get the value of `city`:\n\n```js\n{{$json.body.city}}\n```\n\nThis expression accesses the incoming JSON-formatted data using n8n's custom `$json` variable and finds the value of `city` (in this example, \"New York\").",
+ "codeSnippet": "{{$json.body.city}}"
+ },
+ {
+ "role": "assistant",
+ "type": "message",
+ "text": "Did this answer solve your question?",
+ "quickReplies": [
+ {
+ "text": "Yes, thanks",
+ "type": "all-good",
+ "isFeedback": true
+ },
+ {
+ "text": "No, I am still stuck",
+ "type": "still-stuck",
+ "isFeedback": true
+ }
+ ]
+ }
+ ]
+}
diff --git a/cypress/pages/features/ai-assistant.ts b/cypress/pages/features/ai-assistant.ts
index 843407473742a9..ea77724dcf9a4f 100644
--- a/cypress/pages/features/ai-assistant.ts
+++ b/cypress/pages/features/ai-assistant.ts
@@ -37,6 +37,7 @@ export class AIAssistant extends BasePage {
cy.getByTestId('node-error-view-ask-assistant-button').find('button').first(),
credentialEditAssistantButton: () =>
cy.getByTestId('credentail-edit-ask-assistant-button').find('button').first(),
+ codeSnippet: () => cy.getByTestId('assistant-code-snippet'),
};
actions = {
diff --git a/packages/design-system/src/components/AskAssistantChat/AskAssistantChat.vue b/packages/design-system/src/components/AskAssistantChat/AskAssistantChat.vue
index 1fb5d42c63a582..de2f5e17685ea6 100644
--- a/packages/design-system/src/components/AskAssistantChat/AskAssistantChat.vue
+++ b/packages/design-system/src/components/AskAssistantChat/AskAssistantChat.vue
@@ -171,6 +171,7 @@ function growInput() {