forked from Wavyzz/dolibarr
Merge pull request #17822 from frederic34/zapier_create_contacts
add more capabilities for zapier
This commit is contained in:
74
dev/examples/zapier/creates/contact.js
Normal file
74
dev/examples/zapier/creates/contact.js
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
/*jshint esversion: 6 */
|
||||||
|
// create a particular contact by name
|
||||||
|
const createContact = async (z, bundle) => {
|
||||||
|
const apiurl = bundle.authData.url + '/api/index.php/contacts';
|
||||||
|
|
||||||
|
const response = await z.request({
|
||||||
|
method: 'POST',
|
||||||
|
url: apiurl,
|
||||||
|
body: {
|
||||||
|
name: bundle.inputData.name,
|
||||||
|
name_alias: bundle.inputData.name_alias,
|
||||||
|
ref_ext: bundle.inputData.ref_ext,
|
||||||
|
ref_int: bundle.inputData.ref_int,
|
||||||
|
address: bundle.inputData.address,
|
||||||
|
zip: bundle.inputData.zip,
|
||||||
|
town: bundle.inputData.town,
|
||||||
|
country_code: bundle.inputData.country_code,
|
||||||
|
country_id: bundle.inputData.country_id,
|
||||||
|
country: bundle.inputData.country,
|
||||||
|
phone: bundle.inputData.phone,
|
||||||
|
email: bundle.inputData.email,
|
||||||
|
sens: 'fromzapier'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const result = z.JSON.parse(response.content);
|
||||||
|
// api returns an integer when ok, a json when ko
|
||||||
|
return result.response || {id: response};
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
key: 'contact',
|
||||||
|
noun: 'Contact',
|
||||||
|
|
||||||
|
display: {
|
||||||
|
label: 'Create Contact',
|
||||||
|
description: 'Creates a contact.'
|
||||||
|
},
|
||||||
|
|
||||||
|
operation: {
|
||||||
|
inputFields: [
|
||||||
|
{key: 'name', required: true},
|
||||||
|
{key: 'name_alias', required: false},
|
||||||
|
{key: 'address', required: false},
|
||||||
|
{key: 'zip', required: false},
|
||||||
|
{key: 'town', required: false},
|
||||||
|
{key: 'email', required: false}
|
||||||
|
],
|
||||||
|
perform: createContact,
|
||||||
|
|
||||||
|
sample: {
|
||||||
|
id: 1,
|
||||||
|
name: 'DUPOND',
|
||||||
|
name_alias: 'DUPOND Ltd',
|
||||||
|
address: 'Rue des Canaries',
|
||||||
|
zip: '34090',
|
||||||
|
town: 'MONTPELLIER',
|
||||||
|
phone: '0123456789',
|
||||||
|
fax: '2345678901',
|
||||||
|
email: 'robot@domain.com'
|
||||||
|
},
|
||||||
|
|
||||||
|
outputFields: [
|
||||||
|
{key: 'id', type: "integer", label: 'ID'},
|
||||||
|
{key: 'name', label: 'Name'},
|
||||||
|
{key: 'name_alias', label: 'Name alias'},
|
||||||
|
{key: 'address', label: 'Address'},
|
||||||
|
{key: 'zip', label: 'Zip'},
|
||||||
|
{key: 'town', label: 'Town'},
|
||||||
|
{key: 'phone', label: 'Phone'},
|
||||||
|
{key: 'fax', label: 'Fax'},
|
||||||
|
{key: 'email', label: 'Email'}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
};
|
||||||
74
dev/examples/zapier/creates/member.js
Normal file
74
dev/examples/zapier/creates/member.js
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
/*jshint esversion: 6 */
|
||||||
|
// create a particular member by name
|
||||||
|
const createMember = async (z, bundle) => {
|
||||||
|
const apiurl = bundle.authData.url + '/api/index.php/members';
|
||||||
|
|
||||||
|
const response = await z.request({
|
||||||
|
method: 'POST',
|
||||||
|
url: apiurl,
|
||||||
|
body: {
|
||||||
|
name: bundle.inputData.name,
|
||||||
|
name_alias: bundle.inputData.name_alias,
|
||||||
|
ref_ext: bundle.inputData.ref_ext,
|
||||||
|
ref_int: bundle.inputData.ref_int,
|
||||||
|
address: bundle.inputData.address,
|
||||||
|
zip: bundle.inputData.zip,
|
||||||
|
town: bundle.inputData.town,
|
||||||
|
country_code: bundle.inputData.country_code,
|
||||||
|
country_id: bundle.inputData.country_id,
|
||||||
|
country: bundle.inputData.country,
|
||||||
|
phone: bundle.inputData.phone,
|
||||||
|
email: bundle.inputData.email,
|
||||||
|
sens: 'fromzapier'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const result = z.JSON.parse(response.content);
|
||||||
|
// api returns an integer when ok, a json when ko
|
||||||
|
return result.response || {id: response};
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
key: 'member',
|
||||||
|
noun: 'Member',
|
||||||
|
|
||||||
|
display: {
|
||||||
|
label: 'Create Member',
|
||||||
|
description: 'Creates a member.'
|
||||||
|
},
|
||||||
|
|
||||||
|
operation: {
|
||||||
|
inputFields: [
|
||||||
|
{key: 'name', required: true},
|
||||||
|
{key: 'name_alias', required: false},
|
||||||
|
{key: 'address', required: false},
|
||||||
|
{key: 'zip', required: false},
|
||||||
|
{key: 'town', required: false},
|
||||||
|
{key: 'email', required: false}
|
||||||
|
],
|
||||||
|
perform: createMember,
|
||||||
|
|
||||||
|
sample: {
|
||||||
|
id: 1,
|
||||||
|
name: 'DUPOND',
|
||||||
|
name_alias: 'DUPOND Ltd',
|
||||||
|
address: 'Rue des Canaries',
|
||||||
|
zip: '34090',
|
||||||
|
town: 'MONTPELLIER',
|
||||||
|
phone: '0123456789',
|
||||||
|
fax: '2345678901',
|
||||||
|
email: 'robot@domain.com'
|
||||||
|
},
|
||||||
|
|
||||||
|
outputFields: [
|
||||||
|
{key: 'id', type: "integer", label: 'ID'},
|
||||||
|
{key: 'name', label: 'Name'},
|
||||||
|
{key: 'name_alias', label: 'Name alias'},
|
||||||
|
{key: 'address', label: 'Address'},
|
||||||
|
{key: 'zip', label: 'Zip'},
|
||||||
|
{key: 'town', label: 'Town'},
|
||||||
|
{key: 'phone', label: 'Phone'},
|
||||||
|
{key: 'fax', label: 'Fax'},
|
||||||
|
{key: 'email', label: 'Email'}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -2,12 +2,18 @@
|
|||||||
const triggerAction = require('./triggers/action');
|
const triggerAction = require('./triggers/action');
|
||||||
const triggerOrder = require('./triggers/order');
|
const triggerOrder = require('./triggers/order');
|
||||||
const triggerThirdparty = require('./triggers/thirdparty');
|
const triggerThirdparty = require('./triggers/thirdparty');
|
||||||
|
const triggerContact = require('./triggers/contact');
|
||||||
const triggerTicket = require('./triggers/ticket');
|
const triggerTicket = require('./triggers/ticket');
|
||||||
const triggerUser = require('./triggers/user');
|
const triggerUser = require('./triggers/user');
|
||||||
|
const triggerMember = require('./triggers/member');
|
||||||
|
|
||||||
const searchThirdparty = require('./searches/thirdparty');
|
const searchThirdparty = require('./searches/thirdparty');
|
||||||
|
const searchContact = require('./searches/contact');
|
||||||
|
const searchMember = require('./searches/member');
|
||||||
|
|
||||||
const createThirdparty = require('./creates/thirdparty');
|
const createThirdparty = require('./creates/thirdparty');
|
||||||
|
const createContact = require('./creates/contact');
|
||||||
|
const createMember = require('./creates/member');
|
||||||
|
|
||||||
const {
|
const {
|
||||||
config: authentication,
|
config: authentication,
|
||||||
@@ -62,18 +68,24 @@ const App = {
|
|||||||
[triggerAction.key]: triggerAction,
|
[triggerAction.key]: triggerAction,
|
||||||
[triggerOrder.key]: triggerOrder,
|
[triggerOrder.key]: triggerOrder,
|
||||||
[triggerThirdparty.key]: triggerThirdparty,
|
[triggerThirdparty.key]: triggerThirdparty,
|
||||||
|
[triggerContact.key]: triggerContact,
|
||||||
[triggerTicket.key]: triggerTicket,
|
[triggerTicket.key]: triggerTicket,
|
||||||
[triggerUser.key]: triggerUser,
|
[triggerUser.key]: triggerUser,
|
||||||
|
[triggerMember.key]: triggerMember,
|
||||||
},
|
},
|
||||||
|
|
||||||
// If you want your searches to show up, you better include it here!
|
// If you want your searches to show up, you better include it here!
|
||||||
searches: {
|
searches: {
|
||||||
[searchThirdparty.key]: searchThirdparty,
|
[searchThirdparty.key]: searchThirdparty,
|
||||||
|
[searchContact.key]: searchContact,
|
||||||
|
[searchMember.key]: searchMember,
|
||||||
},
|
},
|
||||||
|
|
||||||
// If you want your creates to show up, you better include it here!
|
// If you want your creates to show up, you better include it here!
|
||||||
creates: {
|
creates: {
|
||||||
[createThirdparty.key]: createThirdparty,
|
[createThirdparty.key]: createThirdparty,
|
||||||
|
[createContact.key]: createContact,
|
||||||
|
[createMember.key]: createMember,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "dolibarr",
|
"name": "dolibarr",
|
||||||
"version": "1.13.0",
|
"version": "1.14.0",
|
||||||
"description": "An app for connecting Dolibarr to the Zapier platform.",
|
"description": "An app for connecting Dolibarr to the Zapier platform.",
|
||||||
"repository": "Dolibarr/dolibarr",
|
"repository": "Dolibarr/dolibarr",
|
||||||
"homepage": "https://www.dolibarr.org/",
|
"homepage": "https://www.dolibarr.org/",
|
||||||
@@ -11,7 +11,7 @@
|
|||||||
"test": "mocha --recursive"
|
"test": "mocha --recursive"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": "8.10.0",
|
"node": "14.0.0",
|
||||||
"npm": ">=5.6.0"
|
"npm": ">=5.6.0"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
|||||||
95
dev/examples/zapier/searches/contact.js
Normal file
95
dev/examples/zapier/searches/contact.js
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
module.exports = {
|
||||||
|
key: 'contact',
|
||||||
|
|
||||||
|
// You'll want to provide some helpful display labels and descriptions
|
||||||
|
// for users. Zapier will put them into the UX.
|
||||||
|
noun: 'Contact',
|
||||||
|
display: {
|
||||||
|
label: 'Find a Contact',
|
||||||
|
description: 'Search for contact.'
|
||||||
|
},
|
||||||
|
|
||||||
|
// `operation` is where we make the call to your API to do the search
|
||||||
|
operation: {
|
||||||
|
// This search only has one search field. Your searches might have just one, or many
|
||||||
|
// search fields.
|
||||||
|
inputFields: [
|
||||||
|
{
|
||||||
|
key: 'lastname',
|
||||||
|
type: 'string',
|
||||||
|
label: 'Lastname',
|
||||||
|
helpText: 'Lastname to limit to the search to (i.e. The company or %company%).'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'email',
|
||||||
|
type: 'string',
|
||||||
|
label: 'Email',
|
||||||
|
helpText: 'Email to limit to the search to.'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
perform: async (z, bundle) => {
|
||||||
|
const url = bundle.authData.url + '/api/index.php/contacts/';
|
||||||
|
|
||||||
|
// Put the search value in a query param. The details of how to build
|
||||||
|
// a search URL will depend on how your API works.
|
||||||
|
let filter = '';
|
||||||
|
if (bundle.inputData.lastname) {
|
||||||
|
filter = "t.lastname like \'%"+bundle.inputData.name+"%\'";
|
||||||
|
}
|
||||||
|
if (bundle.inputData.email) {
|
||||||
|
if (bundle.inputData.lastname) {
|
||||||
|
filter += " and ";
|
||||||
|
}
|
||||||
|
filter += "t.email like \'"+bundle.inputData.email+"\'";
|
||||||
|
}
|
||||||
|
const response = await z.request({
|
||||||
|
url: url,
|
||||||
|
// this parameter avoid throwing errors and let us manage them
|
||||||
|
skipThrowForStatus: true,
|
||||||
|
params: {
|
||||||
|
sqlfilters: filter
|
||||||
|
}
|
||||||
|
});
|
||||||
|
//z.console.log(response);
|
||||||
|
if (response.status != 200) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
return response.json;
|
||||||
|
},
|
||||||
|
|
||||||
|
// In cases where Zapier needs to show an example record to the user, but we are unable to get a live example
|
||||||
|
// from the API, Zapier will fallback to this hard-coded sample. It should reflect the data structure of
|
||||||
|
// returned records, and have obviously dummy values that we can show to any user.
|
||||||
|
sample: {
|
||||||
|
id: 1,
|
||||||
|
createdAt: 1472069465,
|
||||||
|
name: 'DOE',
|
||||||
|
firstname: 'John',
|
||||||
|
authorId: 1,
|
||||||
|
directions: '1. Boil Noodles\n2.Serve with sauce',
|
||||||
|
style: 'italian'
|
||||||
|
},
|
||||||
|
|
||||||
|
// If the resource can have fields that are custom on a per-user basis, define a function to fetch the custom
|
||||||
|
// field definitions. The result will be used to augment the sample.
|
||||||
|
// outputFields: () => { return []; }
|
||||||
|
// Alternatively, a static field definition should be provided, to specify labels for the fields
|
||||||
|
outputFields: [
|
||||||
|
{
|
||||||
|
key: 'id',
|
||||||
|
type: "integer",
|
||||||
|
label: 'ID'
|
||||||
|
},
|
||||||
|
{key: 'createdAt', type: "integer", label: 'Created At'},
|
||||||
|
{key: 'name', label: 'Name'},
|
||||||
|
{key: 'firstname', label: 'Firstname'},
|
||||||
|
{key: 'directions', label: 'Directions'},
|
||||||
|
{key: 'authorId', type: "integer", label: 'Author ID'},
|
||||||
|
{
|
||||||
|
key: 'style',
|
||||||
|
label: 'Style'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
};
|
||||||
88
dev/examples/zapier/searches/member.js
Normal file
88
dev/examples/zapier/searches/member.js
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
module.exports = {
|
||||||
|
key: 'member',
|
||||||
|
|
||||||
|
// You'll want to provide some helpful display labels and descriptions
|
||||||
|
// for users. Zapier will put them into the UX.
|
||||||
|
noun: 'Member',
|
||||||
|
display: {
|
||||||
|
label: 'Find a Member',
|
||||||
|
description: 'Search for member.'
|
||||||
|
},
|
||||||
|
|
||||||
|
// `operation` is where we make the call to your API to do the search
|
||||||
|
operation: {
|
||||||
|
// This search only has one search field. Your searches might have just one, or many
|
||||||
|
// search fields.
|
||||||
|
inputFields: [
|
||||||
|
{
|
||||||
|
key: 'lastname',
|
||||||
|
type: 'string',
|
||||||
|
label: 'Lastname',
|
||||||
|
helpText: 'Lastname to limit to the search to (i.e. The company or %company%).'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'email',
|
||||||
|
type: 'string',
|
||||||
|
label: 'Email',
|
||||||
|
helpText: 'Email to limit to the search to.'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
perform: async (z, bundle) => {
|
||||||
|
const url = bundle.authData.url + '/api/index.php/members/';
|
||||||
|
|
||||||
|
// Put the search value in a query param. The details of how to build
|
||||||
|
// a search URL will depend on how your API works.
|
||||||
|
let filter = '';
|
||||||
|
if (bundle.inputData.lastname) {
|
||||||
|
filter = "t.lastname like \'%" + bundle.inputData.name + "%\'";
|
||||||
|
}
|
||||||
|
if (bundle.inputData.email) {
|
||||||
|
if (bundle.inputData.lastname) {
|
||||||
|
filter += " and ";
|
||||||
|
}
|
||||||
|
filter += "t.email like \'" + bundle.inputData.email + "\'";
|
||||||
|
}
|
||||||
|
const response = await z.request({
|
||||||
|
url: url,
|
||||||
|
// this parameter avoid throwing errors and let us manage them
|
||||||
|
skipThrowForStatus: true,
|
||||||
|
params: {
|
||||||
|
sqlfilters: filter
|
||||||
|
}
|
||||||
|
});
|
||||||
|
//z.console.log(response);
|
||||||
|
if (response.status != 200) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
return response.json;
|
||||||
|
},
|
||||||
|
|
||||||
|
// In cases where Zapier needs to show an example record to the user, but we are unable to get a live example
|
||||||
|
// from the API, Zapier will fallback to this hard-coded sample. It should reflect the data structure of
|
||||||
|
// returned records, and have obviously dummy values that we can show to any user.
|
||||||
|
sample: {
|
||||||
|
id: 1,
|
||||||
|
createdAt: 1472069465,
|
||||||
|
name: 'DOE',
|
||||||
|
firstname: 'John',
|
||||||
|
authorId: 1,
|
||||||
|
},
|
||||||
|
|
||||||
|
// If the resource can have fields that are custom on a per-user basis, define a function to fetch the custom
|
||||||
|
// field definitions. The result will be used to augment the sample.
|
||||||
|
// outputFields: () => { return []; }
|
||||||
|
// Alternatively, a static field definition should be provided, to specify labels for the fields
|
||||||
|
outputFields: [
|
||||||
|
{
|
||||||
|
key: 'id',
|
||||||
|
type: "integer",
|
||||||
|
label: 'ID'
|
||||||
|
},
|
||||||
|
{ key: 'createdAt', type: "integer", label: 'Created At' },
|
||||||
|
{ key: 'name', label: 'Name' },
|
||||||
|
{ key: 'firstname', label: 'Firstname' },
|
||||||
|
{ key: 'authorId', type: "integer", label: 'Author ID' },
|
||||||
|
]
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -19,21 +19,43 @@ module.exports = {
|
|||||||
type: 'string',
|
type: 'string',
|
||||||
label: 'Name',
|
label: 'Name',
|
||||||
helpText: 'Name to limit to the search to (i.e. The company or %company%).'
|
helpText: 'Name to limit to the search to (i.e. The company or %company%).'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'email',
|
||||||
|
type: 'string',
|
||||||
|
label: 'Email',
|
||||||
|
helpText: 'Email to limit to the search to.'
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
|
||||||
perform: (z, bundle) => {
|
perform: async (z, bundle) => {
|
||||||
const url = bundle.authData.url + '/api/index.php/thirdparties/';
|
const url = bundle.authData.url + '/api/index.php/thirdparties/';
|
||||||
|
|
||||||
// Put the search value in a query param. The details of how to build
|
// Put the search value in a query param. The details of how to build
|
||||||
// a search URL will depend on how your API works.
|
// a search URL will depend on how your API works.
|
||||||
const options = {
|
let filter = '';
|
||||||
params: {
|
if (bundle.inputData.name) {
|
||||||
sqlfilters: "t.nom like \'%"+bundle.inputData.name+"%\'"
|
filter = "t.nom like \'%"+bundle.inputData.name+"%\'";
|
||||||
}
|
}
|
||||||
};
|
if (bundle.inputData.email) {
|
||||||
|
if (bundle.inputData.name) {
|
||||||
return z.request(url, options).then(response => JSON.parse(response.content));
|
filter += " and ";
|
||||||
|
}
|
||||||
|
filter += "t.email like \'"+bundle.inputData.email+"\'";
|
||||||
|
}
|
||||||
|
const response = await z.request({
|
||||||
|
url: url,
|
||||||
|
// this parameter avoid throwing errors and let us manage them
|
||||||
|
skipThrowForStatus: true,
|
||||||
|
params: {
|
||||||
|
sqlfilters: filter
|
||||||
|
}
|
||||||
|
});
|
||||||
|
//z.console.log(response);
|
||||||
|
if (response.status != 200) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
return response.json;
|
||||||
},
|
},
|
||||||
|
|
||||||
// In cases where Zapier needs to show an example record to the user, but we are unable to get a live example
|
// In cases where Zapier needs to show an example record to the user, but we are unable to get a live example
|
||||||
|
|||||||
171
dev/examples/zapier/triggers/contact.js
Normal file
171
dev/examples/zapier/triggers/contact.js
Normal file
@@ -0,0 +1,171 @@
|
|||||||
|
const subscribeHook = (z, bundle) => {
|
||||||
|
// `z.console.log()` is similar to `console.log()`.
|
||||||
|
z.console.log('suscribing hook!');
|
||||||
|
|
||||||
|
// bundle.targetUrl has the Hook URL this app should call when an action is created.
|
||||||
|
const data = {
|
||||||
|
url: bundle.targetUrl,
|
||||||
|
event: bundle.event,
|
||||||
|
module: 'contact',
|
||||||
|
action: bundle.inputData.action
|
||||||
|
};
|
||||||
|
|
||||||
|
const url = bundle.authData.url + '/api/index.php/zapierapi/hook';
|
||||||
|
|
||||||
|
// You can build requests and our client will helpfully inject all the variables
|
||||||
|
// you need to complete. You can also register middleware to control this.
|
||||||
|
const options = {
|
||||||
|
url: url,
|
||||||
|
method: 'POST',
|
||||||
|
body: data,
|
||||||
|
};
|
||||||
|
|
||||||
|
// You may return a promise or a normal data structure from any perform method.
|
||||||
|
return z.request(options).then((response) => JSON.parse(response.content));
|
||||||
|
};
|
||||||
|
|
||||||
|
const unsubscribeHook = (z, bundle) => {
|
||||||
|
// bundle.subscribeData contains the parsed response JSON from the subscribe
|
||||||
|
// request made initially.
|
||||||
|
z.console.log('unsuscribing hook!');
|
||||||
|
|
||||||
|
// You can build requests and our client will helpfully inject all the variables
|
||||||
|
// you need to complete. You can also register middleware to control this.
|
||||||
|
const options = {
|
||||||
|
url: bundle.authData.url + '/api/index.php/zapierapi/hook/' + bundle.subscribeData.id,
|
||||||
|
method: 'DELETE',
|
||||||
|
};
|
||||||
|
|
||||||
|
// You may return a promise or a normal data structure from any perform method.
|
||||||
|
return z.request(options).then((response) => JSON.parse(response.content));
|
||||||
|
};
|
||||||
|
|
||||||
|
const getContact = (z, bundle) => {
|
||||||
|
// bundle.cleanedRequest will include the parsed JSON object (if it's not a
|
||||||
|
// test poll) and also a .querystring property with the URL's query string.
|
||||||
|
const contact = {
|
||||||
|
id: bundle.cleanedRequest.id,
|
||||||
|
name: bundle.cleanedRequest.name,
|
||||||
|
name_alias: bundle.cleanedRequest.name_alias,
|
||||||
|
firstname: bundle.cleanedRequest.firstname,
|
||||||
|
address: bundle.cleanedRequest.address,
|
||||||
|
zip: bundle.cleanedRequest.zip,
|
||||||
|
town: bundle.cleanedRequest.town,
|
||||||
|
email: bundle.cleanedRequest.email,
|
||||||
|
phone_pro: bundle.cleanedRequest.phone_pro,
|
||||||
|
phone_perso: bundle.cleanedRequest.phone_perso,
|
||||||
|
phone_mobile: bundle.cleanedRequest.phone_mobile,
|
||||||
|
authorId: bundle.cleanedRequest.authorId,
|
||||||
|
createdAt: bundle.cleanedRequest.createdAt,
|
||||||
|
action: bundle.cleanedRequest.action
|
||||||
|
};
|
||||||
|
|
||||||
|
return [contact];
|
||||||
|
};
|
||||||
|
|
||||||
|
const getFallbackRealContact = (z, bundle) => {
|
||||||
|
// For the test poll, you should get some real data, to aid the setup process.
|
||||||
|
const module = bundle.inputData.module;
|
||||||
|
const options = {
|
||||||
|
url: bundle.authData.url + '/api/index.php/contacts/0',
|
||||||
|
};
|
||||||
|
|
||||||
|
return z.request(options).then((response) => [JSON.parse(response.content)]);
|
||||||
|
};
|
||||||
|
|
||||||
|
// const getModulesChoices = (z/*, bundle*/) => {
|
||||||
|
// // For the test poll, you should get some real data, to aid the setup process.
|
||||||
|
// const options = {
|
||||||
|
// url: bundle.authData.url + '/api/index.php/zapierapi/getmoduleschoices',
|
||||||
|
// };
|
||||||
|
|
||||||
|
// return z.request(options).then((response) => JSON.parse(response.content));
|
||||||
|
// };
|
||||||
|
// const getModulesChoices = () => {
|
||||||
|
// return {
|
||||||
|
// orders: "Order",
|
||||||
|
// invoices: "Invoice",
|
||||||
|
// contacts: "Contact",
|
||||||
|
// contacts: "Contacts"
|
||||||
|
// };
|
||||||
|
// };
|
||||||
|
|
||||||
|
// const getActionsChoices = (z, bundle) => {
|
||||||
|
// // For the test poll, you should get some real data, to aid the setup process.
|
||||||
|
// const module = bundle.inputData.module;
|
||||||
|
// const options = {
|
||||||
|
// url: url: bundle.authData.url + '/api/index.php/zapierapi/getactionschoices/thirparty`,
|
||||||
|
// };
|
||||||
|
|
||||||
|
// return z.request(options).then((response) => JSON.parse(response.content));
|
||||||
|
// };
|
||||||
|
|
||||||
|
// We recommend writing your triggers separate like this and rolling them
|
||||||
|
// into the App definition at the end.
|
||||||
|
module.exports = {
|
||||||
|
key: 'contact',
|
||||||
|
|
||||||
|
// You'll want to provide some helpful display labels and descriptions
|
||||||
|
// for users. Zapier will put them into the UX.
|
||||||
|
noun: 'Contact',
|
||||||
|
display: {
|
||||||
|
label: 'New Contact',
|
||||||
|
description: 'Triggers when a new contact action is done in Dolibarr.'
|
||||||
|
},
|
||||||
|
|
||||||
|
// `operation` is where the business logic goes.
|
||||||
|
operation: {
|
||||||
|
|
||||||
|
// `inputFields` can define the fields a user could provide,
|
||||||
|
// we'll pass them in as `bundle.inputData` later.
|
||||||
|
inputFields: [
|
||||||
|
{
|
||||||
|
key: 'action',
|
||||||
|
required: true,
|
||||||
|
type: 'string',
|
||||||
|
helpText: 'Which action of contact this should trigger on.',
|
||||||
|
choices: {
|
||||||
|
create: "Create",
|
||||||
|
modify: "Modify",
|
||||||
|
validate: "Validate",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
type: 'hook',
|
||||||
|
|
||||||
|
performSubscribe: subscribeHook,
|
||||||
|
performUnsubscribe: unsubscribeHook,
|
||||||
|
|
||||||
|
perform: getContact,
|
||||||
|
performList: getFallbackRealContact,
|
||||||
|
|
||||||
|
// In cases where Zapier needs to show an example record to the user, but we are unable to get a live example
|
||||||
|
// from the API, Zapier will fallback to this hard-coded sample. It should reflect the data structure of
|
||||||
|
// returned records, and have obviously dummy values that we can show to any user.
|
||||||
|
sample: {
|
||||||
|
id: 1,
|
||||||
|
createdAt: 1472069465,
|
||||||
|
lastname: 'DOE',
|
||||||
|
firstname: 'John',
|
||||||
|
authorId: 1,
|
||||||
|
action: 'create'
|
||||||
|
},
|
||||||
|
|
||||||
|
// If the resource can have fields that are custom on a per-user basis, define a function to fetch the custom
|
||||||
|
// field definitions. The result will be used to augment the sample.
|
||||||
|
// outputFields: () => { return []; }
|
||||||
|
// Alternatively, a static field definition should be provided, to specify labels for the fields
|
||||||
|
outputFields: [
|
||||||
|
{key: 'id', type: "integer", label: 'ID'},
|
||||||
|
{key: 'createdAt', label: 'Created At'},
|
||||||
|
{key: 'lastname', label: 'Lastname'},
|
||||||
|
{key: 'firstname', label: 'Firstname'},
|
||||||
|
{key: 'phone', label: 'Phone pro'},
|
||||||
|
{key: 'phone_perso', label: 'Phone perso'},
|
||||||
|
{key: 'phone_mobile', label: 'Phone mobile'},
|
||||||
|
{key: 'authorId', type: "integer", label: 'Author ID'},
|
||||||
|
{key: 'action', label: 'Action'}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
};
|
||||||
171
dev/examples/zapier/triggers/member.js
Normal file
171
dev/examples/zapier/triggers/member.js
Normal file
@@ -0,0 +1,171 @@
|
|||||||
|
const subscribeHook = (z, bundle) => {
|
||||||
|
// `z.console.log()` is similar to `console.log()`.
|
||||||
|
z.console.log('suscribing hook!');
|
||||||
|
|
||||||
|
// bundle.targetUrl has the Hook URL this app should call when an action is created.
|
||||||
|
const data = {
|
||||||
|
url: bundle.targetUrl,
|
||||||
|
event: bundle.event,
|
||||||
|
module: 'member',
|
||||||
|
action: bundle.inputData.action
|
||||||
|
};
|
||||||
|
|
||||||
|
const url = bundle.authData.url + '/api/index.php/zapierapi/hook';
|
||||||
|
|
||||||
|
// You can build requests and our client will helpfully inject all the variables
|
||||||
|
// you need to complete. You can also register middleware to control this.
|
||||||
|
const options = {
|
||||||
|
url: url,
|
||||||
|
method: 'POST',
|
||||||
|
body: data,
|
||||||
|
};
|
||||||
|
|
||||||
|
// You may return a promise or a normal data structure from any perform method.
|
||||||
|
return z.request(options).then((response) => JSON.parse(response.content));
|
||||||
|
};
|
||||||
|
|
||||||
|
const unsubscribeHook = (z, bundle) => {
|
||||||
|
// bundle.subscribeData contains the parsed response JSON from the subscribe
|
||||||
|
// request made initially.
|
||||||
|
z.console.log('unsuscribing hook!');
|
||||||
|
|
||||||
|
// You can build requests and our client will helpfully inject all the variables
|
||||||
|
// you need to complete. You can also register middleware to control this.
|
||||||
|
const options = {
|
||||||
|
url: bundle.authData.url + '/api/index.php/zapierapi/hook/' + bundle.subscribeData.id,
|
||||||
|
method: 'DELETE',
|
||||||
|
};
|
||||||
|
|
||||||
|
// You may return a promise or a normal data structure from any perform method.
|
||||||
|
return z.request(options).then((response) => JSON.parse(response.content));
|
||||||
|
};
|
||||||
|
|
||||||
|
const getMember = (z, bundle) => {
|
||||||
|
// bundle.cleanedRequest will include the parsed JSON object (if it's not a
|
||||||
|
// test poll) and also a .querystring property with the URL's query string.
|
||||||
|
const member = {
|
||||||
|
id: bundle.cleanedRequest.id,
|
||||||
|
name: bundle.cleanedRequest.name,
|
||||||
|
name_alias: bundle.cleanedRequest.name_alias,
|
||||||
|
firstname: bundle.cleanedRequest.firstname,
|
||||||
|
address: bundle.cleanedRequest.address,
|
||||||
|
zip: bundle.cleanedRequest.zip,
|
||||||
|
town: bundle.cleanedRequest.town,
|
||||||
|
email: bundle.cleanedRequest.email,
|
||||||
|
phone_pro: bundle.cleanedRequest.phone_pro,
|
||||||
|
phone_perso: bundle.cleanedRequest.phone_perso,
|
||||||
|
phone_mobile: bundle.cleanedRequest.phone_mobile,
|
||||||
|
authorId: bundle.cleanedRequest.authorId,
|
||||||
|
createdAt: bundle.cleanedRequest.createdAt,
|
||||||
|
action: bundle.cleanedRequest.action
|
||||||
|
};
|
||||||
|
|
||||||
|
return [member];
|
||||||
|
};
|
||||||
|
|
||||||
|
const getFallbackRealMember = (z, bundle) => {
|
||||||
|
// For the test poll, you should get some real data, to aid the setup process.
|
||||||
|
const module = bundle.inputData.module;
|
||||||
|
const options = {
|
||||||
|
url: bundle.authData.url + '/api/index.php/members/0',
|
||||||
|
};
|
||||||
|
|
||||||
|
return z.request(options).then((response) => [JSON.parse(response.content)]);
|
||||||
|
};
|
||||||
|
|
||||||
|
// const getModulesChoices = (z/*, bundle*/) => {
|
||||||
|
// // For the test poll, you should get some real data, to aid the setup process.
|
||||||
|
// const options = {
|
||||||
|
// url: bundle.authData.url + '/api/index.php/zapierapi/getmoduleschoices',
|
||||||
|
// };
|
||||||
|
|
||||||
|
// return z.request(options).then((response) => JSON.parse(response.content));
|
||||||
|
// };
|
||||||
|
// const getModulesChoices = () => {
|
||||||
|
// return {
|
||||||
|
// orders: "Order",
|
||||||
|
// invoices: "Invoice",
|
||||||
|
// members: "Member",
|
||||||
|
// members: "Members"
|
||||||
|
// };
|
||||||
|
// };
|
||||||
|
|
||||||
|
// const getActionsChoices = (z, bundle) => {
|
||||||
|
// // For the test poll, you should get some real data, to aid the setup process.
|
||||||
|
// const module = bundle.inputData.module;
|
||||||
|
// const options = {
|
||||||
|
// url: url: bundle.authData.url + '/api/index.php/zapierapi/getactionschoices/thirparty`,
|
||||||
|
// };
|
||||||
|
|
||||||
|
// return z.request(options).then((response) => JSON.parse(response.content));
|
||||||
|
// };
|
||||||
|
|
||||||
|
// We recommend writing your triggers separate like this and rolling them
|
||||||
|
// into the App definition at the end.
|
||||||
|
module.exports = {
|
||||||
|
key: 'member',
|
||||||
|
|
||||||
|
// You'll want to provide some helpful display labels and descriptions
|
||||||
|
// for users. Zapier will put them into the UX.
|
||||||
|
noun: 'Member',
|
||||||
|
display: {
|
||||||
|
label: 'New Member',
|
||||||
|
description: 'Triggers when a new member action is done in Dolibarr.'
|
||||||
|
},
|
||||||
|
|
||||||
|
// `operation` is where the business logic goes.
|
||||||
|
operation: {
|
||||||
|
|
||||||
|
// `inputFields` can define the fields a user could provide,
|
||||||
|
// we'll pass them in as `bundle.inputData` later.
|
||||||
|
inputFields: [
|
||||||
|
{
|
||||||
|
key: 'action',
|
||||||
|
required: true,
|
||||||
|
type: 'string',
|
||||||
|
helpText: 'Which action of member this should trigger on.',
|
||||||
|
choices: {
|
||||||
|
create: "Create",
|
||||||
|
modify: "Modify",
|
||||||
|
validate: "Validate",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
type: 'hook',
|
||||||
|
|
||||||
|
performSubscribe: subscribeHook,
|
||||||
|
performUnsubscribe: unsubscribeHook,
|
||||||
|
|
||||||
|
perform: getMember,
|
||||||
|
performList: getFallbackRealMember,
|
||||||
|
|
||||||
|
// In cases where Zapier needs to show an example record to the user, but we are unable to get a live example
|
||||||
|
// from the API, Zapier will fallback to this hard-coded sample. It should reflect the data structure of
|
||||||
|
// returned records, and have obviously dummy values that we can show to any user.
|
||||||
|
sample: {
|
||||||
|
id: 1,
|
||||||
|
createdAt: 1472069465,
|
||||||
|
lastname: 'DOE',
|
||||||
|
firstname: 'John',
|
||||||
|
authorId: 1,
|
||||||
|
action: 'create'
|
||||||
|
},
|
||||||
|
|
||||||
|
// If the resource can have fields that are custom on a per-user basis, define a function to fetch the custom
|
||||||
|
// field definitions. The result will be used to augment the sample.
|
||||||
|
// outputFields: () => { return []; }
|
||||||
|
// Alternatively, a static field definition should be provided, to specify labels for the fields
|
||||||
|
outputFields: [
|
||||||
|
{key: 'id', type: "integer", label: 'ID'},
|
||||||
|
{key: 'createdAt', label: 'Created At'},
|
||||||
|
{key: 'lastname', label: 'Lastname'},
|
||||||
|
{key: 'firstname', label: 'Firstname'},
|
||||||
|
{key: 'phone', label: 'Phone pro'},
|
||||||
|
{key: 'phone_perso', label: 'Phone perso'},
|
||||||
|
{key: 'phone_mobile', label: 'Phone mobile'},
|
||||||
|
{key: 'authorId', type: "integer", label: 'Author ID'},
|
||||||
|
{key: 'action', label: 'Action'}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -118,7 +118,7 @@ module.exports = {
|
|||||||
noun: 'Thirdparty',
|
noun: 'Thirdparty',
|
||||||
display: {
|
display: {
|
||||||
label: 'New Thirdparty',
|
label: 'New Thirdparty',
|
||||||
description: 'Triggers when a new thirdpaty action is done in Dolibarr.'
|
description: 'Triggers when a new thirdparty action is done in Dolibarr.'
|
||||||
},
|
},
|
||||||
|
|
||||||
// `operation` is where the business logic goes.
|
// `operation` is where the business logic goes.
|
||||||
|
|||||||
@@ -1043,7 +1043,7 @@ function get_next_value($db, $mask, $table, $field, $where = '', $objsoc = '', $
|
|||||||
$regType = array();
|
$regType = array();
|
||||||
if (preg_match('/\{(t+)\}/i', $mask, $regType)) {
|
if (preg_match('/\{(t+)\}/i', $mask, $regType)) {
|
||||||
$masktype = $regType[1];
|
$masktype = $regType[1];
|
||||||
$masktype_value = substr(preg_replace('/^TE_/', '', $objsoc->typent_code), 0, dol_strlen($regType[1])); // get n first characters of thirdpaty typent_code (where n is length in mask)
|
$masktype_value = substr(preg_replace('/^TE_/', '', $objsoc->typent_code), 0, dol_strlen($regType[1])); // get n first characters of thirdparty typent_code (where n is length in mask)
|
||||||
$masktype_value = str_pad($masktype_value, dol_strlen($regType[1]), "#", STR_PAD_RIGHT); // we fill on right with # to have same number of char than into mask
|
$masktype_value = str_pad($masktype_value, dol_strlen($regType[1]), "#", STR_PAD_RIGHT); // we fill on right with # to have same number of char than into mask
|
||||||
} else {
|
} else {
|
||||||
$masktype = '';
|
$masktype = '';
|
||||||
|
|||||||
@@ -174,8 +174,27 @@ class InterfaceZapierTriggers extends DolibarrTriggers
|
|||||||
|
|
||||||
// Contacts
|
// Contacts
|
||||||
case 'CONTACT_CREATE':
|
case 'CONTACT_CREATE':
|
||||||
|
$resql = $this->db->query($sql);
|
||||||
|
while ($resql && $obj = $this->db->fetch_array($resql)) {
|
||||||
|
$cleaned = cleanObjectDatas(dol_clone($object));
|
||||||
|
$json = json_encode($cleaned);
|
||||||
|
// call the zapierPostWebhook() function
|
||||||
|
zapierPostWebhook($obj['url'], $json);
|
||||||
|
}
|
||||||
|
$logtriggeraction = true;
|
||||||
|
break;
|
||||||
case 'CONTACT_MODIFY':
|
case 'CONTACT_MODIFY':
|
||||||
|
$resql = $this->db->query($sql);
|
||||||
|
while ($resql && $obj = $this->db->fetch_array($resql)) {
|
||||||
|
$cleaned = cleanObjectDatas(dol_clone($object));
|
||||||
|
$json = json_encode($cleaned);
|
||||||
|
// call the zapierPostWebhook() function
|
||||||
|
zapierPostWebhook($obj['url'], $json);
|
||||||
|
}
|
||||||
|
$logtriggeraction = true;
|
||||||
|
break;
|
||||||
case 'CONTACT_DELETE':
|
case 'CONTACT_DELETE':
|
||||||
|
break;
|
||||||
case 'CONTACT_ENABLEDISABLE':
|
case 'CONTACT_ENABLEDISABLE':
|
||||||
break;
|
break;
|
||||||
// Products
|
// Products
|
||||||
@@ -320,10 +339,28 @@ class InterfaceZapierTriggers extends DolibarrTriggers
|
|||||||
// case 'LINEFICHINTER_DELETE':
|
// case 'LINEFICHINTER_DELETE':
|
||||||
|
|
||||||
// Members
|
// Members
|
||||||
// case 'MEMBER_CREATE':
|
case 'MEMBER_CREATE':
|
||||||
|
$resql = $this->db->query($sql);
|
||||||
|
while ($resql && $obj = $this->db->fetch_array($resql)) {
|
||||||
|
$cleaned = cleanObjectDatas(dol_clone($object));
|
||||||
|
$json = json_encode($cleaned);
|
||||||
|
// call the zapierPostWebhook() function
|
||||||
|
zapierPostWebhook($obj['url'], $json);
|
||||||
|
}
|
||||||
|
$logtriggeraction = true;
|
||||||
|
break;
|
||||||
|
case 'MEMBER_MODIFY':
|
||||||
|
$resql = $this->db->query($sql);
|
||||||
|
while ($resql && $obj = $this->db->fetch_array($resql)) {
|
||||||
|
$cleaned = cleanObjectDatas(dol_clone($object));
|
||||||
|
$json = json_encode($cleaned);
|
||||||
|
// call the zapierPostWebhook() function
|
||||||
|
zapierPostWebhook($obj['url'], $json);
|
||||||
|
}
|
||||||
|
$logtriggeraction = true;
|
||||||
|
break;
|
||||||
// case 'MEMBER_VALIDATE':
|
// case 'MEMBER_VALIDATE':
|
||||||
// case 'MEMBER_SUBSCRIPTION':
|
// case 'MEMBER_SUBSCRIPTION':
|
||||||
// case 'MEMBER_MODIFY':
|
|
||||||
// case 'MEMBER_NEW_PASSWORD':
|
// case 'MEMBER_NEW_PASSWORD':
|
||||||
// case 'MEMBER_RESILIATE':
|
// case 'MEMBER_RESILIATE':
|
||||||
// case 'MEMBER_DELETE':
|
// case 'MEMBER_DELETE':
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ class Contacts extends DolibarrApi
|
|||||||
throw new RestException(401, 'No permission to read contacts');
|
throw new RestException(401, 'No permission to read contacts');
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($id == 0) {
|
if ($id === 0) {
|
||||||
$result = $this->contact->initAsSpecimen();
|
$result = $this->contact->initAsSpecimen();
|
||||||
} else {
|
} else {
|
||||||
$result = $this->contact->fetch($id);
|
$result = $this->contact->fetch($id);
|
||||||
|
|||||||
Reference in New Issue
Block a user