Merge branch 'warehouse-add-email-parsing' into 'master'
added new tracking support See merge request fronk/thetool!636
This commit is contained in:
@@ -12,6 +12,7 @@ class WarehouseEShopOrderController extends TTCrud {
|
||||
'modal' => ['type' => 'select',
|
||||
'items' => [['value' => 'new', 'text' => 'Neu'],
|
||||
['value' => 'accepted', 'text' => 'An Lieferant übergeben'],
|
||||
['value' => 'acceptedInternally', 'text' => 'Interne verarbeitung'],
|
||||
['value' => 'sent', 'text' => 'Gesendet'],
|
||||
['value' => 'done', 'text' => 'Erledigt'],]],
|
||||
'table' => ['filter' => 'select']],
|
||||
@@ -26,6 +27,7 @@ class WarehouseEShopOrderController extends TTCrud {
|
||||
['key' => 'deliveryAddressLine', 'text' => 'Adresse', 'required' => true, 'required_length' => 4],
|
||||
['key' => 'deliveryAddressPLZ', 'text' => 'PLZ', 'required' => true, 'regex' => '/^\d{4}$/'],
|
||||
['key' => 'deliveryAddressCity', 'text' => 'Stadt', 'required' => true, 'required_length' => 3],
|
||||
['key' => 'trackingNumber', 'text' => 'Trackingnummer', 'required' => false, 'modal' => false],
|
||||
['key' => 'create', 'text' => 'Erstellt', 'required' => true, 'modal' => false, 'filter' => 'datetime'],
|
||||
['key' => 'createBy',
|
||||
'text' => 'Erstellt von',
|
||||
@@ -39,6 +41,7 @@ class WarehouseEShopOrderController extends TTCrud {
|
||||
'table' => ['filter' => false, 'sortable' => false, 'class' => 'text-center']],];
|
||||
|
||||
protected array $additionalActions = [['key' => 'openHistory', 'title' => 'Historie', 'class' => 'fas fa-history text-primary'],
|
||||
['key' => 'showTrackingHistory', 'title' => 'Tracking Historie', 'class' => 'fas fa-truck text-primary'],
|
||||
['key' => 'openSingleOrderEmail', 'title' => 'Bestellbestätigung', 'class' => 'fas fa-envelope text-primary'],];
|
||||
|
||||
protected array $infoMessages = ['create' => 'Bestellung wurde erfolgreich erstellt, sie erhalten in Kürze eine Bestätigungsmail',
|
||||
@@ -55,7 +58,7 @@ class WarehouseEShopOrderController extends TTCrud {
|
||||
return ['value' => intval($user->id), 'text' => $user->name];
|
||||
}, UserModel::search());
|
||||
|
||||
$this->columns[9]['modal']['items'] = $users;
|
||||
$this->columns[10]['modal']['items'] = $users;
|
||||
}
|
||||
|
||||
protected function singleOrderEmailAction() {
|
||||
@@ -106,6 +109,8 @@ class WarehouseEShopOrderController extends TTCrud {
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
protected function CSVExportNewOrdersMarkAcceptedAction($returnCSV = false, $orderIds = []) {
|
||||
$orders = WarehouseEShopOrderModel::getAll(['status' => 'new']);
|
||||
$orders = array_map(function ($order) {
|
||||
@@ -292,4 +297,143 @@ class WarehouseEShopOrderController extends TTCrud {
|
||||
self::returnJson((new WarehouseHistoryController)->getHistory($this->request->id, $this->mod, $this->columns));
|
||||
}
|
||||
|
||||
protected function GLSTrackingApi($trackingNumber, $plz) {
|
||||
$url = "https://gls-group.eu/app/service/open/rest/AT/de/rstt028/$trackingNumber?postalCode=$plz";
|
||||
return file_get_contents($url);
|
||||
}
|
||||
|
||||
protected function checkAndUpdateTrackingState($tracking, $orderId) {
|
||||
// check inside the tracking json if data.history[xxx].evtDscr (lowercase) contains "erfolgreich zugestellt"
|
||||
$tracking = json_decode($tracking, true);
|
||||
$history = $tracking['history'];
|
||||
$successfulDelivery = false;
|
||||
foreach ($history as $entry) {
|
||||
if (strpos(strtolower($entry['evtDscr']), 'erfolgreich zugestellt') !== false) {
|
||||
$successfulDelivery = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if ($successfulDelivery) {
|
||||
$order = (array) WarehouseEShopOrderModel::get($orderId);
|
||||
$order['status'] = 'done';
|
||||
WarehouseEShopOrderModel::update($order);
|
||||
}
|
||||
}
|
||||
|
||||
protected function fetchGLSTrackingAction() {
|
||||
$orderId = $this->request->id;
|
||||
|
||||
$order = WarehouseEShopOrderModel::get($orderId);
|
||||
$trackingNumber = $order->trackingNumber;
|
||||
$plz = $order->deliveryAddressPLZ;
|
||||
|
||||
$tracking = $this->GLSTrackingApi($trackingNumber, $plz);
|
||||
$this->checkAndUpdateTrackingState($tracking, $orderId);
|
||||
die($tracking);
|
||||
}
|
||||
|
||||
protected function readGLSEmailAction() {
|
||||
$host = '{mail.xinon.at:993/imap/ssl/novalidate-cert}INBOX';
|
||||
$mbox = imap_open($host, 'eshop-versand@xinon.at', 'savemanfb545aw');
|
||||
$emails = imap_search($mbox, 'ALL');
|
||||
|
||||
if ($emails) {
|
||||
// Sort emails from newest to oldest
|
||||
rsort($emails);
|
||||
|
||||
// Loop through each email
|
||||
foreach ($emails as $email_number) {
|
||||
$overview = imap_fetch_overview($mbox, $email_number, 0);
|
||||
|
||||
if (strpos($overview[0]->from, 'gls') === false) {
|
||||
//continue;
|
||||
}
|
||||
|
||||
$message = imap_fetchbody($mbox, $email_number, 1); // 1 for plain text part
|
||||
|
||||
// START ADDRESS PARSING
|
||||
$address = '';
|
||||
$lines = explode("\n", $message);
|
||||
$start = false;
|
||||
foreach ($lines as $line) {
|
||||
if (strpos($line, 'ZUSTELLADRESSE') !== false) {
|
||||
$start = true;
|
||||
}
|
||||
if ($start) {
|
||||
$address .= $line . "\n";
|
||||
}
|
||||
if ($start && preg_match('/[a-zA-Z]/', $line) === 0) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
$address = substr($address, strpos($address, "\n") + 1);
|
||||
|
||||
$addressLines = explode("\n", $address);
|
||||
$addressLine = trim(array_shift($addressLines));
|
||||
$plzCity = trim(array_shift($addressLines));
|
||||
$plzCityParts = explode(' ', $plzCity);
|
||||
$plz = $plzCityParts[0];
|
||||
$city = $plzCityParts[1];
|
||||
// END ADDRESS PARSING
|
||||
|
||||
// START TRACKING NUMBER PARSING
|
||||
$trackingNumber = '';
|
||||
preg_match('/\d{6,}/', $message, $matches);
|
||||
if (!empty($matches)) {
|
||||
$trackingNumber = $matches[0];
|
||||
}
|
||||
// END TRACKING NUMBER PARSING
|
||||
|
||||
|
||||
$orders = WarehouseEShopOrderModel::getAll(['deliveryAddressLine' => $addressLine, 'deliveryAddressPLZ' => $plz, 'deliveryAddressCity' => $city]);
|
||||
if (empty($orders)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// now check if the trackingNumber is already set and if not set it and create a history entry
|
||||
$order = (array) $orders[0];
|
||||
//
|
||||
if ($order['trackingNumber']) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$order['trackingNumber'] = $trackingNumber;
|
||||
WarehouseEShopOrderModel::update($order);
|
||||
|
||||
WarehouseHistoryModel::create(['table' => 'WarehouseEShopOrder',
|
||||
'row_id' => $order['id'],
|
||||
'key' => 'trackingNumber',
|
||||
'old_value' => '',
|
||||
'new_value' => $trackingNumber,
|
||||
'note' => '',
|
||||
'user_id' => 1,
|
||||
'create' => date('U')]);
|
||||
|
||||
die();
|
||||
|
||||
die($address);
|
||||
|
||||
echo "Subject: " . $overview[0]->subject . "\n";
|
||||
echo "From: " . $overview[0]->from . "\n";
|
||||
echo "Date: " . $overview[0]->date . "\n";
|
||||
echo "Message:\n" . $message . "\n\n";
|
||||
echo "---------------------------------------\n\n";
|
||||
}
|
||||
} else {
|
||||
echo "No emails found.";
|
||||
}
|
||||
|
||||
// Close the IMAP connection
|
||||
imap_close($mbox);
|
||||
die(); // do without ssl
|
||||
$mbox = imap_open("{mail.xino.at:143/novalidate-cert}INBOX", "eshop-versand@xinon.at", "savemanfb545aw");
|
||||
// print all mails in mailbox with content
|
||||
$mails = imap_search($mbox, 'ALL');
|
||||
$mails = array_reverse($mails);
|
||||
$mails = array_slice($mails, 0, 10);
|
||||
|
||||
$result = [];
|
||||
}
|
||||
}
|
||||
@@ -21,6 +21,7 @@ class WarehouseEShopOrderModel extends TTCrudBaseModel {
|
||||
public string $deliveryAddressLine;
|
||||
public string $deliveryAddressPLZ;
|
||||
public string $deliveryAddressCity;
|
||||
public string $trackingNumber;
|
||||
public int $create;
|
||||
public int $createBy;
|
||||
}
|
||||
@@ -0,0 +1,24 @@
|
||||
<?php /** @noinspection ALL */
|
||||
declare(strict_types=1);
|
||||
|
||||
use Phinx\Migration\AbstractMigration;
|
||||
|
||||
final class WarehouseAddEShopExtRef extends AbstractMigration {
|
||||
public function up(): void {
|
||||
if ($this->getEnvironment() == "thetool") {
|
||||
$WarehouseEShopOrder = $this->table("WarehouseEShopOrder");
|
||||
$WarehouseEShopOrder->addColumn("trackingNumber", "string", ["limit" => 255, "null" => true]);
|
||||
$WarehouseEShopOrder->changeColumn("status", "enum", ["values" => ["new", "accepted", "acceptedInternally", "sent", "done"], "null" => false]);
|
||||
$WarehouseEShopOrder->update();
|
||||
}
|
||||
}
|
||||
|
||||
public function down(): void {
|
||||
if ($this->getEnvironment() == "thetool") {
|
||||
$WarehouseEShopOrder = $this->table("WarehouseEShopOrder");
|
||||
$WarehouseEShopOrder->removeColumn("trackingNumber");
|
||||
$WarehouseEShopOrder->changeColumn("status", "enum", ["values" => ["new", "accepted", "sent", "done"], "null" => false]);
|
||||
$WarehouseEShopOrder->update();
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -3,8 +3,9 @@ Vue.component('warehouse-e-shop-order', {
|
||||
//language=Vue
|
||||
template: `
|
||||
<tt-card>
|
||||
<tt-table-crud @openHistory="historyModal = true; historyModalId = $event.id"
|
||||
<tt-table-crud @openHistory="historyModal = true; historyModalId = $event.id"
|
||||
@openSingleOrderEmail="openSingleOrderEmailModal = true; singleOrderEmailModalId = $event.id"
|
||||
@showTrackingHistory="openTrackingHistoryModal = true; trackingHistoryModalId = $event.id"
|
||||
ref="table">
|
||||
|
||||
<template v-slot:table-top-buttons>
|
||||
@@ -19,48 +20,71 @@ Vue.component('warehouse-e-shop-order', {
|
||||
</template>
|
||||
|
||||
<template v-slot:expandedRow="{ row }">
|
||||
<div>
|
||||
<ul class="list-group" v-if="articleItems && articleItems[row.id]">
|
||||
<li class="list-group-item" v-for="item in articleItems[row.id]">
|
||||
Menge: {{ item.quantity }} | {{ item.articlePacketTitle || item.articleTitle }}
|
||||
</li>
|
||||
</ul>
|
||||
<div>
|
||||
<ul class="list-group" v-if="articleItems && articleItems[row.id]">
|
||||
<li class="list-group-item" v-for="item in articleItems[row.id]">
|
||||
Menge: {{ item.quantity }} | {{ item.articlePacketTitle || item.articleTitle }}
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
|
||||
</tt-table-crud>
|
||||
<warehouse-history-modal :show.sync="historyModal" :id="historyModalId"/>
|
||||
|
||||
<!-- add a tt-modal which just shows some text from the api to show a email for a single order-->
|
||||
<tt-modal :show.sync="openSingleOrderEmailModal"
|
||||
:title="'Email für Bestellung ' + singleOrderEmailModalId"
|
||||
|
||||
<!-- add a tt-modal which just shows some text from the api to show a email for a single order-->
|
||||
<tt-modal :show.sync="openSingleOrderEmailModal"
|
||||
:title="'Email für Bestellung ' + singleOrderEmailModalId"
|
||||
@close="openSingleOrderEmailModal = false"
|
||||
save-text="E-Mail senden"
|
||||
@submit="sendSingleOrderEmail"
|
||||
:delete="false"
|
||||
>
|
||||
<div v-if="singleOrderEmailModalId && singleOrderEmailModalText">
|
||||
<p v-html="singleOrderEmailModalText.body.replaceAll('\\n', '<br>')"></p>
|
||||
<p v-html="singleOrderEmailModalText.body.replaceAll('\\n', '<br>')"></p>
|
||||
</div>
|
||||
<!-- else show loader-->
|
||||
<!-- else show loader-->
|
||||
<div v-else>
|
||||
<tt-loader/>
|
||||
<tt-loader/>
|
||||
</div>
|
||||
</tt-modal>
|
||||
|
||||
|
||||
|
||||
<tt-modal :show.sync="openTrackingHistoryModal"
|
||||
:title="'Tracking Historie für Bestellung ' + trackingHistoryModalId"
|
||||
@close="openTrackingHistoryModal = false"
|
||||
:delete="false"
|
||||
:save="false"
|
||||
>
|
||||
<div v-if="trackingHistoryModalData">
|
||||
<ul class="list-group">
|
||||
<li class="list-group item" v-for="entry in trackingHistoryModalData.history">
|
||||
<strong>{{ entry.date }} {{ entry.time }}</strong> - {{ entry.evtDscr }}
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
</tt-modal>
|
||||
|
||||
|
||||
</tt-card>
|
||||
`, data() {
|
||||
return {
|
||||
window: window, historyModal: false, historyModalId: null, articleItems: null, openSingleOrderEmailModal: false, singleOrderEmailModalId: null, singleOrderEmailModalText: null
|
||||
window: window,
|
||||
historyModal: false,
|
||||
historyModalId: null,
|
||||
articleItems: null,
|
||||
openSingleOrderEmailModal: false,
|
||||
singleOrderEmailModalId: null,
|
||||
singleOrderEmailModalText: null,
|
||||
openTrackingHistoryModal: false,
|
||||
trackingHistoryModalId: null,
|
||||
trackingHistoryModalData: null,
|
||||
}
|
||||
},
|
||||
async mounted() {
|
||||
}, async mounted() {
|
||||
const response = await axios.get(`${window['TT_CONFIG']['BASE_PATH']}/WarehouseEShopOrder/getAllItemsPerOrder`);
|
||||
console.log(response.data);
|
||||
this.articleItems = response.data;
|
||||
},
|
||||
methods: {
|
||||
}, methods: {
|
||||
async sendSingleOrderEmail() {
|
||||
const response = await axios.get(`${window['TT_CONFIG']['BASE_PATH']}/WarehouseEShopOrder/singleOrderEmail?id=${this.singleOrderEmailModalId}`);
|
||||
if (response.data.message) {
|
||||
@@ -70,8 +94,7 @@ Vue.component('warehouse-e-shop-order', {
|
||||
window.notify('error', 'Ein Fehler ist aufgetreten');
|
||||
}
|
||||
await this.$refs.table.$refs.table.fetchData();
|
||||
},
|
||||
async createCSVExportAndMarkAsAccepted() {
|
||||
}, async createCSVExportAndMarkAsAccepted() {
|
||||
const response = await axios.post(`${window['TT_CONFIG']['BASE_PATH']}/WarehouseEShopOrder/CSVExportNewOrdersMarkAccepted`);
|
||||
|
||||
if (response.data.message) {
|
||||
@@ -108,8 +131,7 @@ Vue.component('warehouse-e-shop-order', {
|
||||
|
||||
await this.$refs.table.$refs.table.fetchData();
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
}, watch: {
|
||||
singleOrderEmailModalId() {
|
||||
this.openSingleOrderEmailModal = !!this.singleOrderEmailModalId;
|
||||
this.singleOrderEmailModalText = null;
|
||||
@@ -118,6 +140,21 @@ Vue.component('warehouse-e-shop-order', {
|
||||
.then(response => this.singleOrderEmailModalText = response.data)
|
||||
.catch(error => window.notify('error', error.response?.data?.message || 'Ein Fehler ist aufgetreten'));
|
||||
}
|
||||
},
|
||||
async trackingHistoryModalId() {
|
||||
this.openTrackingHistoryModal = !!this.trackingHistoryModalId;
|
||||
this.trackingHistoryModalData = null;
|
||||
if (this.trackingHistoryModalId) {
|
||||
const response = await axios.get(`${window['TT_CONFIG']['BASE_PATH']}/WarehouseEShopOrder/fetchGLSTracking?id=${this.trackingHistoryModalId}`);
|
||||
const data = response.data;
|
||||
// if data is a empty string, show error
|
||||
if (!data) {
|
||||
window.notify('error', data.message || 'Keine Daten gefunden');
|
||||
return;
|
||||
}
|
||||
// if data is not empty, show data
|
||||
this.$set(this, 'trackingHistoryModalData', data);
|
||||
}
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
Reference in New Issue
Block a user