where('account', $account); if (!empty($unit)) { $clientMapper->where('unit', $unit); } /** @var \Modules\ClientManagement\Models\Client $client */ $client = $clientMapper->execute(); return $client->id === 0 ? null : $client; } /** * Api method to create news article * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiClientCreate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { if (!empty($val = $this->validateClientCreate($request))) { $response->data['client_create'] = new FormValidation($val); $response->header->status = RequestStatusCode::R_400; return; } $client = $this->createClientFromRequest($request); $this->createModel($request->header->account, $client, ClientMapper::class, 'client', $request->getOrigin()); // Set VAT Id // @todo: move to separate function if ($request->hasData('vat_id')) { /** @var \Modules\Organization\Models\Unit $unit */ $unit = UnitMapper::get() ->with('attributes') ->where('id', $this->app->unitId) ->execute(); $validate = ['status' => -1]; if (\in_array($client->mainAddress->getCountry(), ISO3166CharEnum::getRegion('eu'))) { $validate = EUVATVies::validateQualified( $request->getDataString('vat_id') ?? '', $unit->getAttribute('vat_id')->value->valueStr ?? '', $client->account->name1, $client->mainAddress->city, $client->mainAddress->postal, $client->mainAddress->address ); } $audit = new Audit( new NullAccount($request->header->account), null, (string) $validate['status'], StringUtils::intHash(EUVATVies::class), 'vat_validation', self::NAME, (string) $client->id, \json_encode($validate), (int) \ip2long($request->getOrigin()) ); AuditMapper::create()->execute($audit); if (($validate['status'] === 0 && $validate['vat'] === 'A' && $validate['name'] === 'A' && $validate['city'] === 'A') || $validate['status'] !== 0 // Api out of order -> accept it -> @todo: test it during invoice creation ) { /** @var \Modules\Attribute\Models\AttributeType $type */ $type = ClientAttributeTypeMapper::get()->where('name', 'vat_id')->execute(); $internalRequest = new HttpRequest(new HttpUri('')); $internalResponse = new HttpResponse(); $internalRequest->header->account = $request->header->account; $internalRequest->setData('client', $client->id); $internalRequest->setData('type', $type->id); $internalRequest->setData('custom', $request->hasData('vat_id')); $this->apiClientAttributeCreate($internalRequest, $internalResponse); } } // Find tax code if ($this->app->moduleManager->isActive('Billing')) { /** @var \Modules\Organization\Models\Unit $unit */ $unit = UnitMapper::get() ->with('mainAddress') ->where('id', $this->app->unitId) ->execute(); /** @var \Modules\Attribute\Models\AttributeType $type */ $type = ClientAttributeTypeMapper::get() ->where('name', 'sales_tax_code') ->execute(); $value = $this->app->moduleManager->get('Billing', 'ApiTax')->getClientTaxCode($client, $unit->mainAddress); $internalRequest = new HttpRequest(new HttpUri('')); $internalResponse = new HttpResponse(); $internalRequest->header->account = $request->header->account; $internalRequest->setData('client', $client->id); $internalRequest->setData('type', $type->id); $internalRequest->setData('value', $value->id); $this->apiClientAttributeCreate($internalRequest, $internalResponse); } $this->fillJsonResponse($request, $response, NotificationLevel::OK, 'Client', 'Client successfully created', $client); } /** * Method to create news article from request. * * @param RequestAbstract $request Request * * @return Client * * @since 1.0.0 */ private function createClientFromRequest(RequestAbstract $request) : Client { $account = null; if (!$request->hasData('account')) { $account = new Account(); $account->name1 = $request->getDataString('name1') ?? ''; $account->name2 = $request->getDataString('name2') ?? ''; } else { $account = new NullAccount((int) $request->getData('account')); } $client = new Client(); $client->number = $request->getDataString('number') ?? ''; $client->account = $account; $addr = new Address(); $addr->address = $request->getDataString('address') ?? ''; $addr->postal = $request->getDataString('postal') ?? ''; $addr->city = $request->getDataString('city') ?? ''; $addr->setCountry($request->getDataString('country') ?? ISO3166TwoEnum::_XXX); $addr->state = $request->getDataString('state') ?? ''; $geocoding = Nominatim::geocoding($addr->country, $addr->city, $addr->address); if ($geocoding === ['lat' => 0.0, 'lon' => 0.0]) { $geocoding = Nominatim::geocoding($addr->country, $addr->city); } $addr->lat = $geocoding['lat']; $addr->lon = $geocoding['lon']; $client->mainAddress = $addr; $client->unit = $request->getDataInt('unit'); return $client; } /** * Validate news create request * * @param RequestAbstract $request Request * * @return array * * @since 1.0.0 */ private function validateClientCreate(RequestAbstract $request) : array { $val = []; if (($val['number'] = !$request->hasData('number')) || ($val['account'] = !$request->hasData('name1') && !$request->hasData('account')) ) { return $val; } return []; } /** * Api method to update an account * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiMainAddressUpdate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { if (!empty($val = $this->validateMainAddressUpdate($request))) { $response->data['client_main_address'] = new FormValidation($val); $response->header->status = RequestStatusCode::R_400; return; } $clientMapper = $client = ClientMapper::get() ->with('mainAddress'); if ($request->hasData('account')) { $clientMapper->where('account', $request->getDataInt('account')) ->where('unit', $request->getDataInt('unit')); } elseif ($request->hasData('client')) { $clientMapper->where('id', $request->getDataInt('client')); } else { $clientMapper->where('account', $request->header->account) ->where('unit', $request->getDataInt('unit')); } /** @var \Modules\ClientManagement\Models\Client $client */ $client = $clientMapper->execute(); $old = $client->mainAddress; $new = $this->updateMainAddressFromRequest($request, clone $old); $this->updateModel($request->header->account, $old, $new, AddressMapper::class, 'address', $request->getOrigin()); $this->fillJsonResponse( $request, $response, NotificationLevel::OK, '', $this->app->l11nManager->getText($response->header->l11n->language, '0', '0', 'SuccessfulUpdate'), $new ); } /** * Validate news create request * * @param RequestAbstract $request Request * * @return array * * @since 1.0.0 */ private function validateMainAddressUpdate(RequestAbstract $request) : array { $val = []; if (($val['client'] = (!$request->hasData('client') && !$request->hasData('account'))) ) { return $val; } return []; } /** * Method to update an account from a request * * @param RequestAbstract $request Request * @param Address $address Address * * @return Address * * @since 1.0.0 */ private function updateMainAddressFromRequest(RequestAbstract $request, Address $address) : Address { $address->address = $request->getDataString('address') ?? $address->address; $address->postal = $request->getDataString('postal') ?? $address->postal; $address->city = $request->getDataString('city') ?? $address->city; $address->state = $request->getDataString('state') ?? $address->state; $address->setCountry($request->getDataString('country') ?? $address->getCountry()); return $address; } /** * Api method to create client l11n * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiClientL11nCreate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { if (!empty($val = $this->validateClientL11nCreate($request))) { $response->data['client_l11n_create'] = new FormValidation($val); $response->header->status = RequestStatusCode::R_400; return; } $clientL11n = $this->createClientL11nFromRequest($request); $this->createModel($request->header->account, $clientL11n, ClientL11nMapper::class, 'client_l11n', $request->getOrigin()); $this->fillJsonResponse($request, $response, NotificationLevel::OK, 'Localization', 'Localization successfully created', $clientL11n); } /** * Method to create client l11n from request. * * @param RequestAbstract $request Request * * @return BaseStringL11n * * @since 1.0.0 */ private function createClientL11nFromRequest(RequestAbstract $request) : BaseStringL11n { $clientL11n = new BaseStringL11n(); $clientL11n->ref = $request->getDataInt('client') ?? 0; $clientL11n->type = new NullBaseStringL11nType($request->getDataInt('type') ?? 0); $clientL11n->setLanguage( $request->getDataString('language') ?? $request->header->l11n->language ); $clientL11n->content = $request->getDataString('description') ?? ''; return $clientL11n; } /** * Validate client l11n create request * * @param RequestAbstract $request Request * * @return array * * @since 1.0.0 */ private function validateClientL11nCreate(RequestAbstract $request) : array { $val = []; if (($val['client'] = !$request->hasData('client')) || ($val['type'] = !$request->hasData('type')) || ($val['description'] = !$request->hasData('description')) ) { return $val; } return []; } /** * Api method to create client l11n type * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiClientL11nTypeCreate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { if (!empty($val = $this->validateClientL11nTypeCreate($request))) { $response->data['client_l11n_type_create'] = new FormValidation($val); $response->header->status = RequestStatusCode::R_400; return; } $clientL11nType = $this->createClientL11nTypeFromRequest($request); $this->createModel($request->header->account, $clientL11nType, ClientL11nTypeMapper::class, 'client_l11n_type', $request->getOrigin()); $this->fillJsonResponse($request, $response, NotificationLevel::OK, 'Localization type', 'Localization type successfully created', $clientL11nType); } /** * Method to create client l11n type from request. * * @param RequestAbstract $request Request * * @return BaseStringL11nType * * @since 1.0.0 */ private function createClientL11nTypeFromRequest(RequestAbstract $request) : BaseStringL11nType { $clientL11nType = new BaseStringL11nType(); $clientL11nType->title = $request->getDataString('title') ?? ''; $clientL11nType->isRequired = (bool) ($request->getData('is_required') ?? false); return $clientL11nType; } /** * Validate client l11n type create request * * @param RequestAbstract $request Request * * @return array * * @since 1.0.0 */ private function validateClientL11nTypeCreate(RequestAbstract $request) : array { $val = []; if (($val['title'] = !$request->hasData('title'))) { return $val; } return []; } /** * Api method to create client attribute * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiClientAttributeCreate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { if (!empty($val = $this->validateClientAttributeCreate($request))) { $response->data['attribute_create'] = new FormValidation($val); $response->header->status = RequestStatusCode::R_400; return; } $attribute = $this->createClientAttributeFromRequest($request); $this->createModel($request->header->account, $attribute, ClientAttributeMapper::class, 'attribute', $request->getOrigin()); $this->fillJsonResponse($request, $response, NotificationLevel::OK, 'Attribute', 'Attribute successfully created', $attribute); } /** * Method to create client attribute from request. * * @param RequestAbstract $request Request * * @return Attribute * * @since 1.0.0 */ private function createClientAttributeFromRequest(RequestAbstract $request) : Attribute { $attribute = new Attribute(); $attribute->ref = (int) $request->getData('client'); $attribute->type = new NullAttributeType((int) $request->getData('type')); if ($request->hasData('value')) { $attribute->value = new NullAttributeValue((int) $request->getData('value')); } else { $newRequest = clone $request; $newRequest->setData('value', $request->getData('custom'), true); $value = $this->createClientAttributeValueFromRequest($newRequest); $attribute->value = $value; } return $attribute; } /** * Validate client attribute create request * * @param RequestAbstract $request Request * * @return array * * @since 1.0.0 */ private function validateClientAttributeCreate(RequestAbstract $request) : array { $val = []; if (($val['type'] = !$request->hasData('type')) || ($val['value'] = (!$request->hasData('value') && !$request->hasData('custom'))) || ($val['client'] = !$request->hasData('client')) ) { return $val; } return []; } /** * Api method to create client attribute l11n * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiClientAttributeTypeL11nCreate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { if (!empty($val = $this->validateClientAttributeTypeL11nCreate($request))) { $response->data['attr_type_l11n_create'] = new FormValidation($val); $response->header->status = RequestStatusCode::R_400; return; } $attrL11n = $this->createClientAttributeTypeL11nFromRequest($request); $this->createModel($request->header->account, $attrL11n, ClientAttributeTypeL11nMapper::class, 'attr_type_l11n', $request->getOrigin()); $this->fillJsonResponse($request, $response, NotificationLevel::OK, 'Localization', 'Localization successfully created', $attrL11n); } /** * Method to create client attribute l11n from request. * * @param RequestAbstract $request Request * * @return BaseStringL11n * * @since 1.0.0 */ private function createClientAttributeTypeL11nFromRequest(RequestAbstract $request) : BaseStringL11n { $attrL11n = new BaseStringL11n(); $attrL11n->ref = $request->getDataInt('type') ?? 0; $attrL11n->setLanguage( $request->getDataString('language') ?? $request->header->l11n->language ); $attrL11n->content = $request->getDataString('title') ?? ''; return $attrL11n; } /** * Validate client attribute l11n create request * * @param RequestAbstract $request Request * * @return array * * @since 1.0.0 */ private function validateClientAttributeTypeL11nCreate(RequestAbstract $request) : array { $val = []; if (($val['title'] = !$request->hasData('title')) || ($val['type'] = !$request->hasData('type')) ) { return $val; } return []; } /** * Api method to create client attribute type * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiClientAttributeTypeCreate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { if (!empty($val = $this->validateClientAttributeTypeCreate($request))) { $response->data['attr_type_create'] = new FormValidation($val); $response->header->status = RequestStatusCode::R_400; return; } $attrType = $this->createClientAttributeTypeFromRequest($request); $this->createModel($request->header->account, $attrType, ClientAttributeTypeMapper::class, 'attr_type', $request->getOrigin()); $this->fillJsonResponse($request, $response, NotificationLevel::OK, 'Attribute type', 'Attribute type successfully created', $attrType); } /** * Method to create client attribute from request. * * @param RequestAbstract $request Request * * @return AttributeType * * @since 1.0.0 */ private function createClientAttributeTypeFromRequest(RequestAbstract $request) : AttributeType { $attrType = new AttributeType($request->getDataString('name') ?? ''); $attrType->datatype = $request->getDataInt('datatype') ?? 0; $attrType->custom = $request->getDataBool('custom') ?? false; $attrType->isRequired = (bool) ($request->getData('is_required') ?? false); $attrType->validationPattern = $request->getDataString('validation_pattern') ?? ''; $attrType->setL11n($request->getDataString('title') ?? '', $request->getDataString('language') ?? ISO639x1Enum::_EN); $attrType->setFields($request->getDataInt('fields') ?? 0); return $attrType; } /** * Validate client attribute create request * * @param RequestAbstract $request Request * * @return array * * @since 1.0.0 */ private function validateClientAttributeTypeCreate(RequestAbstract $request) : array { $val = []; if (($val['title'] = !$request->hasData('title')) || ($val['name'] = !$request->hasData('name')) ) { return $val; } return []; } /** * Api method to create client attribute value * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiClientAttributeValueCreate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { if (!empty($val = $this->validateClientAttributeValueCreate($request))) { $response->data['attr_value_create'] = new FormValidation($val); $response->header->status = RequestStatusCode::R_400; return; } $attrValue = $this->createClientAttributeValueFromRequest($request); $this->createModel($request->header->account, $attrValue, ClientAttributeValueMapper::class, 'attr_value', $request->getOrigin()); if ($attrValue->isDefault) { $this->createModelRelation( $request->header->account, (int) $request->getData('type'), $attrValue->id, ClientAttributeTypeMapper::class, 'defaults', '', $request->getOrigin() ); } $this->fillJsonResponse($request, $response, NotificationLevel::OK, 'Attribute value', 'Attribute value successfully created', $attrValue); } /** * Method to create client attribute value from request. * * @param RequestAbstract $request Request * * @return AttributeValue * * @since 1.0.0 */ private function createClientAttributeValueFromRequest(RequestAbstract $request) : AttributeValue { /** @var AttributeType $type */ $type = ClientAttributeTypeMapper::get() ->where('id', $request->getDataInt('type') ?? 0) ->execute(); $attrValue = new AttributeValue(); $attrValue->isDefault = $request->getDataBool('default') ?? false; $attrValue->setValue($request->getData('value'), $type->datatype); if ($request->hasData('title')) { $attrValue->setL11n($request->getDataString('title') ?? '', $request->getDataString('language') ?? ISO639x1Enum::_EN); } return $attrValue; } /** * Validate client attribute value create request * * @param RequestAbstract $request Request * * @return array * * @since 1.0.0 */ private function validateClientAttributeValueCreate(RequestAbstract $request) : array { $val = []; if (($val['type'] = !$request->hasData('type')) || ($val['value'] = !$request->hasData('value')) ) { return $val; } return []; } /** * Api method to create client attribute l11n * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiClientAttributeValueL11nCreate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { if (!empty($val = $this->validateClientAttributeValueL11nCreate($request))) { $response->data['attr_value_l11n_create'] = new FormValidation($val); $response->header->status = RequestStatusCode::R_400; return; } $attrL11n = $this->createClientAttributeValueL11nFromRequest($request); $this->createModel($request->header->account, $attrL11n, ClientAttributeValueL11nMapper::class, 'attr_value_l11n', $request->getOrigin()); $this->fillJsonResponse($request, $response, NotificationLevel::OK, 'Localization', 'Localization successfully created', $attrL11n); } /** * Method to create Client attribute l11n from request. * * @param RequestAbstract $request Request * * @return BaseStringL11n * * @since 1.0.0 */ private function createClientAttributeValueL11nFromRequest(RequestAbstract $request) : BaseStringL11n { $attrL11n = new BaseStringL11n(); $attrL11n->ref = $request->getDataInt('value') ?? 0; $attrL11n->setLanguage( $request->getDataString('language') ?? $request->header->l11n->language ); $attrL11n->content = $request->getDataString('title') ?? ''; return $attrL11n; } /** * Validate Client attribute l11n create request * * @param RequestAbstract $request Request * * @return array * * @since 1.0.0 */ private function validateClientAttributeValueL11nCreate(RequestAbstract $request) : array { $val = []; if (($val['title'] = !$request->hasData('title')) || ($val['value'] = !$request->hasData('value')) ) { return $val; } return []; } /** * Api method to create client files * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiFileCreate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { $uploadedFiles = $request->files; if (empty($uploadedFiles)) { $this->fillJsonResponse($request, $response, NotificationLevel::ERROR, 'Client', 'Invalid file', $uploadedFiles); $response->header->status = RequestStatusCode::R_400; return; } $uploaded = $this->app->moduleManager->get('Media')->uploadFiles( names: $request->getDataList('names'), fileNames: $request->getDataList('filenames'), files: $uploadedFiles, account: $request->header->account, basePath: __DIR__ . '/../../../Modules/Media/Files/Modules/ClientManagement/' . ($request->getData('client') ?? '0'), virtualPath: '/Modules/ClientManagement/' . ($request->getData('client') ?? '0'), pathSettings: PathSettings::FILE_PATH ); if ($request->hasData('type')) { foreach ($uploaded as $file) { $this->createModelRelation( $request->header->account, $file->id, $request->getDataInt('type'), MediaMapper::class, 'types', '', $request->getOrigin() ); } } $this->createModelRelation( $request->header->account, (int) $request->getData('client'), \reset($uploaded)->id, ClientMapper::class, 'files', '', $request->getOrigin() ); $this->fillJsonResponse($request, $response, NotificationLevel::OK, 'File', 'File successfully updated', $uploaded); } /** * Api method to create client files * * @param RequestAbstract $request Request * @param ResponseAbstract $response Response * @param mixed $data Generic data * * @return void * * @api * * @since 1.0.0 */ public function apiNoteCreate(RequestAbstract $request, ResponseAbstract $response, mixed $data = null) : void { $request->setData('virtualpath', '/Modules/ClientManagement/' . $request->getData('id'), true); $this->app->moduleManager->get('Editor')->apiEditorCreate($request, $response, $data); $responseData = $response->get($request->uri->__toString()); if (!\is_array($responseData)) { return; } $model = $responseData['response']; $this->createModelRelation($request->header->account, $request->getData('id'), $model->id, ClientMapper::class, 'notes', '', $request->getOrigin()); } }