如何实现对话框实现Google Calendar集成?

问题描述 投票:0回答:1

我是node.js的新手,我正在尝试使用Dialogflow构建聊天机器人,该对话框可让您计划约会并将约会插入Google日历。无论尝试如何,我似乎一直都在遇到相同的错误(没有针对请求意图的处理程序)。是否有人看到可能出了问题的内容或尝试过本教程并能正常工作?

我尝试过的事情:

  • 更新了package.json中的依赖项
  • 更正了时区(我居住在比利时/欧洲)和timeZoneOffset
  • 将结算帐户关联到Google Cloud Platform中的项目

这是我遵循的教程:

GoogleCalendar Github

This是特定意图“计划约会”的屏幕截图。它会自动转到默认的响应,但应该转到:Ok, let me see if we can fit you in. ${appointmentTimeString} is fine!.此意图中包含一个自定义实体:@AppointmentType,具有2个选项:- 车辆登记-驾照

在意图屏幕截图中,您还看到时间给出了今天的日期(在这种情况下为2020-02-28),而时间应该给出了所请求的约会日期(在这种情况下为2020-03-10)。这可能会导致错误,但是我不知道如何在嵌入式编辑器中解决该问题。

我的Inline编辑器中的index.js(日历ID和服务帐户数据正确,只是从这里取出来了):

 'use strict';

 const functions = require('firebase-functions');
 const {google} = require('googleapis');
 const {WebhookClient} = require('dialogflow-fulfillment');
 // const nodemailer = require('nodemailer');
 // const xoauth2 = require('xoauth2');
 const axios = require('axios');
 
 // Enter your calendar ID below and service account JSON below
 const calendarId = "CALENDAR ID";
 const serviceAccount = {
  "type": "service_account",
  "project_id": "PROJECT ID",
  "private_key_id": "PRIVATE KEY ID",
  "private_key": "PRIVATE KEY",
  "client_email": "CLIENT EMAIL",
  "client_id": "CLIENT ID",
  "auth_uri": "https://accounts.google.com/o/oauth2/auth",
  "token_uri": "https://oauth2.googleapis.com/token",
  "auth_provider_x509_cert_url": "https://www.googleapis.com/oauth2/v1/certs",
  "client_x509_cert_url": "https://www.googleapis.com/robot/v1/metadata/x509/XXXXXXXXXXXXXXX.iam.gserviceaccount.com"
}; // Starts with {"type": "service_account",...
 
 // Set up Google Calendar Service account credentials
 const serviceAccountAuth = new google.auth.JWT({
   email: serviceAccount.client_email,
   key: serviceAccount.private_key,
   scopes: 'https://www.googleapis.com/auth/calendar'
 });
 
 const calendar = google.calendar('v3');
 process.env.DEBUG = 'dialogflow:*'; // enables lib debugging statements
 
 const timeZone = 'Europe/Madrid';
 const timeZoneOffset = '+01:00';
 
 exports.dialogflowFirebaseFulfillment = functions.https.onRequest((request, response) => {
   const agent = new WebhookClient({ request, response });
   const appointment_type = agent.parameters.AppointmentType;
   function makeAppointment (agent) {
     // Calculate appointment start and end datetimes (end = +1hr from start)
     const dateTimeStart = new Date(Date.parse(agent.parameters.date.split('T')[0] + 'T' + agent.parameters.time + timeZoneOffset));
     // const dateTimeStart = new Date(Date.parse(agent.parameters.date.split('T')[0] + 'T' + agent.parameters.time.split('T')[1].split('+')[0]));
     const dateTimeEnd = new Date(new Date(dateTimeStart).setHours(dateTimeStart.getHours() + 1));
     const appointmentTimeString = dateTimeStart.toLocaleString(
       'en-GB',
       { month: 'long', day: 'numeric', hour: 'numeric', timeZone: timeZone}
     );
 
     // Check the availibility of the time, and make an appointment if there is time on the calendar
     return createCalendarEvent(dateTimeStart, dateTimeEnd, appointment_type).then(() => {
       agent.add(`Ok, let me see if we can fit you in. ${appointmentTimeString} is fine!.`);
     }).catch(() => {
       agent.add(`I'm sorry, there are no slots available for ${appointmentTimeString}.`);
     });
   }
 
   let intentMap = new Map();
   intentMap.set('Schedule Appointment', makeAppointment);
   agent.handleRequest(intentMap);
 });
 
 function createCalendarEvent (dateTimeStart, dateTimeEnd, appointment_type) {
   return new Promise((resolve, reject) => {
     calendar.events.list({
       auth: serviceAccountAuth, // List events for time period
       calendarId: calendarId,
       timeMin: dateTimeStart.toISOString(),
       timeMax: dateTimeEnd.toISOString()
     }, (err, calendarResponse) => {
       // Check if there is a event already on the Calendar
       if (err || calendarResponse.data.items.length > 0) {
         reject(err || new Error('Requested time conflicts with another appointment'));
       } else {
         // Create event for the requested time period
         calendar.events.insert({ auth: serviceAccountAuth,
           calendarId: calendarId,
           resource: {summary: appointment_type +' Appointment', description: appointment_type,
             start: {dateTime: dateTimeStart},
             end: {dateTime: dateTimeEnd}}
         }, (err, event) => {
           err ? reject(err) : resolve(event);
         }
         );
       }
     });
   });
 }

我的package.json:

{
  "name": "dialogflowFirebaseFulfillment",
  "description": "Dialogflow fulfillment for the bike shop sample",
  "version": "0.0.1",
  "private": true,
  "license": "Apache Version 2.0",
  "author": "Google Inc.",
  "engines": {
    "node": "6"
  },
  "scripts": {
    "lint": "semistandard --fix \"**/*.js\"",
    "start": "firebase deploy --only functions",
    "deploy": "firebase deploy --only functions"
  },
  "dependencies": {
    "firebase-functions": "^2.0.2",
    "firebase-admin": "^5.13.1",
    "actions-on-google": "^2.2.0", 
    "googleapis": "^27.0.0",
    "dialogflow": "^0.6.0",
    "dialogflow-fulfillment": "^0.5.0",
    "nodemailer": "^4.4.2",
    "apiai": "^4.0.3",
    "xoauth2": "^1.2.0",
    "axios": "^0.19.2"
  }
}

这是我在Google Cloud Platform日志查看器中遇到的错误:Google Cloud Platform error

这是我在Firebase控制台中遇到的错误:Firebase error

node.js google-calendar-api dialogflow dialogflow-fulfillment
1个回答
0
投票

在代理将您的请求路由到正确的功能之前,您似乎正在读取参数。尝试像这样在函数内部删除函数makeAppointment上方的那一行。

   function makeAppointment (agent) {
     // Calculate appointment start and end datetimes (end = +1hr from start)
     const appointment_type = agent.parameters.AppointmentType;
     const dateTimeStart = new Date(Date.parse(agent.parameters.date.split('T')[0] + 'T' + agent.parameters.time + timeZoneOffset));

希望这对您有用,因为其他所有东西似乎都被正确吸收了。

© www.soinside.com 2019 - 2024. All rights reserved.