{
"require": {
"firebed/aade-mydata": "^5.0"
}
}
- Removed vat-registry dependency from composer. You can include vat-registry by running
composer require firebed/vat-registry
. See vat-registry documentation for more information. If you were not using vat search, you can safely ignore this change. - Removed ext-soap dependency from composer.
$invoice->squashInvoiceRows()
Ο Πάροχος ηλεκτρονικής τιμολόγησης και τα ERP διαβιβάζουν υποχρεωτικά μόνο τη σύνοψη γραμμών και χαρακτηρισμών των παραστατικών και όχι αναλυτικά τις γραμμές. Δείτε Σύνοψη Γραμμών Παραστατικού για περισσότερες λεπτομέρειες.
$invoice->validate()
$invoice->toXml()
Δείτε Συνδυασμοί Χαρακτηρισμών για περισσότερες λεπτομέρειες.
use Firebed\AadeMyData\Enums\InvoiceType;
use Firebed\AadeMyData\Services\Classifications;
dump(Classifications::incomeClassifications(InvoiceType::TYPE_1_1));
// array:9 [
// "category1_1" => array:3 [
// 0 => "E3_561_001"
// 1 => "E3_561_002"
// 2 => "E3_561_007"
// ]
// "category1_2" => array:3 [
// 0 => "E3_561_001"
// 1 => "E3_561_002"
// 2 => "E3_561_007"
// ]
// ...
// ]
// Alternative 2
IncomeClassificationType::for(InvoiceType::TYPE_1_1);
// Alternative 3
InvoiceType::TYPE_1_1->incomeClassifications();
use Firebed\AadeMyData\Enums\InvoiceType;
use Firebed\AadeMyData\Services\Classifications;
dump(Classifications::incomeClassifications(InvoiceType::TYPE_1_1)->toKeyLabel());
//array:9 [
// "category1_1" => "Έσοδα από Πώληση Εμπορευμάτων"
// "category1_2" => "Έσοδα από Πώληση Προϊόντων"
// "category1_3" => "Έσοδα από Παροχή Υπηρεσιών"
// "category1_4" => "Έσοδα από Πώληση Παγίων"
// "category1_5" => "Λοιπά Έσοδα / Κέρδη"
// "category1_7" => "Έσοδα για λογαριασμό τρίτων"
// "category1_8" => "Έσοδα προηγούμενων χρήσεων"
// "category1_9" => "Έσοδα επομένων χρήσεων"
// "category1_95" => "Λοιπά Πληροφοριακά Στοιχεία Εσόδων"
//]
dump(Classifications::incomeClassifications(InvoiceType::TYPE_1_1)->toKeyLabels())
//array:9 [
// "category1_1" => array:3 [
// "E3_561_001" => "Πωλήσεις αγαθών και υπηρεσιών Χονδρικές - Επιτηδευματιών"
// "E3_561_002" => "Πωλήσεις αγαθών και υπηρεσιών Χονδρικές βάσει άρθρου 39α παρ 5 του Κώδικα Φ.Π.Α. (Ν.2859/2000)"
// "E3_561_007" => "Πωλήσεις αγαθών και υπηρεσιών Λοιπά"
// ]
// "category1_2" => array:3 [
// "E3_561_001" => "Πωλήσεις αγαθών και υπηρεσιών Χονδρικές - Επιτηδευματιών"
// "E3_561_002" => "Πωλήσεις αγαθών και υπηρεσιών Χονδρικές βάσει άρθρου 39α παρ 5 του Κώδικα Φ.Π.Α. (Ν.2859/2000)"
// "E3_561_007" => "Πωλήσεις αγαθών και υπηρεσιών Λοιπά"
// ]
// "category1_3" => array:4 [
// "E3_561_001" => "Πωλήσεις αγαθών και υπηρεσιών Χονδρικές - Επιτηδευματιών"
// "E3_561_002" => "Πωλήσεις αγαθών και υπηρεσιών Χονδρικές βάσει άρθρου 39α παρ 5 του Κώδικα Φ.Π.Α. (Ν.2859/2000)"
// "E3_561_007" => "Πωλήσεις αγαθών και υπηρεσιών Λοιπά"
// "E3_563" => "Πιστωτικοί τόκοι και συναφή έσοδα"
// ]
// ...
// ]
dump(Classifications::incomeClassifications(InvoiceType::TYPE_1_1, IncomeClassificationCategory::CATEGORY_1_1)->toKeyLabel())
//array:3 [
// "E3_561_001" => "Πωλήσεις αγαθών και υπηρεσιών Χονδρικές - Επιτηδευματιών"
// "E3_561_002" => "Πωλήσεις αγαθών και υπηρεσιών Χονδρικές βάσει άρθρου 39α παρ 5 του Κώδικα Φ.Π.Α. (Ν.2859/2000)"
// "E3_561_007" => "Πωλήσεις αγαθών και υπηρεσιών Λοιπά"
//]
use Firebed\AadeMyData\Enums\InvoiceType;
use Firebed\AadeMyData\Enums\IncomeClassificationCategory;
use Firebed\AadeMyData\Enums\IncomeClassificationType;
use Firebed\AadeMyData\Services\Classifications;
Classifications::incomeClassificationExists('1.1', 'category1_1', 'E3_561_001');
// or
Classifications::incomeClassificationExists(InvoiceType::TYPE_1_1, IncomeClassificationCategory::CATEGORY_1_1, IncomeClassificationType::E3_561_001);
// Outputs: true
// Same for expense classifications
Classifications::expenseClassificationExists('1.1', 'category2_1', 'E3_102_001');
use Firebed\AadeMyData\Enums\InvoiceType;
use Firebed\AadeMyData\Enums\PaymentMethod;
use Firebed\AadeMyData\Enums\VatCategory;
use Firebed\AadeMyData\Enums\CountryCode;
echo InvoiceType::TYPE_1_1->label();
// Outputs: Τιμολόγιο Πώλησης
echo InvoiceType::TYPE_1_2->label();
// Outputs: Τιμολόγιο Πώλησης / Ενδοκοινοτικές Παραδόσεις
echo PaymentMethod::METHOD_5->label();
// Outputs: Επί Πιστώσει
echo VatCategory::VAT_1->label();
// Outputs: ΦΠΑ συντελεστής 24%
echo CountryCode::BE->label();
// Outputs: Βέλγιο
var_dump(InvoiceType::labels());
var_dump(PaymentMethod::labels());
var_dump(VatCategory::labels());
var_dump(CountryCode::labels());
use Firebed\AadeMyData\Enums\InvoiceType;
use Firebed\AadeMyData\Enums\CountryCode;
use Firebed\AadeMyData\Enums\ExpenseClassificationType;
$invoiceType = InvoiceType::TYPE_1_1;
$invoiceType->supportsFuelInvoice();
$invoiceType->hasCounterpart();
$invoiceType->supportsDeliveryNote();
$invoiceType->supportsSelfPricing();
$invoiceType->supportsTaxFree();
var_dump(CountryCode::europeanUnionCountries());
// Outputs: All countries in the European Union
echo CountryCode::BE->isInEuropeanUnion()
// Outputs: true
echo CountryCode::US->isInEuropeanUnion()
// Outputs: false
$type = ExpenseClassificationType::VAT_361;
echo $type->isVatClassification(); // true
var_dump(ExpenseClassificationType::vatClassifications()); // Array of all vat classifications
- CountryCode
- CurrencyCode
use Firebed\AadeMyData\Models\InvoiceDetails;
use Firebed\AadeMyData\Enums\RecType;
use Firebed\AadeMyData\Enums\IncomeClassificationType;
use Firebed\AadeMyData\Enums\IncomeClassificationCategory;
new InvoiceDetails([
'lineNumber' => 1,
'netValue' => 5,
'recType' => RecType::TYPE_2,
'incomeClassification' => [
[
'classificationType' => IncomeClassificationType::E3_561_001,
'classificationCategory' => IncomeClassificationCategory::CATEGORY_1_1,
'amount' => '5'
]
]
])
$invoice->setIssuer(...)->setCounterpart(...)
- Invoice::setTaxesTotals
- Invoice::setOtherTransportDetails
$row->addNetValue(5);
$row->addVatAmount(1.2);
- Fixed tax calculation when summarizing invoice.
- Fixed InvoiceDetails::setOtherMeasurementUnitQuantity
- Fixed InvoiceDetails::setOtherMeasurementUnitTitle
Η έκδοση 4.x περιέχει αρκετές αλλαγές, μεταξύ των οποίων αρκετές προσθήκες επιπλέον δυνατοτήτων.
Note
Η έκδοση 4.x είναι μια ολική αναδιοργάνωση και οι περισσότερες αλλαγές βρίσκονται στο εσωτερικό μέρος του συστήματος. Ωστόσο, ορισμένες αλλαγές ενδέχεται να επηρεάσουν τον τρόπο με τον οποίο χρησιμοποιείται το API.
{
"require": {
"firebed/aade-mydata": "^4.0"
}
}
- Μετονομασία
InvoicesDoc
::addInvoice()σεadd()
- Μετονομασία
RequestedDoc
::getInvoicesDoc()=>getInvoices()
- Μετονομασία
RequestedDoc
::getCancelledInvoicesDoc()=>getCancelledInvoices()
- Μετονομασία
RequestedDoc
::getIncomeClassificationsDoc()=>getIncomeClassifications()
- Μετονομασία
RequestedDoc
::getExpensesClassificationsDoc()=>getExpensesClassifications()
- Μετονομασία
RequestedDoc
::getPaymentMethodsDoc()=>getPaymentMethods()
- Μετονομασία της μεθόδου
put
σεset
για όλα μοντέλα που κληρονομούν την κλάσηFirebed\AadeMyData\Models\Type
- Τρόπος αντιμετώπισης των εξαιρέσεων (exceptions)
Οι μέθοδοι αλληλεπίδρασης με το σύστημα του myDATA όπως SendInvoices
, CancelInvoice
κ.λπ. πλέον
επιστρέφουν εξαιρέσεις τύπου Firebed\AadeMyData\Exceptions\MyDataException
ή εξαιρέσεις που
κληρονομούν αυτή την κλάση.
use Firebed\AadeMyData\Exceptions\MyDataException;
use Firebed\AadeMyData\Http\SendInvoices;
try {
$send = new SendInvoices();
$responses = $send->handle($invoices);
} catch (MyDataException $e) {
echo $e->getMessage();
}
Οι παρακάτω αλλαγές είναι εσωτερικές και δεν επηρεάζουν τη χρήση του API:
Ένα πράγμα που με κρατούσε ξύπνιο τα βράδια ήταν το γεγονός ότι οι παλαιές εκδόσεις του API δεν είχαν καθόλου δοκιμαστική κάλυψη. Όλες οι δοκιμές γινόντουσαν χειροκίνητα χωρίς επιβεβαίωση σωστής λειτουργίας μετά από κάθε αλλαγή στο κώδικα. Αυτό αλλάζει με την έκδοση 4.x, καθώς πλέον υπάρχει πλήρης δοκιμαστική κάλυψη 💥.
Στις προηγούμενες εκδόσεις το API χρησιμοποιούσε την κλάση SimpleXMLElement
για τη διαχείριση των xml δεδομένων. Αυτό αρχικά δεν ήταν πρόβλημα, αλλά με την αύξηση
των δυνατοτήτων και της πολυπλοκότητας των xml δεδομένων που έφερε το myDATA, η
SimpleXMLElement
έδειξε τα όριά της, καθώς έβρισκα τον εαυτό μου να εργάζομαι ενάντια σε
αυτήν, αναζητώντας τρόπους διαχείρισης των δεδομένων (ειδικά με τα namespaced tags).
Αυτή ίσως είναι η καλύτερη προσθήκη στο API. Η σύνοψη παραστατικού είναι μια περίπλοκη διαδικασία όπου πρέπει να υπολογίσουμε τα ποσά των φόρων, το καθαρό σύνολο, το τελικό σύνολο, τα σύνολα των χαρακτηρισμών εσόδων και εξόδων κ.λπ. Η δυσκολία επέρχεται όταν υπάρχουν διάφοροι τύποι φόρων και διάφοροι τύποι χαρακτηρισμών εσόδων/εξόδων, καθώς αυτά θα πρέπει να ομαδοποιούνται κατάλληλα έτσι ώστε να υπολογίσουμε σωστά τα απαραίτητα σύνολα. Αυτό και μόνο είναι πάνω από 200 γραμμές κώδικα.
Το χειρότερο (στη δικιά μου περίπτωση τουλάχιστον) είναι ότι έπρεπε να επαναλαμβάνω το ίδιο κομμάτι κώδικα σε κάθε πρότζεκτ που χρησιμοποιούσε αυτό το API. Αυτό μου γινόταν εφιάλτης, καθώς μετά από κάθε καινούργια έκδοση που έβγαζε το myDATA, έπρεπε να ελέγχω όλα μου τα πρότζεκτ συνεχώς 😒.
Στην έκδοση 4.x, η σύνοψη παραστατικού είναι πλέον μια μέθοδος summarizeInvoice
της κλάσης
Firebed\AadeMyData\Models\Invoice
. Λαμβάνει υπόψιν όλες τις παραμέτρους του παραστατικού
και υπολογίζει τα απαραίτητα σύνολα αυτόματα χωρίς να χρειάζεται να κάνετε τίποτα άλλο 😀.
use Firebed\AadeMyData\Models\Invoice;
$invoice = new Invoice();
// ... set some details
$invoice->summarizeInvoice();
Αυτή ίσως είναι η 2η αγαπημένη μου προσθήκη. Όλα XML που χρησιμοποιεί το myDATA βασίζονται σε xsd (xml schema) τα οποία καθορίζουν τη δομή του xml και τη σειρά των πεδίων. Αυτό σημαίνει ότι τα πεδία του xml πρέπει να είναι ταξινομημένα σύμφωνα με το σχήμα.
Αυτό ήταν πρόβλημα καθώς παλαιότερα το API έβαζε τα πεδία στο xml με τη σειρά που τα πρόσθετε ο χρήστης
και αυτό οδηγούσε σε λάθη XMLSyntaxError
κατά την υποβολή των xml στο myDATA.
Παράδειγμα λανθασμένης σειράς στην έκδοση ≤ 3.x
use Firebed\AadeMyData\Models\Invoice;
use Firebed\AadeMyData\Models\Counterpart;
use Firebed\AadeMyData\Models\Issuer;
use Firebed\AadeMyData\Http\SendInvoices;
$invoice = new Invoice();
// Counterpart must come after Issuer
$invoice->setCounterpart(new Counterpart());
// Issuer must come before Counterpart
$invoice->setIssuer(new Counterpart());
$send = new SendInvoices();
$send->handle($invoice);
Σύμφωνα με το xsd ο issuer
πρέπει να έρχεται πριν το counterpart
, ενώ παραπάνω τα βάλαμε ανάποδα και αυτό
είναι το αποτέλεσμα της εκτέλεσης:
<response>
<index>1</index>
<statusCode>XMLSyntaxError</statusCode>
<errors>
<error>
<message>Line:13.Position:6.The element 'invoice' in namespace 'http://www.aade.gr/myDATA/invoice/v1.0' has invalid child element 'issuer' in namespace 'http://www.aade.gr/myDATA/invoice/v1.0'. List of possible elements expected: 'invoiceHeader' in namespace 'http://www.aade.gr/myDATA/invoice/v1.0'.</message>
<code>101</code>
</error>
</errors>
</response>
Tip
Από την έκδοση 4.x και μετά η σωστή σειρά των πεδίων παρακολουθείται εσωτερικά, και έτσι άσχετα με τη σειρά που δομείτε το παραστατικό σας το τελικό xml θα έχει πάντα τη σωστή ταξινόμηση.
"If I had a dollar for everytime someone asked me "How do I use this?" I'd be a millionaire by now 💰."
Εντάξει, όχι και millionaire. Λαμβάνω όμως συνέχεια ερωτήσεις για το πως μπορεί κάποιος να χρησιμοποιεί το API. Αρχικά νόμιζα πως το 1 README.md αρχείο στο GitHub θα ήταν αρκετό, τελικά όμως αποδείχτηκε και ομολογώ πως δεν ήταν.
Τώρα αντί για 1 README.md
αρχείο έχουμε πάνω από 60 .md αρχεία, το καθένα αφιερωμένο σε ένα μέρος του API.
Επιπλέον, το subdomain docs.invoicemaker.gr περιέχει όλα τα αρχεία αυτά απ' ευθείας
από το GitHub σε πιο οργανωμένη μορφή για εύκολη πλοήγηση 🥳.