'de', 1 => 'en']; private $taxationByDestinationCountry; private $orderSearchLimit; private $configuration_identifier; private $mirakl_error_text_product_missing; private $configuration_order_fetch_status; private $normalTaxId; private $reducedTaxId; private $offer_configuration; private $product_configuration; public function __construct($app, $intern = false) { $this->app = $app; $this->logger = $app->Container->get('Logger'); $this->intern = $intern; if ($intern) return; } /* * See widget.shopexport.php, ShowExtraeinstellungen() */ public function EinstellungenStruktur() { $einstellungen = [ 'ausblenden' => ['abholmodus' => ['ab_nummer', 'zeitbereich']], 'functions' => ['getarticlelist'], 'felder' => [ 'apikey' => [ 'typ' => 'text', 'bezeichnung' => '{|API Key|}:', 'size' => 40, ], 'shopurl' => [ 'typ' => 'text', 'bezeichnung' => '{|Shop URL|}:', 'size' => 40, ], 'mirakl_shopid' => [ 'typ' => 'text', 'bezeichnung' => '{|Shop ID des Shops|}:', 'size' => 40, 'info' => 'optional, int64' ], 'configuration_identifier_source' => [ 'typ' => 'select', 'bezeichnung' => '{|Konfigurationsidentifizierer-Typ|}:', 'size' => 40, 'info' => 'Woher soll die Konfiguration des jeweiligen Artikels bezogen werden?', 'default' => 'feld', 'optionen' => ['feld' => '{|Feld|}', 'freifeld' => '{|Freifeld|}', 'eigenschaft' => '{|Eigenschaft|}', 'wert' => '{|Fester Wert|}'] ], 'configuration_identifier_source_value' => [ 'typ' => 'text', 'bezeichnung' => '{|Konfigurationsidentifizierer-Wert|}:', 'size' => 40, 'info' => '', 'default' => 'kategoriename' ], 'mirakl_error_text_product_missing' => [ 'typ' => 'text', 'bezeichnung' => '{|Fehlertext Produkt fehlt|}:', 'size' => 40, 'info' => 'Der Fehlertext der anzeigt dass das Produkt nicht existiert (Angebotsimport, Fehlerbericht)', 'default' => 'The product does not exist' ], 'configuration_order_fetch_status' => [ 'typ' => 'text', 'bezeichnung' => '{|Status für abzuholende Aufträge|}:', 'size' => 40, 'info' => 'STAGING, WAITING_ACCEPTANCE, WAITING_DEBIT, WAITING_DEBIT_PAYMENT, SHIPPING, SHIPPED, TO_COLLECT, RECEIVED, CLOSED, REFUSED, CANCELED', 'default' => 'WAITING_ACCEPTANCE' ], 'normalTaxId' => [ 'typ' => 'text', 'bezeichnung' => '{|TaxId für Steuersatz "normal"|}', 'size' => 40, ], 'reducedTaxId' => [ 'typ' => 'text', 'bezeichnung' => '{|TaxId für Steuersatz "ermäßigt"|}', 'size' => 40, ], 'product_configuration' => [ 'typ' => 'textarea', 'bezeichnung' => '{|Zuordnung Produkt-Felder je Konfiguration (JSON)|}:', 'cols' => 80, 'rows' => 20, 'info' => 'Die Felder werden vom Mirakl-Betreiber vorgegeben. Ist keine Konfiguration definiert, gilt der Eintrag für alle Artikel. Jedes Feld kann wie folgt zugeordnet werden:
Artikelfeld: "Mirakel-Feldname": {"feld": "xyz"} oder kurz "Mirakel-Feldname": "xyz",
Freifeld: "Mirakel-Feldname": {"freifeld": "Bezeichnung in Shop"} (Siehe Reiter "Freifelder"),
Eigenschaft: "Mirakel-Feldname": {"eigenschaft": "Eigenschaftenname xyz"},
Fester Wert: "Mirakel-Feldname": {"wert": "xyz"}

Optionen:
Text voranstellen: "praefix: "Dieser Text vorne",
Text hinten anstellen: "postfix: "Dieser Text hinten",
Standardwert: "standardwert: "Dieser Wert wenn nichts gefunden wurde",
Als Zusatzfeld senden: "zusatzfeld": true', 'placeholder' => '[ { "konfigurationen": [ "Schuhe", "Hosen" ], "felder": { "category": {"freifeld": "Kategorie"}, "Product.SellerProductID": {"feld": "nummer"}, "SHOP.PRODUCT.TITLE": {"feld": "nummer"}, "ATT.GLOBAL.Brandname": {"feld": "preis"}, "Product.BaseUnit": {"freifeld": "Kategorie"}, "ATT.GLOBAL.NoCUperOU": {"eigenschaft": "Mirakl Steuertext", "praefix": "ST-"}, "ATT.GLOBAL.NoCUperOU__UNIT": {"wert": "false","zusatzfeld": true}, "Product.TaxIndicator": {"wert": "1","zusatzfeld": true} } } ]' ], 'offer_configuration' => [ 'typ' => 'textarea', 'bezeichnung' => '{|Zuordnung Angebots-Felder je Konfiguration (JSON)|}:', 'cols' => 80, 'rows' => 20, 'info' => '', 'placeholder' => '[ { "konfigurationen": [ "Schuhe", "Hosen" ], "felder": { "product_id_type": {"wert": "SHOP_SKU"}, "product_id": {"feld": "nummer"}, "shop_sku": {"feld": "nummer"}, "price": {"feld": "preis"}, "description": {"freifeld": "Kategorie"}, "internal_description": {"eigenschaft": "Mirakl Steuertext"}, "reversecharge": {"wert": "false","zusatzfeld": true}, "warehouse": {"wert": "1","zusatzfeld": true}, "quantity": {"feld": "anzahl_lager"} } } ]' ], 'Artikelfelder' => [ 'heading' => 'Zusatzinformationen', 'typ' => 'info', 'text' => 'Folgende Artikelfelder stehen zur Verfügung:', 'bezeichnung' => null, 'info' => 'artikel, artikelid, nummer, inaktiv, name_de, name_en, einheit, hersteller, herstellernummer, ean, artikelnummer_fremdnummern, kurztext_de, kurztext_en, anabregs_text, anabregs_text_en, beschreibung_de, beschreibung_en, uebersicht_de, uebersicht_en, herkunftsland, texteuebertragen, metadescription_de, metadescription_en, metakeywords_de, metakeywords_en, metatitle_de, metatitle_en, links_de, altersfreigabe, links_en, startseite_de, startseite_en, restmenge, startseite, standardbild, herstellerlink, lieferzeit, lieferzeitmanuell, gewicht, laenge, breite, hoehe, wichtig, porto, gesperrt, sperrgrund, gueltigbis, umsatzsteuer, ausverkauft, variante, variante_von_id, variantevon, pseudopreis, keinrabatterlaubt, einkaufspreis, pseudolager, downloadartikel, zolltarifnummer, freifeld_Kategorie, typ, kategoriename, steuer_art_produkt, steuer_art_produkt_download, anzahl_bilder, anzahl_lager, lagerkorrekturwert, autolagerlampe, waehrung, preis, steuersatz, bruttopreis, checksum, variantevorhanden' ] ] ]; return($einstellungen); } public function getKonfig($shopid, $data) { $this->shopid = $shopid; $this->data = $data; $importerSettings = $this->app->DB->SelectArr("SELECT `einstellungen_json` FROM `shopexport` WHERE `id` = '$shopid' LIMIT 1"); $importerSettings = reset($importerSettings); $einstellungen = []; if (!empty($importerSettings['einstellungen_json'])) { $einstellungen = json_decode($importerSettings['einstellungen_json'], true); } $this->apiKey = $einstellungen['felder']['apikey']; $this->shopUrl = rtrim($einstellungen['felder']['shopurl'], '/') . '/'; $this->mirakl_shopid = $einstellungen['felder']['mirakl_shopid']; $this->configuration_identifier = array($einstellungen['felder']['configuration_identifier_source'] => $einstellungen['felder']['configuration_identifier_source_value']); $this->mirakl_error_text_product_missing = $einstellungen['felder']['mirakl_error_text_product_missing']; $this->configuration_order_fetch_status = $einstellungen['felder']['configuration_order_fetch_status']; $this->normalTaxId = $einstellungen['felder']['normalTaxId']; $this->reducedTaxId = $einstellungen['felder']['reducedTaxId']; $this->offer_configuration = json_decode($einstellungen['felder']['offer_configuration'], true, flags: JSON_THROW_ON_ERROR); $this->product_configuration = json_decode($einstellungen['felder']['product_configuration'], true, flags: JSON_THROW_ON_ERROR); } private function miraklRequest(string $endpoint, $postdata = null, array $getdata = null, string $content_type = null, bool $raw = false, $debug = false, $debugurl = null) { $ch = curl_init(); $url_addition = ""; $headers = array("Authorization: " . $this->apiKey); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); if (!empty($this->mirakl_shopid)) { $getdata['shop_id'] = $this->mirakl_shopid; } if (!empty($getdata)) { $url_addition = "?"; $ampersand = ""; foreach ($getdata as $key => $value) { $url_addition .= $ampersand . $key . "=" . $value; $ampersand = "&"; } } if (!empty($postdata)) { curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST'); curl_setopt($ch, CURLOPT_POSTFIELDS, $postdata); $headers[] = 'Content-Type: ' . $content_type; } if ($debugurl) { $url = $debugurl; } else { $url = $this->shopUrl; } curl_setopt($ch, CURLOPT_URL, $url . $endpoint . $url_addition); curl_setopt($ch, CURLOPT_HTTPHEADER, $headers); curl_setopt($ch, CURLINFO_HEADER_OUT, true); curl_setopt($ch, CURLOPT_VERBOSE, true); $response = curl_exec($ch); if (curl_error($ch)) { $this->error[] = curl_error($ch); } curl_close($ch); $information = curl_getinfo($ch); if ($debug) { print_r($information); print_r($postdata); print_r($response); } if ($raw) return $response; return simplexml_load_string($response); } public function ImportAuth() { $ch = curl_init($this->shopUrl . "version"); curl_setopt($ch, CURLOPT_HTTPHEADER, array("Authorization: " . $this->apiKey)); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); $response = curl_exec($ch); $code = curl_getinfo($ch, CURLINFO_RESPONSE_CODE); if ($code == 200) { return 'success'; } return $response; } /* * Fetches article list from the shop, puts them into table shopexport_getarticles, starts the prozessstarter getarticles which fetches details for each article via ImportGetArticle() */ public function ImportGetArticleList() { $result = []; $response = $this->miraklRequest('offers', raw: true); $result_array = json_decode($response); foreach ($result_array->offers as $offer) { $result[] = $offer->shop_sku; } array_unique($result); return $result; } /* * Fetches article details from the shop */ public function ImportGetArticle() { $articlelist = $this->CatchRemoteCommand('data'); $parameters = array('product_references' => 'productID', 'product_'); $response = $this->miraklRequest('products?', raw: true); throw new Exception("Not implemented"); } /* * Gets a flexible mapped fieldvalue from feld, wert or eigenschaft */ public function GetFieldValue($article, $field_map_entry) { $prefix = null; $postfix = null; $returnval = null; foreach ($field_map_entry as $key => $value) { switch ($key) { case 'feld': if (isset($article[$value])) { $returnval = $article[$value]; } else { throw new Exception("Artikelfeld existiert nicht: \"".$value."\""); } break; case 'freifeld': if (isset($article['freifelder']['DE'][$value])) { $returnval = $article['freifelder']['DE'][$value]; } else { throw new Exception("Freifeld existiert nicht: \"".$value."\""); } break; case 'eigenschaft': $sql = "SELECT wert FROM artikeleigenschaften ae INNER JOIN artikeleigenschaftenwerte aew ON aew.artikeleigenschaften = ae.id WHERE aew.artikel = '".$article['artikelid']."' AND ae.name = '".$value."' LIMIT 1"; $result = $this->app->DB->SelectRow($sql); if (!empty($result)) { $returnval = $result['wert']; } break; case 'wert': return($prefix.$value.$postfix); break; case 'praefix': $prefix = $value; break; case 'postfix': $postfix = $value; break; case 'standardwert': if(empty($returnval)) { $returnval = $value; } break; } } return($prefix.$returnval.$postfix); } public function ImportSendListLager() { return $this->mirakl_export_offers_and_products(export_products : false); } public function ImportSendList() { return $this->mirakl_export_offers_and_products(export_products : true); } /* * Send articles to shop */ public function mirakl_export_offers_and_products(bool $export_products) { // $this->Log(Logger::DEBUG, "mirakl_export_offers_and_products", array('export products' => $export_products)); $message = ""; $komma = ""; $status = true; $level = Logger::NOTICE; $log_result = null; $articlelist = $this->CatchRemoteCommand('data'); if ($export_products) { $export_products_result = $this->mirakl_export_products($articlelist); if ($export_products_result['export_products_returncode'] != 0) { $level = Logger::ERROR; $message = "Produktsync nach Mirakl hat Fehler"; $komma = ", "; $status = false; } else { $level = Logger::NOTICE; $this->Log(Logger::INFO, "Produktsync nach Mirakl ok", $export_products_result); $message = "Produktimport in Mirakl ok"; $komma = ", "; } } $export_offers_result = $this->mirakl_export_offers($articlelist); if ($export_offers_result['returncode'] != 0) { $level = Logger::ERROR; $message .= $komma."Angebotsync in Mirakl hat Fehler"; $status = false; } else { $this->Log(Logger::INFO, "Angebotsync nach Mirakl ok", $export_offers_result); $message .= $komma."Angebotsimport in Mirakl ok"; } $this->Log($level, $message, array('export_products_result' => $export_products_result, 'export_offers_result' => $export_offers_result)); foreach ($articlelist as $key => $article) { $articlelist[$key] = (array) $export_products_result['articlelist'][$key] + (array) $export_offers_result['articlelist'][$key]; // This merges the fields for each article } return( array( 'status' => $status, 'message' => $message, 'articlelist' => $articlelist ) ); // Check for missing products and try to create /* if ($export_products) { $export_products_list = array(); foreach ($export_offers_result['articlelist'] as $key => $article) { if ($article['mirakl_export_offers_result']['returncode'] == 12) { switch ($article['mirakl_export_offers_result']['message']) { case $this->mirakl_error_text_product_missing: $export_products_list[] = $article; break; } } } if (empty($export_products_list)) { return(array('status' => false, 'message' => "Produktimport nicht möglich")); } $export_products_result = $this->mirakl_export_products($export_products_list); if ($export_products_result['returncode'] != 0) { $this->Log(Logger::DEBUG, "Produktimport in Mirakl hat Fehler", $export_products_result); return(array('status' => false, 'message' => "Produktimport in Mirakl hat Fehler")); } // Retry offer import $export_offers_result = $this->mirakl_export_offers($articlelist); if ($export_offers_result['returncode'] == 0) { return(array('status' => true, 'message' => "Angebots und Produktimport in Mirakl ok")); } } $this->Log(Logger::DEBUG, "Angebotsimport in Mirakl hat Fehler", $export_offers_result); return(array('status' => false, 'message' => "Angebotsimport in Mirakl hat Fehler")); */ } // STAGING, WAITING_ACCEPTANCE, WAITING_DEBIT, WAITING_DEBIT_PAYMENT, SHIPPING, SHIPPED, TO_COLLECT, RECEIVED, CLOSED, REFUSED, CANCELED public function ImportGetAuftraegeAnzahl() { $response = $this->miraklRequest('orders', getdata: array('order_state_codes' => $this->configuration_order_fetch_status), raw: true); $this->Log(Logger::DEBUG, 'ImportGetAuftraegeAnzahl', $response); $result_array = json_decode($response); return($result_array->total_count); } public function ImportGetAuftrag() { $parameters = array('order_state_codes' => $this->configuration_order_fetch_status); if(!empty($this->data['nummer'])) { $parameters['order_ids'] = $this->data['nummer']; } $response = $this->miraklRequest('orders', getdata: $parameters, raw: true); $this->Log(Logger::DEBUG, 'ImportGetAuftrag', $response); $result_array = json_decode($response); $fetchedOrders = []; foreach ($result_array->orders as $order) { $cart = []; $cart['zeitstempel'] = strval($order->created_date); $cart['auftrag'] = strval($order->order_id); $cart['onlinebestellnummer'] = strval($order->commercial_id); $cart['gesamtsumme'] = strval($order->total_price); $cart['bestelldatum'] = strval($order->created_date); $cart['lieferung'] = strval($order->shipping_type_code); $cart['email'] = strval($order->customer_notification_email); // $cart['kunde_sprache'] = '?'; $cart['kundennummer'] = $order->customer->customer_id; $cart['name'] = ($order->customer->civility?$order->customer->civility." ":"").$order->customer->firstname." ".$order->customer->lastname; if (!empty($order->customer->billing_address->company)) { $cart['anrede'] = 'firma'; $cart['ansprechpartner'] = $cart['name']; $cart['name'] = strval($order->customer->billing_address->company); } $cart['strasse'] = strval($order->customer->billing_address->street_1); $cart['adresszusatz'] = strval($order->customer->billing_address->street_2); $cart['telefon'] = strval($order->customer->billing_address->phone); $cart['plz'] = strval($order->customer->billing_address->zip_code); $cart['ort'] = strval($order->customer->billing_address->city); // $cart['ustid'] = '?'; if ($order->customer->billing_address->country_iso_code) { $sql = "SELECT iso FROM laender WHERE iso3 = '".$order->customer->billing_address->country_iso_code."' LIMIT 1"; } $cart['land'] = $this->app->DB->Select($sql); $cart['abweichendelieferadresse'] = 1; $sql = "SELECT iso FROM laender WHERE iso3 = '".$order->customer->shipping_address->country_iso_code."'"; $cart['lieferadresse_land'] = $this->app->DB->Select($sql); $cart['lieferadresse_name'] = ($order->customer->shipping_address->civility?$order->customer->shipping_address->civility." ":"").$order->customer->shipping_address->firstname." ".$order->customer->shipping_address->lastname; if (!empty($order->customer->shipping_address->company)) { $cart['lieferadresse_ansprechpartner'] = $cart['lieferadresse_name']; $cart['lieferadresse_name'] = strval($order->customer->shipping_address->company); } $cart['lieferadresse_strasse'] = strval($order->customer->shipping_address->street_1); $cart['lieferadresse_adresszusatz'] = strval($order->customer->shipping_address->street_2); $cart['lieferadresse_telefon'] = strval($order->customer->shipping_address->phone); $cart['lieferadresse_plz'] = strval($order->customer->shipping_address->zip_code); $cart['lieferadresse_ort'] = strval($order->customer->shipping_address->city); $cart['internebemerkung'] = strval($order->customer->shipping_address->additional_info); if (is_array($order->order_additional_fields)) { $order_reference = array_search('customer-order-reference',array_column($order->order_additional_fields,'code')); if ($order_reference !== false) { $cart['ihrebestellnummer'] .= $order->order_additional_fields[$order_reference]->value; } } $cart['zahlungsweise'] = strval($order->payment_type); $cart['waehrung'] = strval($order->currency_iso_code); // $cart['ust_befreit'] = '?'; // 1, 2 // $cart['steuersatz_normal'] = 19;//strval($order->taxes[0]->amount); // $cart['steuersatz_ermaessigt'] 07; // strval($order->taxes[0]->amount); $cart['articlelist'] = []; $shipping_tax_amount = 0; foreach ($order->order_lines as $order_row) { $steuersatz = 0; $umsatzsteuer_typ = 'normal'; foreach($order_row->taxes as $tax) { if($tax->rate > $steuersatz) { $steuersatz = $tax->rate; } switch ($tax->code) { case $this->reducedTaxId: $umsatzsteuer_typ = 'ermaessigt'; break; default: case $this->normalTaxId: $umsatzsteuer_typ = 'normal'; break; } } $article = [ 'articleid' => strval($order_row->offer_sku), // 'name' => strval($order_row->product_title), 'quantity' => strval($order_row->quantity), 'price_netto' => strval($order_row->price_unit), 'umsatzsteuer' => $steuersatz ]; foreach ($order_row->shipping_taxes as $shipment_tax) { $shipping_tax_amount += $shipment_tax->amount; } $cart['articlelist'][] = $article; } // foreach articles $cart['versandkostennetto'] = round(strval($order->shipping_price),2); $cart['versandkostenbrutto'] = round($cart['versandkostennetto']+$shipping_tax_amount,2); } // foreach orders $fetchedOrders[] = [ 'id' => $cart['auftrag'], 'sessionid' => $cart['onlinebestellnummer'], 'logdatei' => '', 'warenkorb' => base64_encode(serialize($cart)), 'warenkorbjson' => base64_encode(json_encode($cart)) ]; $this->Log(Logger::DEBUG, 'Auftragsimport', $cart); return $fetchedOrders; } /* * Send offer data to mirakl * $articlelist = $this->CatchRemoteCommand('data'); * Return * array (returncode, message, articlelist) articlelist with added mirakl_export_offers_result element (array (returncode, message)) for further processing (e.g. create product) * returncode 0 = ok, 1 = not ok * returncode articlelist 0 = ok, 10 = missing required attributes, 11 category not found, 12 rejected from mirakl */ private function mirakl_export_offers(array $articlelist) : array { $mirakl_export_offers_return_value = array(); $mirakl_export_offers_return_value['returncode'] = 0; $this->Log(Logger::DEBUG, 'Angebotsexport Start', $articlelist); // First gather all articles as offers and send them // Wait for import to finish // Evaluate import foreach ($articlelist as $key => $article) { $return_article = array( 'artikel' => $article['artikel'], 'nummer' => $article['nummer'] ); $skip_article = false; /* * Export offer */ $configuration_found = false; $additional_fields = array(); // Prepare volume prices $volume_prices = array(); foreach ($article['staffelpreise_standard'] as $volume_price) { if ($volume_price['ab_menge'] > 1) { $volume_prices[] = array ( 'quantity_threshold' => (int) $volume_price['ab_menge'], 'unit_discount_price' => $volume_price['preis'], 'unit_origin_price' => $article['preis'] ); } } $all_prices = array ( 'volume_prices' => $volume_prices ); $offer_for_mirakl = array( 'state_code' => '11', // Condition new 'price' => $article['preis'], 'all_prices' => array($all_prices), 'update_delete' => null // Update delete flag. Could be empty (means "update"), "update" or "delete". ); $offer_configuration_name = $this->GetFieldValue($article, $this->configuration_identifier); $return_article['offer_configuration'] = $offer_configuration_name; foreach ($this->offer_configuration as $offer_configuration_entry) { if ($offer_configuration_entry['konfigurationen'] != null) { if (!in_array($offer_configuration_name,$offer_configuration_entry['konfigurationen'])) { continue; } } $configuration_found = true; $offer_configuration = $offer_configuration_entry; break; } if ($configuration_found) { // Check required attributes $required = [ 'product_id_type', 'product_id', 'shop_sku' ]; $missing = null; foreach ($required as $key) { if (!isset($offer_configuration['felder'][$key])) { $missing[] = $key; } } if ($missing) { $mirakl_export_offers_return_value['returncode'] = 1; $return_article['export_offers_returncode'] = 10; $return_article['export_offers_message'] = "Pflichtfelder fehlen in Angebotskonfiguration \"".$offer_configuration_name."\": ".implode(', ',$missing); $mirakl_export_offers_return_value['articlelist'][] = $return_article; $skip_article = true; } // Check Required attributes foreach ($offer_configuration['felder'] as $offer_field => $offer_field_source) { if (!is_array($offer_field_source)) { $offer_field_source = array('feld' => $offer_field_source); } $offer_field_value = null; $is_additional_field = false; $offer_field_value = $this->GetFieldValue($article, $offer_field_source); if (in_array('zusatzfeld', $offer_field_source)) { $is_additional_field = true; } if ($is_additional_field) { $additional_field = array ( "code" => $offer_field, "value" => $offer_field_value ); $additional_fields[] = $additional_field; } else { $offer_for_mirakl[$offer_field] = $offer_field_value; } } if ($skip_article) { continue; } if (!empty($additional_fields)) { $offer_for_mirakl['offer_additional_fields'] = $additional_fields; } $offers_for_mirakl[] = $offer_for_mirakl; $return_article['export_offers_returncode'] = 0; $mirakl_export_offers_return_value['articlelist'][] = $return_article; } else { // configuration_found $mirakl_export_offers_return_value['returncode'] = 1; $return_article['export_offers_returncode'] = 11; $return_article['export_offers_message'] = "Angebotskonfiguration für Artikel ".$article['nummer'].", Konfiguration \"".$offer_configuration_name."\" nicht gefunden"; $mirakl_export_offers_return_value['articlelist'][] = $return_article; continue; } } if (empty($offers_for_mirakl)) { $mirakl_export_offers_return_value['returncode'] = 1; $this->Log(Logger::ERROR, 'Angebotsexport keine Artikel bereit', $mirakl_export_offers_return_value); $mirakl_export_offers_return_value['message'] = "Angebotsexport keine Artikel bereit"; return($mirakl_export_offers_return_value); } $data_for_mirakl = array(); $data_for_mirakl['offers'] = $offers_for_mirakl; $json_for_mirakl = json_encode($data_for_mirakl); $this->Log(Logger::DEBUG, 'Angebotsexport Daten', ['json' => $json_for_mirakl]); $result = []; $response = $this->miraklRequest('offers', postdata: $json_for_mirakl, content_type: 'application/json', raw: true); $result = json_decode($response); if (!isset($result->import_id)) { $mirakl_export_offers_return_value['returncode'] = 1; $this->Log(Logger::ERROR, 'Angebotsimport abgelehnt', $response); $mirakl_export_offers_return_value['message'] = "Angebotsimport abgelehnt: ".print_r($response,true); return($mirakl_export_offers_return_value); } $this->Log(Logger::INFO, 'Angebotsimport angelegt', $response); $import_id = $result->import_id; // Wait for import to finish $status = null; /* WAITING_SYNCHRONIZATION_PRODUCT, WAITING, RUNNING, COMPLETE, FAILED */ while ($status != 'COMPLETE' && $status != 'FAILED') { sleep(5); $response = $this->miraklRequest('offers/imports/'.$import_id, raw: true); $result = json_decode($response); $status = $result->status; } if ($status == 'FAILED') { $this->Log(Logger::ERROR, 'Angebotsimport fehlgeschlagen', $response); $mirakl_export_offers_return_value['returncode'] = 2; $mirakl_export_offers_return_value['message'] = "Angebotsimport fehlgeschlagen: ".print_r($response,true); return($mirakl_export_offers_return_value); } if ($result->lines_in_error == 0) { $this->Log(Logger::INFO, 'Angebotsimport ok', $response); return($mirakl_export_offers_return_value); } $this->Log(Logger::ERROR, 'Angebotsimport meldet Fehler in '.$result->lines_in_error.' Zeilen', $response); $result = array(); // Check errors with CSV unfucking... $response = $this->miraklRequest('offers/imports/'.$import_id.'/error_report', raw: true); $response_lines = preg_split('/\R/', $response, flags: PREG_SPLIT_NO_EMPTY); $error_message_key = null; $firstline = true; foreach ($response_lines as $key => $response_line) { $response_array = str_getcsv($response_line,';','"'); if ($firstline) { $error_message_key = array_search("error-message",$response_array); $firstline = false; continue; } $article_key = array_search( $response_array['product_id'], array_column( $mirakl_export_offers_return_value['articlelist'], 'product_id') ); $mirakl_export_offers_return_value['articlelist'][$article_key]['mirakl_export_offers_result'] = array('returncode' => 12, 'message' => $response_array[$error_message_key]); } $this->Log(Logger::ERROR, 'Angebotsimport Fehlerbericht', $response); $mirakl_export_offers_return_value['returncode'] = 1; return($mirakl_export_offers_return_value); } /* * Create products * Return * array (returncode, message, articlelist) articlelist with added mirakl_export_offers_result element (array (returncode, message)) for further processing * returncode 0 = ok, 1 = not ok * returncode articlelist 0 = ok */ private function mirakl_export_products(array $articlelist) : array { $mirakl_export_products_return_value = array(); $mirakl_export_products_return_value['export_products_returncode'] = 0; $this->Log(Logger::DEBUG, 'Produktexport Start', $articlelist); $articlelist_per_configuration = array(); foreach ($articlelist as $key => $article) { // Determine configuration $configuration_found = false; $product_configuration_name = $this->GetFieldValue($article, $this->configuration_identifier); $mirakl_export_products_return_value['articlelist'][] = array( 'artikel' => $article['artikel'], 'nummer' => $article['nummer'], 'product_configuration' => $product_configuration_name ); foreach ($this->product_configuration as $product_configuration_id => $product_configuration) { if ($product_configuration['konfigurationen'] != null) { if (!in_array($product_configuration_name,$product_configuration['konfigurationen'])) { continue; } } $configuration_found = true; break; } if (!$configuration_found) { $mirakl_export_products_return_value['export_products_returncode'] = 1; $mirakl_export_products_return_value['articlelist'][$key]['export_products_returncode'] = 11; $mirakl_export_products_return_value['articlelist'][$key]['export_products_message'] = "Produktkonfiguration für Artikel ".$article['nummer'].", Konfiguration \"".$product_configuration_name."\" nicht gefunden"; continue; } $articlelist_per_configuration[$product_configuration_name]['product_configuration'] = $product_configuration; $articlelist_per_configuration[$product_configuration_name]['product_configuration_name'] = $product_configuration_name; $articlelist_per_configuration[$product_configuration_name]['articles'][] = $article; } // $this->Log(Logger::DEBUG, 'Articlelist per config', $articlelist_per_configuration); $komma = ''; foreach ($articlelist_per_configuration as $articlelist_one_configuration) { $result = $this->mirakl_export_products_one_configuration($articlelist_one_configuration); // $this->Log(Logger::DEBUG, 'Articlelist per config result', $result); if ($result['export_products_returncode']) { $mirakl_export_products_return_value['export_products_returncode'] = 1; $mirakl_export_products_return_value['message'] .= $komma.$result['message']." (".$articlelist_one_configuration['product_configuration_name'].")"; $komma = ', '; } foreach ($result['articlelist'] as $article_result) { $article_key = array_search( $article_result['artikel'], array_column( $mirakl_export_products_return_value['articlelist'], 'artikel') ); $mirakl_export_products_return_value['articlelist'][$article_key] += $article_result; } // $this->Log(Logger::DEBUG, 'mirakl_export_products_return_value', $mirakl_export_products_return_value); } return($mirakl_export_products_return_value); } private function mirakl_export_products_one_configuration (array $articlelist_one_configuration) { // Build CSV $csv_header = ""; $newline = ""; $mirakl_export_products_return_value = array(); foreach ($articlelist_one_configuration['articles'] as $article) { $product_for_mirakl = array(); $mirakl_export_products_return_value['articlelist'][] = array( 'artikel' => $article['artikel'], 'nummer' => $article['nummer'], 'product_configuration' => $product_configuration_name, 'export_products_returncode' => 0 ); foreach ($articlelist_one_configuration['product_configuration']['felder'] as $product_field => $product_field_source) { if (!is_array($product_field_source)) { $product_field_source = array('feld' => $product_field_source); } $product_field_value = null; $product_field_value = $this->GetFieldValue($article, $product_field_source); $product_for_mirakl[$product_field] = $product_field_value; } // Create CSV from array if (empty($csv_header)) { $csv_header .= '"'.implode('";"',array_keys($product_for_mirakl)).'"'; } $csv .= $newline.'"'.implode('";"',$product_for_mirakl).'"'; $newline = "\r\n"; } if (!$csv) { // No articles found return($mirakl_export_products_return_value); } $csv = $csv_header.$newline.$csv; $result = []; $this->Log(Logger::DEBUG, 'Produktexport Daten '.$articlelist_one_configuration['product_configuration_name'], array('csv' => $csv)); $postdata = array('file' => new CURLStringFile(postname: 'import.csv', data: $csv)); $response = $this->miraklRequest('products/imports', postdata: $postdata, content_type: 'multipart/form-data', raw: true); $result = json_decode($response); if (!isset($result->import_id)) { $this->Log(Logger::ERROR, 'Produktimport abgelehnt', $response); $mirakl_export_products_return_value['export_products_returncode'] = 1; $mirakl_export_products_return_value['export_products_message'] = 'Produktimport abgelehnt'; return($mirakl_export_products_return_value); } $this->Log(Logger::INFO, 'Produktimport angelegt', $result); $import_id = $result->import_id; // Wait for import to finish $status = null; /* WAITING_SYNCHRONIZATION_PRODUCT, WAITING, RUNNING, COMPLETE, FAILED */ while ($status != 'COMPLETE' && $status != 'FAILED') { sleep(30); $response = $this->miraklRequest('products/imports/'.$import_id, raw: true); $result = json_decode($response); $status = $result->import_status; } if ($status == 'FAILED') { $this->Log(Logger::ERROR, 'Produktimport fehlgeschlagen', $response); $mirakl_export_products_return_value['export_products_returncode'] = 1; return($mirakl_export_products_return_value); } if (!$result->has_error_report && !$result->has_transformation_error_report) { $this->Log(Logger::INFO, 'Produktimport ok', $response); } else { $this->Log(Logger::ERROR, 'Produktimport meldet Fehler', $response); $mirakl_export_products_return_value['export_products_returncode'] = 1; } if ($result->has_new_product_report) { $response = $this->miraklRequest('products/imports/'.$import_id.'/new_product_report', raw: true); $this->Log(Logger::DEBUG, 'Produktimport "Hinzugefügte Produkte"-Bericht', $response); } if ($result->has_transformed_file) { $response = $this->miraklRequest('products/imports/'.$import_id.'/transformed_file', raw: true); $this->Log(Logger::DEBUG, 'Produktimport Datei im Marketplace-Format', $response); } if ($result->has_error_report) { // Check errors with CSV unfucking... $response = $this->miraklRequest('products/imports/'.$import_id.'/error_report', raw: true); $response_lines = preg_split('/\R/', $response, flags: PREG_SPLIT_NO_EMPTY); $error_message_key = null; $firstline = true; foreach ($response_lines as $key => $response_line) { $response_array = str_getcsv($response_line,';','"'); if ($firstline) { $error_message_key = array_search("ERRORS",$response_array); $firstline = false; continue; } $article_key = array_search( $response_array['product_id'], array_column( $mirakl_export_products_return_value['articlelist'], 'product_id') ); $mirakl_export_products_return_value['articlelist'][$article_key]['export_products_returncode'] = 13; $mirakl_export_products_return_value['articlelist'][$article_key]['export_products_message'] = $response_array[$error_message_key]; } $this->Log(Logger::ERROR, 'Produktimport Fehlerbericht', $response); } if ($result->has_transformation_error_report) { // Check errors with CSV unfucking... $response = $this->miraklRequest('products/imports/'.$import_id.'/transformation_error_report', raw: true); $response_lines = preg_split('/\R/', $response, flags: PREG_SPLIT_NO_EMPTY); $error_message_key = null; $firstline = true; foreach ($response_lines as $key => $response_line) { $response_array = str_getcsv($response_line,';','"'); if ($firstline) { $error_message_key = array_search("errors",$response_array); $firstline = false; continue; } $article_key = array_search( $response_array['product_id'], array_column( $mirakl_export_products_return_value['articlelist'], 'product_id') ); $mirakl_export_products_return_value['articlelist'][$article_key]['export_products_returncode'] = 14; $mirakl_export_products_return_value['articlelist'][$article_key]['export_products_message'] = $response_array[$error_message_key]; } $this->Log(Logger::ERROR, 'Produktimport Transformation-Fehlerbericht', $response); } return($mirakl_export_products_return_value); } private function Log($level, $message, $dump = array()) { $this->logger->Log($level, 'Mirakl (Shop '.$this->shopid.') '.$message, (array) $dump); } private function getOrdersToProcess(int $limit) { echo("getOrdersToProcess"); exit(); } public function ImportDeleteAuftrag() { echo("ImportDeleteAuftrag"); exit(); } public function ImportUpdateAuftrag() { echo("ImportUpdateAuftrag"); exit(); } }