Skip to content

Commit

Permalink
Merge pull request #350 from cofacts/google-auth
Browse files Browse the repository at this point in the history
Use Google cloud's Application Default Credentials
  • Loading branch information
MrOrz authored May 3, 2023
2 parents 76f4c6e + b0c2a8e commit 194fb8a
Show file tree
Hide file tree
Showing 6 changed files with 52 additions and 126 deletions.
6 changes: 3 additions & 3 deletions .env.sample
Original file line number Diff line number Diff line change
Expand Up @@ -87,10 +87,10 @@ REVIEW_REPLY_BUFFER={"seconds":0,"minutes":0,"hours":-12,"days":0}
# LINE add friend url, used for the message to guide user from LINE Notify to cofacts
LINE_FRIEND_URL=https://line.me/R/ti/p/@cofacts

# Google service
GOOGLE_APPLICATION_CREDENTIALS=<path-to-service-account-credential>

# Dialogflow
DAILOGFLOW_CLIENT_EMAIL=
DAILOGFLOW_PRIVATE_KEY=
DAILOGFLOW_PROJECT_ID=
DAILOGFLOW_LANGUAGE=
DAILOGFLOW_ENV=

Expand Down
28 changes: 14 additions & 14 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -204,25 +204,25 @@ $ npm run notify
$ node build/scripts/scanRepliesAndNotify.js
```
### Dialogflow
### Google cloud services
We use dialogflow to detect if user is chatting with bot.
If userinput matches one of dialogflow intents, we can directly return predefined responses in that intent.
rumors-line-bot uses Google cloud services that is authenticated and authorized using Google Cloud
service accounts and [Application Default Credentials](https://cloud.google.com/docs/authentication/application-default-credentials).
To use Dialogflow,
Please create a service account under the project, download its key and use `GOOGLE_APPLICATION_CREDENTIALS` env var to
provide the path to your downloaded service account key. See [documentation](https://cloud.google.com/docs/authentication/provide-credentials-adc#local-key) for detail.
1. You should [create a project](https://cloud.google.com/dialogflow/es/docs/quick/setup#project) and take note of the project ID then [enable api](https://cloud.google.com/dialogflow/es/docs/quick/setup#api).
2. [Build an agent](https://cloud.google.com/dialogflow/es/docs/quick/build-agent).
3. You will get a JSON file after [seting up authentication](https://cloud.google.com/dialogflow/es/docs/quick/setup#auth), copy `client_email` and `private_key` in the file.
4. sets in `.env` file
#### Dialogflow
```
DAILOGFLOW_CLIENT_EMAIL=<paste client_email get from step3 here>
DAILOGFLOW_PRIVATE_KEY=<paste private_key get from step3 here>
DAILOGFLOW_PROJECT_ID=<paste project_id get from step1 here>
```
We use Dialogflow to detect if user is trying to chit-chat.
If user input matches any of the Dialogflow intents, we can directly return predefined responses in that intent.
Optional env variables:
To use Dialogflow, please do the following setup:
1. Please ensure your [GCP project](https://cloud.google.com/dialogflow/es/docs/quick/setup#project) has [enabled Dialogflow api](https://cloud.google.com/dialogflow/es/docs/quick/setup#api).
2. [Build an agent](https://cloud.google.com/dialogflow/es/docs/quick/build-agent) connected to the GCP project.
3. Please ensure the service account has `dialogflow.sessions.detectIntent` permission.
4. Set these env variables (optional):
- `DAILOGFLOW_LANGUAGE` : Empty to agent's default language, or you can specify a [language](https://cloud.google.com/dialogflow/es/docs/reference/language).
- `DAILOGFLOW_ENV` : Default to draft agent, or you can create different [versions](https://cloud.google.com/dialogflow/es/docs/agents-versions).
Expand Down
8 changes: 7 additions & 1 deletion babel.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,13 @@ module.exports = {
],
plugins: [
['ttag', { resolve: { translations: `i18n/${locale}.po` } }],
['module-resolver', { root: ['./'] }],
[
'module-resolver',
{
root: ['./'],
extensions: ['.js', '.ts'],
},
],
'@babel/plugin-proposal-class-properties',
],
};
82 changes: 0 additions & 82 deletions scripts/authGoogleDrive.js

This file was deleted.

30 changes: 17 additions & 13 deletions src/lib/__tests__/detectDialogflowIntent.js
Original file line number Diff line number Diff line change
@@ -1,33 +1,27 @@
const mockSessionPath = jest.fn();
const mockDetectIntent = jest.fn();
const mockGetProjectId = jest.fn();

let detectDialogflowIntent;
beforeEach(() => {
// following variables are just for `detectDialogflowIntent` env varialbe check
process.env.DAILOGFLOW_PROJECT_ID = 'projectId';
process.env.DAILOGFLOW_CLIENT_EMAIL = 'client_email';
process.env.DAILOGFLOW_PRIVATE_KEY = 'private_key';
const sleep = async (ms) => new Promise((resolve) => setTimeout(resolve, ms));

beforeEach(() => {
mockSessionPath.mockClear();
mockDetectIntent.mockClear();
mockGetProjectId.mockClear();

// To test different case of process.env.* in `detectDialogflowIntent` we should resetModules
// To test different case in `detectDialogflowIntent` we should resetModules
//
jest.resetModules();
jest.mock('@google-cloud/dialogflow', () => ({
SessionsClient: jest.fn().mockImplementation(() => ({
projectAgentEnvironmentUserSessionPath: mockSessionPath,
detectIntent: mockDetectIntent,
getProjectId: mockGetProjectId,
})),
}));
});

afterEach(() => {
delete process.env.DAILOGFLOW_PROJECT_ID;
delete process.env.DAILOGFLOW_CLIENT_EMAIL;
delete process.env.DAILOGFLOW_PRIVATE_KEY;
});

const intentResponse = [
{
responseId: 'response_Id',
Expand Down Expand Up @@ -80,16 +74,23 @@ const intentResponse = [
];

it('skip detecting intent', async () => {
delete process.env.DAILOGFLOW_CLIENT_EMAIL;
mockGetProjectId.mockImplementation(() =>
Promise.reject('Test the case when Google service account is not provided')
);
detectDialogflowIntent = require('../detectDialogflowIntent').default;
await sleep(1); // Wait for module initialization (project ID detection)

mockDetectIntent.mockImplementation(() => intentResponse);
expect(await detectDialogflowIntent('Hi')).toMatchInlineSnapshot(`undefined`);
expect(mockSessionPath).not.toHaveBeenCalled();
expect(mockDetectIntent).not.toHaveBeenCalled();
});

it('detects intent', async () => {
mockGetProjectId.mockImplementation(() => Promise.resolve('test-gcp-id'));
detectDialogflowIntent = require('../detectDialogflowIntent').default;
await sleep(1); // Wait for module initialization (project ID detection)

mockDetectIntent.mockImplementation(() => intentResponse);
expect(await detectDialogflowIntent('Hi')).toMatchInlineSnapshot(`
Object {
Expand Down Expand Up @@ -155,7 +156,10 @@ it('detects intent', async () => {
});

it('handles error', async () => {
mockGetProjectId.mockImplementation(() => Promise.resolve('test-gcp-id'));
detectDialogflowIntent = require('../detectDialogflowIntent').default;
await sleep(1); // Wait for module initialization (project ID detection)

mockDetectIntent.mockImplementation(() => {
const error = new Error(
`3 INVALID_ARGUMENT: Resource name 'projects/undefined_project_id/agent/sessions/sessionId/' does not match 'projects/*/locations/*/agent/environments/*/users/*/sessions/*'.`
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,21 +2,19 @@
import dialogflow from '@google-cloud/dialogflow';
import crypto from 'crypto';

const projectId = process.env.DAILOGFLOW_PROJECT_ID;
const credentials = {
client_email: process.env.DAILOGFLOW_CLIENT_EMAIL,
// https://stackoverflow.com/questions/39492587/escaping-issue-with-firebase-privatekey-as-a-heroku-config-variable/41044630#41044630
private_key: (process.env.DAILOGFLOW_PRIVATE_KEY || '').replace(/\\n/g, '\n'),
};

// https://googleapis.dev/nodejs/dialogflow/latest/v2beta1.SessionsClient.html
const sessionClient = new dialogflow.SessionsClient({ credentials });
const sessionClient = new dialogflow.SessionsClient(/* { credentials } */);
let projectId: string | null = null;
sessionClient
.getProjectId()
.then((id) => {
projectId = id;
console.log(`[Dialogflow] Connected to project ID = ${id}`);
})
.catch((e) => console.error('[Dialogflow]', e));

export default async function (input) {
if (!projectId || !credentials.client_email || !credentials.private_key) {
console.log(
'[Dialogflow] Skip detecting intent, one of env variables not set.'
);
export default async function (input: string) {
if (!projectId) {
return;
}
// https://cloud.google.com/dialogflow/es/docs/api-overview#sessions
Expand Down

0 comments on commit 194fb8a

Please sign in to comment.