dolibarr 24.0.0-beta
api_contacts.class.php
1<?php
2/* Copyright (C) 2015 Jean-François Ferry <jfefe@aternatik.fr>
3 * Copyright (C) 2019-2025 Frédéric France <frederic.france@free.fr>
4 * Copyright (C) 2024 MDW <mdeweerd@users.noreply.github.com>
5 * Copyright (C) 2025 William Mead <william@m34d.com>
6 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 3 of the License, or
10 * (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program. If not, see <https://www.gnu.org/licenses/>.
19 */
20
21use Luracast\Restler\RestException;
22
23//require_once DOL_DOCUMENT_ROOT.'/contact/class/contact.class.php';
24//require_once DOL_DOCUMENT_ROOT.'/categories/class/categorie.class.php';
25require_once DOL_DOCUMENT_ROOT.'/societe/class/societe.class.php';
26
27
36class Contacts extends DolibarrApi
37{
42 public static $FIELDS = array(
43 'lastname',
44 );
45
49 public $contact;
50
54 public function __construct()
55 {
56 global $db, $conf;
57 $this->db = $db;
58
59 require_once DOL_DOCUMENT_ROOT.'/contact/class/contact.class.php';
60 require_once DOL_DOCUMENT_ROOT.'/categories/class/categorie.class.php';
61
62 $this->contact = new Contact($this->db);
63 }
64
79 public function get($id, $includecount = 0, $includeroles = 0)
80 {
81 if (!DolibarrApiAccess::$user->hasRight('societe', 'contact', 'lire')) {
82 throw new RestException(403, 'No permission to read contacts');
83 }
84
85 if ($id === 0) {
86 $result = $this->contact->initAsSpecimen();
87 } else {
88 $result = $this->contact->fetch($id);
89 }
90
91 if (!$result) {
92 throw new RestException(404, 'Contact not found');
93 }
94
95 if (!DolibarrApi::_checkAccessToResource('contact', $this->contact->id, 'socpeople&societe')) {
96 throw new RestException(403, 'Access not allowed for login '.DolibarrApiAccess::$user->login);
97 }
98
99 if ($includecount) {
100 $this->contact->load_ref_elements();
101 }
102
103 if ($includeroles) {
104 $this->contact->fetchRoles();
105 }
106
107 if (isModEnabled('mailing')) {
108 $this->contact->getNoEmail();
109 }
110
111 return $this->_cleanObjectDatas($this->contact);
112 }
113
129 public function getByEmail($email, $includecount = 0, $includeroles = 0)
130 {
131 if (!DolibarrApiAccess::$user->hasRight('societe', 'contact', 'lire')) {
132 throw new RestException(403, 'No permission to read contacts');
133 }
134
135 if (empty($email)) {
136 $result = $this->contact->initAsSpecimen();
137 } else {
138 $result = $this->contact->fetch(0, null, '', $email);
139 }
140
141 if (!$result) {
142 throw new RestException(404, 'Contact not found');
143 }
144
145 if (!DolibarrApi::_checkAccessToResource('contact', $this->contact->id, 'socpeople&societe')) {
146 throw new RestException(403, 'Access not allowed for login '.DolibarrApiAccess::$user->login);
147 }
148
149 if ($includecount) {
150 $this->contact->load_ref_elements();
151 }
152
153 if ($includeroles) {
154 $this->contact->fetchRoles();
155 }
156
157 if (isModEnabled('mailing')) {
158 $this->contact->getNoEmail();
159 }
160
161 return $this->_cleanObjectDatas($this->contact);
162 }
163
186 public function index($sortfield = "t.rowid", $sortorder = 'ASC', $limit = 100, $page = 0, $thirdparty_ids = '', $category = 0, $sqlfilters = '', $includecount = 0, $includeroles = 0, $properties = '', $pagination_data = false)
187 {
188 global $db, $conf;
189
190 $obj_ret = array();
191
192 if (!DolibarrApiAccess::$user->hasRight('societe', 'contact', 'lire')) {
193 throw new RestException(403, 'No permission to read contacts');
194 }
195
196 // case of external user, $thirdparty_ids param is ignored and replaced by user's socid
197 $socids = DolibarrApiAccess::$user->socid ?: $thirdparty_ids;
198
199 // If the internal user must only see his customers, force searching by him
200 $search_sale = 0;
201 if (!DolibarrApiAccess::$user->hasRight('societe', 'client', 'voir') && !$socids) {
202 $search_sale = DolibarrApiAccess::$user->id;
203 }
204
205 $sql = "SELECT t.rowid";
206 $sql .= " FROM ".MAIN_DB_PREFIX."socpeople as t";
207 $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."socpeople_extrafields as te ON te.fk_object = t.rowid";
208 $sql .= " LEFT JOIN ".MAIN_DB_PREFIX."societe as s ON t.fk_soc = s.rowid";
209 $sql .= ' WHERE t.entity IN ('.getEntity('contact').')';
210 if ($socids) {
211 $sql .= " AND t.fk_soc IN (".$this->db->sanitize($socids).")";
212 }
213 // Search on sale representative
214 if ($search_sale && $search_sale != '-1') {
215 if ($search_sale == -2) {
216 $sql .= " AND NOT EXISTS (SELECT sc.fk_soc FROM ".MAIN_DB_PREFIX."societe_commerciaux as sc WHERE sc.fk_soc = t.fk_soc)";
217 } elseif ($search_sale > 0) {
218 $sql .= " AND EXISTS (SELECT sc.fk_soc FROM ".MAIN_DB_PREFIX."societe_commerciaux as sc WHERE sc.fk_soc = t.fk_soc AND sc.fk_user = ".((int) $search_sale).")";
219 }
220 }
221 // Select contacts of given category
222 if ($category > 0) {
223 // Search Contact Categories
224 $searchCategoryContactList = $category ? array($category) : array();
225 // $searchCategoryContactOperator = 0;
226 // Search for tag/category ($searchCategoryContactList is an array of ID)
227 if (!empty($searchCategoryContactList)) {
228 $searchCategoryContactSqlList = array();
229 // $listofcategoryid = '';
230 foreach ($searchCategoryContactList as $searchCategoryContact) {
231 if (intval($searchCategoryContact) == -2) {
232 $searchCategoryContactSqlList[] = "NOT EXISTS (SELECT ck.fk_socpeople FROM ".MAIN_DB_PREFIX."categorie_contact as ck WHERE t.rowid = ck.fk_socpeople)";
233 } elseif (intval($searchCategoryContact) > 0) {
234 // if ($searchCategoryContactOperator == 0) {
235 $searchCategoryContactSqlList[] = " EXISTS (SELECT ck.fk_socpeople FROM ".MAIN_DB_PREFIX."categorie_contact as ck WHERE t.rowid = ck.fk_socpeople AND ck.fk_categorie = ".((int) $searchCategoryContact).")";
236 // } else {
237 // $listofcategoryid .= ($listofcategoryid ? ', ' : '') .((int) $searchCategoryContact);
238 // }
239 }
240 }
241 // if ($listofcategoryid) {
242 // $searchCategoryContactSqlList[] = " EXISTS (SELECT ck.fk_socpeople FROM ".MAIN_DB_PREFIX."categorie_contact as ck WHERE t.rowid = ck.fk_socpeople AND ck.fk_categorie IN (".$this->db->sanitize($listofcategoryid)."))";
243 // }
244 // if ($searchCategoryContactOperator == 1) {
245 // if (!empty($searchCategoryContactSqlList)) {
246 // $sql .= " AND (".implode(' OR ', $searchCategoryContactSqlList).")";
247 // }
248 // } else {
249 if (!empty($searchCategoryContactSqlList)) {
250 $sql .= " AND (".implode(' AND ', $searchCategoryContactSqlList).")";
251 }
252 // }
253 }
254 }
255
256 // Add sql filters
257 if ($sqlfilters) {
258 $errormessage = '';
259 $sql .= forgeSQLFromUniversalSearchCriteria($sqlfilters, $errormessage);
260 if ($errormessage) {
261 throw new RestException(400, 'Error when validating parameter sqlfilters -> '.$errormessage);
262 }
263 }
264
265 //this query will return total orders with the filters given
266 $sqlTotals = str_replace('SELECT t.rowid', 'SELECT count(t.rowid) as total', $sql);
267
268 $sql .= $this->db->order($sortfield, $sortorder);
269
270 if ($limit) {
271 if ($page < 0) {
272 $page = 0;
273 }
274 $offset = $limit * $page;
275
276 $sql .= $this->db->plimit($limit + 1, $offset);
277 }
278 $result = $this->db->query($sql);
279 if ($result) {
280 $num = $this->db->num_rows($result);
281 $min = min($num, ($limit <= 0 ? $num : $limit));
282 $i = 0;
283 while ($i < $min) {
284 $obj = $this->db->fetch_object($result);
285 $contact_static = new Contact($this->db);
286 if ($contact_static->fetch($obj->rowid)) {
287 $contact_static->fetchRoles();
288 if ($includecount) {
289 $contact_static->load_ref_elements();
290 }
291 if ($includeroles) {
292 $contact_static->fetchRoles();
293 }
294 if (isModEnabled('mailing')) {
295 $contact_static->getNoEmail();
296 }
297
298 $obj_ret[] = $this->_filterObjectProperties($this->_cleanObjectDatas($contact_static), $properties);
299 }
300
301 $i++;
302 }
303 } else {
304 throw new RestException(503, 'Error when retrieve contacts : '.$sql);
305 }
306
307 //if $pagination_data is true the response will contain element data with all values and element pagination with pagination data(total,page,limit)
308 if ($pagination_data) {
309 $totalsResult = $this->db->query($sqlTotals);
310 $total = $this->db->fetch_object($totalsResult)->total;
311
312 $tmp = $obj_ret;
313 $obj_ret = [];
314
315 $obj_ret['data'] = $tmp;
316 $obj_ret['pagination'] = [
317 'total' => (int) $total,
318 'page' => $page, //count starts from 0
319 'page_count' => ceil((int) $total / $limit),
320 'limit' => $limit
321 ];
322 }
323
324 return $obj_ret;
325 }
326
339 public function post($request_data = null)
340 {
341 if (!DolibarrApiAccess::$user->hasRight('societe', 'contact', 'creer')) {
342 throw new RestException(403, 'No permission to create/update contacts');
343 }
344 // Check mandatory fields
345 $result = $this->_validate($request_data);
346
347 // External api user does not know internal country ID
348 if (!isset($request_data['country_id']) && isset($request_data['country_code'])) {
349 $field = strlen($request_data['country_code']) > 2 ? 'code_iso' : 'code';
350 $id = dol_getIdFromCode($this->db, $request_data['country_code'], "c_country", $field, "rowid");
351 if ($id < 0) {
352 throw new RestException(404, 'Country code not found in database: ' . $this->db->error);
353 }
354 $request_data['country_id'] = $id;
355 }
356
357 foreach ($request_data as $field => $value) {
358 if ($field === 'caller') {
359 // Add a mention of caller so on trigger called after action, we can filter to avoid a loop if we try to sync back again with the caller
360 $this->contact->context['caller'] = sanitizeVal($request_data['caller'], 'aZ09');
361 continue;
362 }
363 if ($field == 'array_options' && is_array($value)) {
364 $this->contact->fetch_optionals(); // To force the load of the extrafields definition by fetch_name_optionals_label()
365
366 foreach ($value as $index => $val) {
367 $this->contact->array_options[$index] = $this->_checkValExtrafieldsForAPI($index, $val, $this->contact);
368 }
369 continue;
370 }
371 if ($field == 'socid') {
372 $new_socid = (int) $value;
373 $loopthirdpartytmp = new Societe($this->db);
374 $new_thirdparty_result = $loopthirdpartytmp->fetch($new_socid);
375 if ($new_thirdparty_result < 1) {
376 throw new RestException(404, 'Thirdparty with id='.$new_socid.' not found or not allowed');
377 }
378 if (!DolibarrApi::_checkAccessToResource('societe', $new_socid)) {
379 throw new RestException(403, 'Access to socid/thirdparty='.$new_socid.' is not allowed for login '.DolibarrApiAccess::$user->login);
380 }
381 }
382
383 $this->contact->$field = $this->_checkValForAPI($field, $value, $this->contact);
384 }
385 if ($this->contact->create(DolibarrApiAccess::$user) < 0) {
386 throw new RestException(500, "Error creating contact", array_merge(array($this->contact->error), $this->contact->errors));
387 }
388 if (isModEnabled('mailing') && !empty($this->contact->email) && isset($this->contact->no_email)) {
389 $this->contact->setNoEmail($this->contact->no_email);
390 }
391 return $this->contact->id;
392 }
393
409 public function put($id, $request_data = null)
410 {
411 if (!DolibarrApiAccess::$user->hasRight('societe', 'contact', 'creer')) {
412 throw new RestException(403, 'No permission to create/update contacts');
413 }
414
415 $result = $this->contact->fetch($id);
416 if (!$result) {
417 throw new RestException(404, 'Contact not found');
418 }
419
420 if (!DolibarrApi::_checkAccessToResource('contact', $this->contact->id, 'socpeople&societe')) {
421 throw new RestException(403, 'Access not allowed for login '.DolibarrApiAccess::$user->login);
422 }
423
424 foreach ($request_data as $field => $value) {
425 if ($field == 'id') {
426 continue;
427 }
428 if ($field === 'caller') {
429 // Add a mention of caller so on trigger called after action, we can filter to avoid a loop if we try to sync back again with the caller
430 $this->contact->context['caller'] = sanitizeVal($request_data['caller'], 'aZ09');
431 continue;
432 }
433 if ($field == 'array_options' && is_array($value)) {
434 foreach ($value as $index => $val) {
435 $this->contact->array_options[$index] = $this->_checkValExtrafieldsForAPI($index, $val, $this->contact);
436 }
437 continue;
438 }
439 if ($field == 'socid') {
440 $new_socid = (int) $value;
441 $loopthirdpartytmp = new Societe($this->db);
442 $new_thirdparty_result = $loopthirdpartytmp->fetch($new_socid);
443 if ($new_thirdparty_result < 1) {
444 throw new RestException(404, 'Thirdparty with id='.$new_socid.' not found or not allowed');
445 }
446 if (!DolibarrApi::_checkAccessToResource('societe', $new_socid)) {
447 throw new RestException(403, 'Access to socid/thirdparty='.$new_socid.' is not allowed for login '.DolibarrApiAccess::$user->login);
448 }
449 }
450
451 $this->contact->$field = $this->_checkValForAPI($field, $value, $this->contact);
452 }
453
454 if (isModEnabled('mailing') && !empty($this->contact->email) && isset($this->contact->no_email)) {
455 $this->contact->setNoEmail($this->contact->no_email);
456 }
457
458 if ($this->contact->update($id, DolibarrApiAccess::$user, 0, 'update') > 0) {
459 return $this->get($id);
460 } else {
461 throw new RestException(500, $this->contact->error);
462 }
463 }
464
475 public function delete($id)
476 {
477 if (!DolibarrApiAccess::$user->hasRight('societe', 'contact', 'supprimer')) {
478 throw new RestException(403, 'No permission to delete contacts');
479 }
480 $result = $this->contact->fetch($id);
481 if (!$result) {
482 throw new RestException(404, 'Contact not found');
483 }
484
485 if (!DolibarrApi::_checkAccessToResource('contact', $this->contact->id, 'socpeople&societe')) {
486 throw new RestException(403, 'Access not allowed for login '.DolibarrApiAccess::$user->login);
487 }
488 $this->contact->oldcopy = clone $this->contact; // @phan-suppress-current-line PhanTypeMismatchProperty
489
490 if ($this->contact->delete(DolibarrApiAccess::$user) <= 0) {
491 throw new RestException(500, 'Error when delete contact ' . $this->contact->error);
492 }
493
494 return array(
495 'success' => array(
496 'code' => 200,
497 'message' => 'Contact deleted'
498 )
499 );
500 }
501
516 public function createUser($id, $request_data = null)
517 {
518 //if (!DolibarrApiAccess::$user->hasRight('user', 'user', 'creer')) {
519 //throw new RestException(403);
520 //}
521
522 if (!isset($request_data["login"])) {
523 throw new RestException(400, "login field missing");
524 }
525 if (!isset($request_data["password"])) {
526 throw new RestException(400, "password field missing");
527 }
528
529 if (!DolibarrApiAccess::$user->hasRight('societe', 'contact', 'lire')) {
530 throw new RestException(403, 'No permission to read contacts');
531 }
532 if (!DolibarrApiAccess::$user->hasRight('user', 'user', 'creer')) {
533 throw new RestException(403, 'No permission to create user');
534 }
535
536 $contact = new Contact($this->db);
537 $contact->fetch($id);
538 if ($contact->id <= 0) {
539 throw new RestException(404, 'Contact not found');
540 }
541
542 if (!DolibarrApi::_checkAccessToResource('contact', $contact->id, 'socpeople&societe')) {
543 throw new RestException(403, 'Access not allowed for login '.DolibarrApiAccess::$user->login);
544 }
545
546 // Check mandatory fields
547 $login = $request_data["login"];
548 $password = $request_data["password"];
549 $useraccount = new User($this->db);
550 $result = $useraccount->create_from_contact($contact, $login, $password);
551 if ($result <= 0) {
552 throw new RestException(500, "User not created");
553 }
554 // password parameter not used in create_from_contact
555 $useraccount->setPassword($useraccount, $password);
556
557 return $result;
558 }
559
575 public function getCategories($id, $sortfield = "s.rowid", $sortorder = 'ASC', $limit = 0, $page = 0)
576 {
577 if (!DolibarrApiAccess::$user->hasRight('categorie', 'lire')) {
578 throw new RestException(403);
579 }
580
581 $categories = new Categorie($this->db);
582
583 $result = $categories->getListForItem($id, 'contact', $sortfield, $sortorder, $limit, $page);
584
585 if ($result < 0) {
586 throw new RestException(503, 'Error when retrieve category list : '.$categories->error);
587 }
588
589 return $result;
590 }
591
607 public function addCategory($id, $category_id)
608 {
609 if (!DolibarrApiAccess::$user->hasRight('societe', 'contact', 'creer')) {
610 throw new RestException(403, 'Insufficient rights');
611 }
612
613 $result = $this->contact->fetch($id);
614 if (!$result) {
615 throw new RestException(404, 'Contact not found');
616 }
617 $category = new Categorie($this->db);
618 $result = $category->fetch($category_id);
619 if (!$result) {
620 throw new RestException(404, 'category not found');
621 }
622
623 if (!DolibarrApi::_checkAccessToResource('contact', $this->contact->id)) {
624 throw new RestException(403, 'Access not allowed for login '.DolibarrApiAccess::$user->login);
625 }
626 if (!DolibarrApi::_checkAccessToResource('category', $category->id)) {
627 throw new RestException(403, 'Access not allowed for login '.DolibarrApiAccess::$user->login);
628 }
629
630 $category->add_type($this->contact, 'contact');
631
632 return $this->_cleanObjectDatas($this->contact);
633 }
634
649 public function deleteCategory($id, $category_id)
650 {
651 if (!DolibarrApiAccess::$user->hasRight('societe', 'contact', 'creer')) {
652 throw new RestException(403, 'Insufficient rights');
653 }
654
655 $result = $this->contact->fetch($id);
656 if (!$result) {
657 throw new RestException(404, 'Contact not found');
658 }
659 $category = new Categorie($this->db);
660 $result = $category->fetch($category_id);
661 if (!$result) {
662 throw new RestException(404, 'category not found');
663 }
664
665 if (!DolibarrApi::_checkAccessToResource('contact', $this->contact->id)) {
666 throw new RestException(403, 'Access not allowed for login '.DolibarrApiAccess::$user->login);
667 }
668 if (!DolibarrApi::_checkAccessToResource('category', $category->id)) {
669 throw new RestException(403, 'Access not allowed for login '.DolibarrApiAccess::$user->login);
670 }
671
672 $category->del_type($this->contact, 'contact');
673
674 return $this->_cleanObjectDatas($this->contact);
675 }
676
677 // phpcs:disable PEAR.NamingConventions.ValidFunctionName.PublicUnderscore
687 protected function _cleanObjectDatas($object)
688 {
689 // phpcs:enable
690 $object = parent::_cleanObjectDatas($object);
691
692 unset($object->total_ht);
693 unset($object->total_tva);
694 unset($object->total_localtax1);
695 unset($object->total_localtax2);
696 unset($object->total_ttc);
697
698 unset($object->note);
699 unset($object->lines);
700 unset($object->thirdparty);
701
702 return $object;
703 }
704
712 private function _validate($data)
713 {
714 $contact = array();
715 foreach (Contacts::$FIELDS as $field) {
716 if (!isset($data[$field])) {
717 throw new RestException(400, "$field field missing");
718 }
719 $contact[$field] = $data[$field];
720 }
721
722 return $contact;
723 }
724}
$id
Support class for third parties, contacts, members, users or resources.
Definition account.php:47
if(! $sortfield) if(! $sortorder) $object
Definition account.php:100
Class to manage categories.
Class to manage contact/addresses.
addCategory($id, $category_id)
Add a category to a contact.
put($id, $request_data=null)
Update a contact.
_validate($data)
Validate fields before create or update object.
deleteCategory($id, $category_id)
Remove the link between a category and a contact.
createUser($id, $request_data=null)
Create a user account object from contact (external user)
post($request_data=null)
Create a contact.
getByEmail($email, $includecount=0, $includeroles=0)
Get a contact by Email.
getCategories($id, $sortfield="s.rowid", $sortorder='ASC', $limit=0, $page=0)
Get categories of a contact.
_cleanObjectDatas($object)
Clean sensible object data @phpstan-template T.
__construct()
Constructor.
index($sortfield="t.rowid", $sortorder='ASC', $limit=100, $page=0, $thirdparty_ids='', $category=0, $sqlfilters='', $includecount=0, $includeroles=0, $properties='', $pagination_data=false)
List contacts.
Class for API REST v1.
Definition api.class.php:35
_checkValExtrafieldsForAPI($field, $value, $object)
Check and convert a string depending on its type/name.
_filterObjectProperties($object, $properties)
Filter properties that will be returned on object.
static _checkAccessToResource($resource, $resource_id=0, $dbtablename='', $feature2='', $dbt_keyfield='fk_soc', $dbt_select='rowid')
Check access by user to a given resource.
_checkValForAPI($field, $value, $object)
Check and convert a string depending on its type/name.
Class to manage third parties objects (customers, suppliers, prospects...)
Class to manage Dolibarr users.
if(!isModEnabled('ai')||!getDolGlobalString('AI_ASSISTANT_ENABLED')) global $conf
The main.inc.php has been included so the following variable are now defined:
if(!isModEnabled('ai')||!getDolGlobalString('AI_ASSISTANT_ENABLED')) global $db
API class for accounts.
dol_getIdFromCode($db, $key, $tablename, $fieldkey='code', $fieldid='id', $entityfilter=0, $filters='', $useCache=true)
Return an id or code from a code or id.
sanitizeVal($out='', $check='alphanohtml', $filter=null, $options=null)
Return a sanitized or empty value after checking value against a rule.
isModEnabled($module)
Is Dolibarr module enabled.