Program Link Triggered E-Mail for Next appointment using n8n

For requests or help with our API
Post Reply
joergzastrau
Posts: 23
Joined: Sun Feb 27, 2022 2:53 am

Program Link Triggered E-Mail for Next appointment using n8n

Post by joergzastrau » Sun Nov 06, 2022 10:40 am

Dear all,

this n8n workflow will send an HTML E-Mail to inform the current selected Patient about his next appointment (with iCal attachment).

It is triggered by a webhook, e.g. by creating an Open Dental Program Link to curl.exe with the command line arguments

Code: Select all

"-X POST http://192.168.1.60:5678/webhook-test/EmailAptConfirmation?PatNum=[PatNum]"
(192.168.1.60 being the server hosting the n8n installation).

Requirements:

.) OD version 21.4 or later with RO API access
.) n8n (tested with version 0.195.5, port 5678)
.) curl

We use Docker images for n8n. Setup see http://opendentalsoft.com/forum/viewtop ... f=9&t=7749 (similar).

Code (to be copy&pasted into n8n GUI, adoption of credentials and Server IP necessary):

Code: Select all

{
  "meta": {
    "instanceId": "c5bbe0e4dd0c2a71ee01ba0477f0fe876e4ef1ddb06022d07739fda528d4f9f1"
  },
  "nodes": [
    {
      "parameters": {},
      "name": "Start",
      "type": "n8n-nodes-base.start",
      "position": [
        -460,
        340
      ],
      "typeVersion": 1,
      "id": "e20309fb-c936-416b-8b1a-cfb12e904c3b"
    },
    {
      "parameters": {
        "httpMethod": "POST",
        "path": "EmailAptConfirmation",
        "responseMode": "lastNode",
        "options": {}
      },
      "name": "Webhook",
      "type": "n8n-nodes-base.webhook",
      "position": [
        -260,
        140
      ],
      "webhookId": "db437850-0e90-4eb7-b383-f8438ea1bd66",
      "typeVersion": 1,
      "id": "b3cd36ac-a553-4c10-a5e1-e4fcde5a018f"
    },
    {
      "parameters": {
        "title": "Appointment Dr. Yang",
        "start": "={{$json[\"AptDateTimeUTC\"]}}",
        "end": "={{$json[\"AptDateTimeEndUTC\"]}}",
        "binaryPropertyName": "iCalEventData",
        "additionalFields": {
          "description": "=Appointment with {{$json[\"provAbbr\"]}}\n\nDr. Judith Yang, DMD MS \nOffice Address: Steinmetzstr. 1\n67655 Kaiserslautern\n\nTel: +49(0)631 3437309-0\nFax: +49(0)631 3437309-1\n\nEmail: info@american-orthodontics.com\nWeb: http://www.american-orthodontics.com"
        }
      },
      "id": "83eb5606-d1c1-4bab-bec2-122a3b6d57d7",
      "name": "iCalendar",
      "type": "n8n-nodes-base.iCal",
      "typeVersion": 1,
      "position": [
        1140,
        340
      ]
    },
    {
      "parameters": {
        "fromEmail": "American Orthodontist Dr. Yang <info@american-orthodontics.com>",
        "toEmail": "={{$json[\"Email\"]}}",
        "bccEmail": "info@american-orthodontics.com",
        "subject": "Next appointment with Dr. Yang",
        "html": "=<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">\n<html xmlns=\"http://www.w3.org/1999/xhtml\">\n <head>\n  <meta http-equiv=\"Content-Type\" content=\"text/html; charset=UTF-8\" />\n  <title>Demystifying Email Design</title>\n  <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\"/>\n</head>\n</html>\n\n<body style=\"margin: 0; padding: 0;\">\n<br>\nDear all,<br><br>\nwe have scheduled the next appointment for {{$json[\"FName\"]}} on {{$json[\"AptDateMonth\"]}} {{$json[\"AptDateDate\"]}} at {{$json[\"AptTimeHours\"]}}:{{$json[\"AptTimeMinutes\"]}} (German time). <br>\n<br>\nWith best regards<br>\n<br>\nTeam American Orthodontist<br>\n<br><br>\n--<br>\n<img src=\"\"/>\n<br><br>\nDr. Judith Yang, DMD MS<br>\n Office Address: Steinmetzstr. 1<br>\n 67655 Kaiserslautern<br>\n Tel: +49(0)631 3437309-0<br>\n Fax: +49(0)631 3437309-1<br>\n Email: info@american-orthodontics.com<br>\n Web: http://www.american-orthodontics.com<br><br>\nTax-Reg.: 19/227/50453<br>\n</body>",
        "attachments": "iCalEventData",
        "options": {}
      },
      "id": "3760f640-2e19-425f-b38a-1ed83c64262e",
      "name": "Send Email",
      "type": "n8n-nodes-base.emailSend",
      "typeVersion": 1,
      "position": [
        1540,
        320
      ],
      "credentials": {
        "smtp": {
          "id": "9",
          "name": "SMTP account"
        }
      }
    },
    {
      "parameters": {
        "functionCode": "return [{json:{\n\t\"headers\": {\n\t\t\"host\": \"192.168.1.60:5678\",\n\t\t\"user-agent\": \"curl/7.86.0\",\n\t\t\"accept\": \"*/*\"\n\t},\n\t\"params\": {\n\t},\n\t\"query\": {\n\t\t\"PatNum\": \"11\"\n\t},\n\t\t\"body\": {\n\t}\n  }\n}]\n"
      },
      "id": "c83f7363-b1e6-43ac-a27f-7ecdd96fb0f7",
      "name": "Test Data",
      "type": "n8n-nodes-base.function",
      "typeVersion": 1,
      "position": [
        -260,
        340
      ],
      "disabled": true
    },
    {
      "parameters": {
        "authentication": "headerAuth",
        "url": "=http://192.168.1.60:30223/api/v1/appointments?PatNum={{$json[\"query\"][\"PatNum\"]}}&dateStart={{$json[\"DateToday\"]}}",
        "options": {
          "splitIntoItems": true
        }
      },
      "name": "OD Get Patient appt by ID",
      "type": "n8n-nodes-base.httpRequest",
      "typeVersion": 1,
      "position": [
        140,
        340
      ],
      "alwaysOutputData": true,
      "notesInFlow": true,
      "id": "90d1c110-8942-495d-8f32-fbf9b48edb3f",
      "credentials": {
        "httpHeaderAuth": {
          "id": "5",
          "name": "Header Auth KL Office"
        }
      },
      "notes": "For production use Date_Stop and DateTStamp in query."
    },
    {
      "parameters": {
        "action": "calculate",
        "value": "={{$json[\"AptDateTimeUTC\"]}}",
        "duration": "={{$json[\"Pattern\"].length*5*60}}",
        "timeUnit": "seconds",
        "dataPropertyName": "AptDateTimeEndUTC",
        "options": {
          "fromFormat": ""
        }
      },
      "id": "0e79f2f9-57c3-48af-9a49-a6b4a097cb8e",
      "name": "Calculate AptDateTimeEnd",
      "type": "n8n-nodes-base.dateTime",
      "typeVersion": 1,
      "position": [
        940,
        340
      ]
    },
    {
      "parameters": {
        "value": "={{ new Date(new Date().getTime()+86400000)}}",
        "dataPropertyName": "DateToday",
        "toFormat": "YYYY-MM-DD",
        "options": {}
      },
      "name": "Today",
      "type": "n8n-nodes-base.dateTime",
      "typeVersion": 1,
      "position": [
        -60,
        340
      ],
      "id": "58fb95db-5bd1-4a7b-8030-c3d825d1f336"
    },
    {
      "parameters": {
        "operation": "sort",
        "sortFieldsUi": {
          "sortField": [
            {
              "fieldName": "AptDateTime"
            }
          ]
        },
        "options": {}
      },
      "name": "Sort by AptDateTime",
      "type": "n8n-nodes-base.itemLists",
      "typeVersion": 1,
      "position": [
        340,
        340
      ],
      "id": "fd9ba9ef-11db-43a0-9769-cf7870296b43",
      "continueOnFail": true
    },
    {
      "parameters": {
        "authentication": "headerAuth",
        "url": "=http://192.168.1.60:30223/api/v1/patients/{{$json[\"PatNum\"]}}",
        "options": {
          "splitIntoItems": false
        }
      },
      "name": "OD Get Patient Info by ID",
      "type": "n8n-nodes-base.httpRequest",
      "typeVersion": 1,
      "position": [
        740,
        160
      ],
      "alwaysOutputData": false,
      "id": "ac6d9c8b-4aea-4206-ad4a-7186d383c238",
      "credentials": {
        "httpHeaderAuth": {
          "id": "5",
          "name": "Header Auth KL Office"
        }
      }
    },
    {
      "parameters": {
        "functionCode": "const results = []\nconst date = $json[\"AptDateTime\"];\nconst provAbbr = $json[\"provAbbr\"];\nconst d = new Date(date);\n\nconst month = new Array();\nmonth[0] = \"January\";\nmonth[1] = \"February\";\nmonth[2] = \"March\";\nmonth[3] = \"April\";\nmonth[4] = \"May\";\nmonth[5] = \"June\";\nmonth[6] = \"July\";\nmonth[7] = \"August\";\nmonth[8] = \"September\";\nmonth[9] = \"October\";\nmonth[10] = \"November\";\nmonth[11] = \"December\";\n\nconst AptDateMonth = month[d.getMonth()];\nconst AptDateDate = d.getDate();\nconst AptDateYear = d.getFullYear();\nconst AptTimeHours = d.getHours();\nvar AptTimeMinutes = d.getMinutes();\n\nAptTimeMinutes = d.getMinutes() > 9 ? AptTimeMinutes : '0' + AptTimeMinutes;\n\nresults.push({json:{AptDateMonth, AptDateDate, AptDateYear, AptTimeHours, AptTimeMinutes, provAbbr}})\n\n\nreturn results\n"
      },
      "id": "1db34d1a-9699-40fd-a68b-4d0f6759a4d9",
      "name": "Format Apt Date",
      "type": "n8n-nodes-base.function",
      "typeVersion": 1,
      "position": [
        740,
        0
      ]
    },
    {
      "parameters": {
        "mode": "multiplex",
        "options": {}
      },
      "id": "247d402d-e72c-4521-9c19-5e42376ca9dd",
      "name": "2nd Merge",
      "type": "n8n-nodes-base.merge",
      "typeVersion": 2,
      "position": [
        1340,
        320
      ]
    },
    {
      "parameters": {
        "mode": "multiplex",
        "options": {}
      },
      "id": "e33daa88-951a-49f4-88eb-bef4d40096cc",
      "name": "1st Merge",
      "type": "n8n-nodes-base.merge",
      "typeVersion": 2,
      "position": [
        940,
        80
      ]
    },
    {
      "parameters": {
        "values": {
          "string": [
            {
              "name": "DateTStamp",
              "value": "={{$json[\"DateTStamp\"]}}"
            }
          ]
        },
        "options": {
          "dotNotation": true
        }
      },
      "name": "Select Next Appointment",
      "type": "n8n-nodes-base.set",
      "typeVersion": 1,
      "position": [
        540,
        340
      ],
      "executeOnce": true,
      "id": "fecf72af-1c75-49d6-bc4c-d83b9d1bd9f9"
    },
    {
      "parameters": {
        "value": "={{$json[\"AptDateTime\"]}}",
        "dataPropertyName": "AptDateTimeUTC",
        "custom": true,
        "options": {
          "toTimezone": "UTC"
        }
      },
      "id": "5f7d865b-1c3b-411a-83cd-ceb8ab8a968f",
      "name": "AptDate to UTC",
      "type": "n8n-nodes-base.dateTime",
      "typeVersion": 1,
      "position": [
        740,
        340
      ]
    }
  ],
  "connections": {
    "Start": {
      "main": [
        [
          {
            "node": "Test Data",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Webhook": {
      "main": [
        [
          {
            "node": "Today",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "iCalendar": {
      "main": [
        [
          {
            "node": "2nd Merge",
            "type": "main",
            "index": 1
          }
        ]
      ]
    },
    "Test Data": {
      "main": [
        [
          {
            "node": "Today",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "OD Get Patient appt by ID": {
      "main": [
        [
          {
            "node": "Sort by AptDateTime",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Calculate AptDateTimeEnd": {
      "main": [
        [
          {
            "node": "iCalendar",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Today": {
      "main": [
        [
          {
            "node": "OD Get Patient appt by ID",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Sort by AptDateTime": {
      "main": [
        [
          {
            "node": "Select Next Appointment",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "OD Get Patient Info by ID": {
      "main": [
        [
          {
            "node": "1st Merge",
            "type": "main",
            "index": 1
          }
        ]
      ]
    },
    "Format Apt Date": {
      "main": [
        [
          {
            "node": "1st Merge",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "2nd Merge": {
      "main": [
        [
          {
            "node": "Send Email",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "1st Merge": {
      "main": [
        [
          {
            "node": "2nd Merge",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Select Next Appointment": {
      "main": [
        [
          {
            "node": "AptDate to UTC",
            "type": "main",
            "index": 0
          },
          {
            "node": "Format Apt Date",
            "type": "main",
            "index": 0
          },
          {
            "node": "OD Get Patient Info by ID",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "AptDate to UTC": {
      "main": [
        [
          {
            "node": "Calculate AptDateTimeEnd",
            "type": "main",
            "index": 0
          }
        ]
      ]
    }
  }
}
(Edit: 20221106-07: Corrected getDay() -> getDate(), startDate=Tomorrow)

Please ask.

Joerg

joergzastrau
Posts: 23
Joined: Sun Feb 27, 2022 2:53 am

Re: Program Link Triggered E-Mail for Next appointment using n8n

Post by joergzastrau » Tue Nov 15, 2022 10:21 am

Dear all,

more nodes have been added to the following cleaner version of the n8n workflow in order to post a note to the Open Dental Commlog. This requires API write permissions to the Comm group.

Edit 202211161621 (2x): Added Node to set Confirmation status to "Confirmed".

Joerg

Code: Select all

{
  "meta": {
    "instanceId": "c5bbe0e4dd0c2a71ee01ba0477f0fe876e4ef1ddb06022d07739fda528d4f9f1"
  },
  "nodes": [
    {
      "parameters": {},
      "name": "Start",
      "type": "n8n-nodes-base.start",
      "position": [
        -600,
        120
      ],
      "typeVersion": 1,
      "id": "e20309fb-c936-416b-8b1a-cfb12e904c3b"
    },
    {
      "parameters": {
        "httpMethod": "POST",
        "path": "EmailAptConfirmation",
        "responseMode": "lastNode",
        "options": {}
      },
      "name": "Webhook",
      "type": "n8n-nodes-base.webhook",
      "position": [
        -400,
        340
      ],
      "webhookId": "db437850-0e90-4eb7-b383-f8438ea1bd66",
      "typeVersion": 1,
      "id": "b3cd36ac-a553-4c10-a5e1-e4fcde5a018f",
      "notes": "Setup Program Link in Opendental to trigger.\n\nPath of File to open:\ncurl.exe\n\nOptional command line arguments:\n-X POST http://[n8n-ip]:5678/webhook/EmailAptConfirmation?PatNum=[PatNum]\n"
    },
    {
      "parameters": {
        "title": "Appointment Dr. Yang",
        "start": "={{$json[\"AptDateTimeUTC\"]}}",
        "end": "={{$json[\"AptDateTimeEndUTC\"]}}",
        "binaryPropertyName": "iCalEventData",
        "additionalFields": {
          "description": "=Appointment with {{$json[\"provAbbr\"]}}\n\nDr. Judith Yang, DMD MS \nOffice Address: Steinmetzstr. 1\n67655 Kaiserslautern\n\nTel: +49(0)631 3437309-0\nFax: +49(0)631 3437309-1\n\nEmail: info@american-orthodontics.com\nWeb: http://www.american-orthodontics.com"
        }
      },
      "id": "83eb5606-d1c1-4bab-bec2-122a3b6d57d7",
      "name": "iCalendar",
      "type": "n8n-nodes-base.iCal",
      "typeVersion": 1,
      "position": [
        -200,
        880
      ]
    },
    {
      "parameters": {
        "fromEmail": "American Orthodontist Dr. Yang <info@american-orthodontics.com>",
        "toEmail": "={{$json[\"Email\"]}}",
        "subject": "Next appointment with Dr. Yang",
        "html": "=<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">\n<html xmlns=\"http://www.w3.org/1999/xhtml\">\n <head>\n  <meta http-equiv=\"Content-Type\" content=\"text/html; charset=UTF-8\" />\n  <title>Demystifying Email Design</title>\n  <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\"/>\n</head>\n</html>\n\n<body style=\"margin-left: 10px; padding: 0;\">\n<br>\nDear all,<br><br>\nwe have scheduled the next appointment for {{$json[\"FName\"]}} on {{$json[\"AptDateMonth\"]}} {{$json[\"AptDateDate\"]}} at {{$json[\"AptTimeHours\"]}}:{{$json[\"AptTimeMinutes\"]}}.<br>\n<br>\nWith best regards<br>\n<br>\nTeam American Orthodontist<br>\n<br><br>\n-- <br>\n\n\n<img src=\"\" />\n<br><br>\nDr. Judith Yang, DMD MS<br>\n Office Address: Steinmetzstr. 1<br>\n 67655 Kaiserslautern<br>\n Tel: +49(0)631 3437309-0<br>\n Fax: +49(0)631 3437309-1<br>\n Email: info@american-orthodontics.com<br>\n Web: http://www.american-orthodontics.com<br><br>\nTax-Reg.: 19/227/50453<br>\n<br>\n</body>",
        "attachments": "iCalEventData",
        "options": {}
      },
      "id": "3760f640-2e19-425f-b38a-1ed83c64262e",
      "name": "Send Email",
      "type": "n8n-nodes-base.emailSend",
      "typeVersion": 1,
      "position": [
        200,
        860
      ],
      "credentials": {
        "smtp": {
          "id": "9",
          "name": "SMTP account"
        }
      }
    },
    {
      "parameters": {
        "functionCode": "return [{json:{\n\t\"headers\": {\n\t\t\"host\": \"192.168.1.60:5678\",\n\t\t\"user-agent\": \"curl/7.86.0\",\n\t\t\"accept\": \"*/*\"\n\t},\n\t\"params\": {\n\t},\n\t\"query\": {\n\t\t\"PatNum\": \"11\"\n\t},\n\t\t\"body\": {\n\t}\n  }\n}]\n"
      },
      "id": "c83f7363-b1e6-43ac-a27f-7ecdd96fb0f7",
      "name": "Test Data",
      "type": "n8n-nodes-base.function",
      "typeVersion": 1,
      "position": [
        -400,
        120
      ],
      "disabled": true,
      "notes": "Only Test parameter needed is PatNum"
    },
    {
      "parameters": {
        "authentication": "headerAuth",
        "url": "=http://{{$json[\"OD_API_IP_Port\"]}}/api/v1/appointments",
        "options": {
          "splitIntoItems": true
        },
        "queryParametersUi": {
          "parameter": [
            {
              "name": "PatNum",
              "value": "={{$json[\"query\"][\"PatNum\"]}}"
            },
            {
              "name": "dateStart",
              "value": "={{$json[\"dateStart\"]}}"
            }
          ]
        }
      },
      "name": "OD Get Patient appt by ID",
      "type": "n8n-nodes-base.httpRequest",
      "typeVersion": 1,
      "position": [
        200,
        340
      ],
      "alwaysOutputData": true,
      "notesInFlow": true,
      "id": "90d1c110-8942-495d-8f32-fbf9b48edb3f",
      "credentials": {
        "httpHeaderAuth": {
          "id": "5",
          "name": "Header Auth KL Office"
        }
      }
    },
    {
      "parameters": {
        "action": "calculate",
        "value": "={{$json[\"AptDateTimeUTC\"]}}",
        "duration": "={{$json[\"Pattern\"].length*5}}",
        "timeUnit": "minutes",
        "dataPropertyName": "AptDateTimeEndUTC",
        "options": {
          "fromFormat": ""
        }
      },
      "id": "0e79f2f9-57c3-48af-9a49-a6b4a097cb8e",
      "name": "Calculate AptDateTimeEnd",
      "type": "n8n-nodes-base.dateTime",
      "typeVersion": 1,
      "position": [
        -400,
        880
      ],
      "notes": "Appointment Length = No. of characters in \"Pattern\" times 5 minutes."
    },
    {
      "parameters": {
        "operation": "sort",
        "sortFieldsUi": {
          "sortField": [
            {
              "fieldName": "AptDateTime"
            }
          ]
        },
        "options": {}
      },
      "name": "Sort by AptDateTime",
      "type": "n8n-nodes-base.itemLists",
      "typeVersion": 1,
      "position": [
        400,
        340
      ],
      "id": "fd9ba9ef-11db-43a0-9769-cf7870296b43",
      "continueOnFail": true
    },
    {
      "parameters": {
        "authentication": "headerAuth",
        "url": "=http://{{$node[\"Set OD API IP and Port and Workflow Version\"].json[\"OD_API_IP_Port\"]}}/api/v1/patients/{{$json[\"PatNum\"]}}",
        "options": {
          "splitIntoItems": false
        }
      },
      "name": "OD Get Patient Info by ID",
      "type": "n8n-nodes-base.httpRequest",
      "typeVersion": 1,
      "position": [
        -600,
        700
      ],
      "alwaysOutputData": false,
      "id": "ac6d9c8b-4aea-4206-ad4a-7186d383c238",
      "credentials": {
        "httpHeaderAuth": {
          "id": "5",
          "name": "Header Auth KL Office"
        }
      }
    },
    {
      "parameters": {
        "functionCode": "const results = [];\n\n// keep these\nconst provAbbr = $json[\"provAbbr\"];\nconst AptNum = $json[\"AptNum\"];\n\nconst date = $json[\"AptDateTime\"];\nconst d = new Date(date);\n\nconst month = new Array();\nmonth[0] = \"January\";\nmonth[1] = \"February\";\nmonth[2] = \"March\";\nmonth[3] = \"April\";\nmonth[4] = \"May\";\nmonth[5] = \"June\";\nmonth[6] = \"July\";\nmonth[7] = \"August\";\nmonth[8] = \"September\";\nmonth[9] = \"October\";\nmonth[10] = \"November\";\nmonth[11] = \"December\";\n\nconst AptDateMonth = month[d.getMonth()];\nconst AptDateDate = d.getDate();\nconst AptDateYear = d.getFullYear();\nconst AptTimeHours = d.getHours();\nvar AptTimeMinutes = d.getMinutes();\n\nAptTimeMinutes = d.getMinutes() > 9 ? AptTimeMinutes : '0' + AptTimeMinutes;\n\nresults.push({json:{AptDateMonth, AptDateDate, AptDateYear, AptTimeHours, AptTimeMinutes, provAbbr, AptNum}});\n\nreturn results\n"
      },
      "id": "1db34d1a-9699-40fd-a68b-4d0f6759a4d9",
      "name": "Format Apt Date",
      "type": "n8n-nodes-base.function",
      "typeVersion": 1,
      "position": [
        -600,
        540
      ]
    },
    {
      "parameters": {
        "mode": "multiplex",
        "options": {}
      },
      "id": "247d402d-e72c-4521-9c19-5e42376ca9dd",
      "name": "2nd Merge",
      "type": "n8n-nodes-base.merge",
      "typeVersion": 2,
      "position": [
        0,
        860
      ]
    },
    {
      "parameters": {
        "mode": "multiplex",
        "options": {}
      },
      "id": "e33daa88-951a-49f4-88eb-bef4d40096cc",
      "name": "1st Merge",
      "type": "n8n-nodes-base.merge",
      "typeVersion": 2,
      "position": [
        -400,
        620
      ]
    },
    {
      "parameters": {
        "options": {
          "dotNotation": false
        }
      },
      "name": "Select Next Appointment",
      "type": "n8n-nodes-base.set",
      "typeVersion": 1,
      "position": [
        600,
        340
      ],
      "executeOnce": true,
      "id": "fecf72af-1c75-49d6-bc4c-d83b9d1bd9f9",
      "notes": "Get the topmost item from the list."
    },
    {
      "parameters": {
        "value": "={{$json[\"AptDateTime\"]}}",
        "dataPropertyName": "AptDateTimeUTC",
        "custom": true,
        "options": {
          "toTimezone": "UTC"
        }
      },
      "id": "5f7d865b-1c3b-411a-83cd-ceb8ab8a968f",
      "name": "AptDate to UTC",
      "type": "n8n-nodes-base.dateTime",
      "typeVersion": 1,
      "position": [
        -600,
        880
      ]
    },
    {
      "parameters": {
        "action": "=format",
        "value": "={{ new Date(new Date().getTime()+86400000)}}",
        "dataPropertyName": "dateStart",
        "toFormat": "YYYY-MM-DD",
        "options": {}
      },
      "name": "Tomorrow",
      "type": "n8n-nodes-base.dateTime",
      "typeVersion": 1,
      "position": [
        0,
        340
      ],
      "id": "58fb95db-5bd1-4a7b-8030-c3d825d1f336",
      "notes": "Add 86400000 milliseconds (24 hours) to current time and convert to local TZ in OD date format."
    },
    {
      "parameters": {
        "value": "={{new Date(new Date().getTime())}}",
        "dataPropertyName": "Now",
        "custom": true,
        "toFormat": "YYYY-MM-DD HH:mm:ss",
        "options": {
          "toTimezone": "Europe/Berlin"
        }
      },
      "name": "Now",
      "type": "n8n-nodes-base.dateTime",
      "typeVersion": 1,
      "position": [
        0,
        620
      ],
      "id": "ceae03fb-e608-490d-b535-89d08ed736b5",
      "notes": "OD Times in Local time Zone"
    },
    {
      "parameters": {
        "authentication": "genericCredentialType",
        "genericAuthType": "httpHeaderAuth",
        "requestMethod": "PUT",
        "url": "=http://{{$node[\"Set OD API IP and Port and Workflow Version\"].json[\"OD_API_IP_Port\"]}}/api/v1/appointments/{{$json[\"AptNum\"]}}/Confirm",
        "options": {},
        "bodyParametersUi": {
          "parameter": [
            {
              "name": "confirmVal",
              "value": "Confirmed"
            }
          ]
        }
      },
      "id": "d7f98bb5-3802-4f64-ad49-3f24275d58cc",
      "name": "OD Set Confirmation Status",
      "type": "n8n-nodes-base.httpRequest",
      "typeVersion": 2,
      "position": [
        600,
        640
      ],
      "credentials": {
        "httpHeaderAuth": {
          "id": "5",
          "name": "Header Auth KL Office"
        }
      }
    },
    {
      "parameters": {
        "mode": "multiplex",
        "options": {}
      },
      "id": "f87534a1-dcab-4428-8d26-263ea1149458",
      "name": "3rd Merge",
      "type": "n8n-nodes-base.merge",
      "typeVersion": 2,
      "position": [
        400,
        640
      ]
    },
    {
      "parameters": {
        "authentication": "genericCredentialType",
        "genericAuthType": "httpHeaderAuth",
        "requestMethod": "POST",
        "url": "=http://{{$node[\"Set OD API IP and Port and Workflow Version\"].json[\"OD_API_IP_Port\"]}}/api/v1/commlogs",
        "options": {},
        "bodyParametersUi": {
          "parameter": [
            {
              "name": "PatNum",
              "value": "={{$json[\"PatNum\"]}}"
            },
            {
              "name": "commType",
              "value": "ApptRelated"
            },
            {
              "name": "CommDateTime",
              "value": "={{$json[\"Now\"]}}"
            },
            {
              "name": "Mode_",
              "value": "Email"
            },
            {
              "name": "SentOrReceived",
              "value": "Sent"
            },
            {
              "name": "Note",
              "value": "=Sent E-Mail confirmation for next appointment on {{$json[\"AptDateMonth\"]}} {{$json[\"AptDateDate\"]}} {{$json[\"AptDateYear\"]}} at {{$json[\"AptTimeHours\"]}}:{{$json[\"AptTimeMinutes\"]}}."
            }
          ]
        }
      },
      "id": "5ecdbb2a-598f-426e-86c8-6ec527294f52",
      "name": "OD Post Commlog Entry",
      "type": "n8n-nodes-base.httpRequest",
      "typeVersion": 2,
      "position": [
        600,
        860
      ],
      "credentials": {
        "httpHeaderAuth": {
          "id": "5",
          "name": "Header Auth KL Office"
        }
      }
    },
    {
      "parameters": {
        "values": {
          "string": [
            {
              "name": "OD_API_IP_Port",
              "value": "192.168.1.60:30223"
            },
            {
              "name": "Workflow Version",
              "value": "1.00"
            }
          ]
        },
        "options": {}
      },
      "id": "f76fd18b-d9f0-4e4b-8e34-7a686b9fac4d",
      "name": "Set OD API IP and Port and Workflow Version",
      "type": "n8n-nodes-base.set",
      "typeVersion": 1,
      "position": [
        -200,
        340
      ]
    }
  ],
  "connections": {
    "Start": {
      "main": [
        [
          {
            "node": "Test Data",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Webhook": {
      "main": [
        [
          {
            "node": "Set OD API IP and Port and Workflow Version",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "iCalendar": {
      "main": [
        [
          {
            "node": "2nd Merge",
            "type": "main",
            "index": 1
          }
        ]
      ]
    },
    "Send Email": {
      "main": [
        [
          {
            "node": "3rd Merge",
            "type": "main",
            "index": 1
          }
        ]
      ]
    },
    "Test Data": {
      "main": [
        [
          {
            "node": "Set OD API IP and Port and Workflow Version",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "OD Get Patient appt by ID": {
      "main": [
        [
          {
            "node": "Sort by AptDateTime",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Calculate AptDateTimeEnd": {
      "main": [
        [
          {
            "node": "iCalendar",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Sort by AptDateTime": {
      "main": [
        [
          {
            "node": "Select Next Appointment",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "OD Get Patient Info by ID": {
      "main": [
        [
          {
            "node": "1st Merge",
            "type": "main",
            "index": 1
          }
        ]
      ]
    },
    "Format Apt Date": {
      "main": [
        [
          {
            "node": "1st Merge",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "2nd Merge": {
      "main": [
        [
          {
            "node": "Send Email",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "1st Merge": {
      "main": [
        [
          {
            "node": "2nd Merge",
            "type": "main",
            "index": 0
          },
          {
            "node": "Now",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Select Next Appointment": {
      "main": [
        [
          {
            "node": "AptDate to UTC",
            "type": "main",
            "index": 0
          },
          {
            "node": "Format Apt Date",
            "type": "main",
            "index": 0
          },
          {
            "node": "OD Get Patient Info by ID",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "AptDate to UTC": {
      "main": [
        [
          {
            "node": "Calculate AptDateTimeEnd",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Tomorrow": {
      "main": [
        [
          {
            "node": "OD Get Patient appt by ID",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Now": {
      "main": [
        [
          {
            "node": "3rd Merge",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "3rd Merge": {
      "main": [
        [
          {
            "node": "OD Post Commlog Entry",
            "type": "main",
            "index": 0
          },
          {
            "node": "OD Set Confirmation Status",
            "type": "main",
            "index": 0
          }
        ]
      ]
    },
    "Set OD API IP and Port and Workflow Version": {
      "main": [
        [
          {
            "node": "Tomorrow",
            "type": "main",
            "index": 0
          }
        ]
      ]
    }
  }
}

joergzastrau
Posts: 23
Joined: Sun Feb 27, 2022 2:53 am

Re: Program Link Triggered E-Mail for Next appointment using n8n

Post by joergzastrau » Wed Nov 30, 2022 8:43 am

Dear all,

I have learned to include screenshots. After 1 month in operation I would say that 2/3 of mobile phones users are happy with the ical feature (screenshots on MAC in reverse order).

With best regards

Joerg
Attachments
Result on Mac to add event to local calendar
Result on Mac to add event to local calendar
OD_Appointment_Mac_Calendar_Screenshott.png (108.84 KiB) Viewed 2240 times
E-Mail after pressing the Button in OD
E-Mail after pressing the Button in OD
OD_Appointment_Email.png (66.64 KiB) Viewed 2240 times
Open Dental Trigger (Scrrenshot)
Open Dental Trigger (Scrrenshot)
OD_Appointment_Trigger.png (32.21 KiB) Viewed 2240 times

omar22
Posts: 60
Joined: Mon Dec 05, 2022 2:38 pm

Re: Program Link Triggered E-Mail for Next appointment using n8n

Post by omar22 » Thu Jan 05, 2023 9:58 am

Hi, I have a question regarding creating buttons on Open Dental UI, similar to the one in this post.

I'd like to create a button on the Open Dental task bar that can link to my external software. What is the best most compliant way to approach this?

Thank you!

SLeon
Posts: 476
Joined: Mon Mar 01, 2021 10:00 am

Re: Program Link Triggered E-Mail for Next appointment using n8n

Post by SLeon » Thu Jan 05, 2023 10:15 am

Good morning,

You can create a Program Link to place a button in the UI, like pictured above. Our manual has a lot of details for setup and implementation. I would recommend referencing Program Links for general information and Custom Bridges for creating your own program link.

omar22
Posts: 60
Joined: Mon Dec 05, 2022 2:38 pm

Re: Program Link Triggered E-Mail for Next appointment using n8n

Post by omar22 » Thu Jan 05, 2023 10:19 am

That sounds great, thank you for providing that information.

Jess
Posts: 17
Joined: Fri Jul 10, 2015 3:54 pm

Re: Program Link Triggered E-Mail for Next appointment using n8n

Post by Jess » Wed Sep 06, 2023 9:29 am

SLeon wrote:
Thu Jan 05, 2023 10:15 am
You can create a Program Link to place a button in the UI, like pictured above. Our manual has a lot of details for setup and implementation. I would recommend referencing Program Links for general information and Custom Bridges for creating your own program link.
Are there lists of the "many patient fields" that screen describes as available, for each module, or especially for the Account module? Thanks!

justine
Posts: 169
Joined: Tue Dec 28, 2021 7:59 am

Re: Program Link Triggered E-Mail for Next appointment using n8n

Post by justine » Wed Sep 06, 2023 9:46 am

Jess wrote:
Wed Sep 06, 2023 9:29 am
SLeon wrote:
Thu Jan 05, 2023 10:15 am
You can create a Program Link to place a button in the UI, like pictured above. Our manual has a lot of details for setup and implementation. I would recommend referencing Program Links for general information and Custom Bridges for creating your own program link.
Are there lists of the "many patient fields" that screen describes as available, for each module, or especially for the Account module? Thanks!
Good morning Jess,

Program link description is stored in the program table and the location of the button is stored in the toolbutitem table.

DerekR
Posts: 79
Joined: Wed Aug 31, 2022 1:13 pm

Re: Program Link Triggered E-Mail for Next appointment using n8n

Post by DerekR » Wed Sep 06, 2023 10:28 am

Clicking the "Replacements" button from the Program Link Output File window, will give you a list of all supported replacement fields. Please see https://opendental.com/manual/programli ... tfile.html for more information. Thanks!

Post Reply