mirror of
https://github.com/OpenXE-org/OpenXE.git
synced 2025-01-07 12:30:28 +01:00
476 lines
14 KiB
PHP
476 lines
14 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Class USTID
|
|
*
|
|
* echo $this->check("DE263136143","SE556459933901","Wind River AB","Kista","Finlandsgatan 52","16493","ja");
|
|
* echo $this->check("DE263136143","HU12925481","TGIF KFT.","Egerszaloki","3394","Szechenyiu","ja");
|
|
* echo $this->check("DE263136143","SE556459933901","jkjk","Kista","Finlandsgatan 52","16493","ja");
|
|
*/
|
|
class USTID
|
|
{
|
|
/**
|
|
* online formular @https://evatr.bff-online.de/eVatR/index_html#Einfach_Ergebnis
|
|
*
|
|
* @var string API base
|
|
* @var string API endpoint
|
|
*/
|
|
const BASE = 'https://evatr.bff-online.de';
|
|
const ENDPOINT = 'evatrRPC';
|
|
|
|
/**
|
|
* just for downward compatibility,
|
|
* the classes 'Adresse' and the 'erpAPI'
|
|
* use that public value to access the
|
|
* response data
|
|
*
|
|
* @var array
|
|
*/
|
|
var $answer = array(
|
|
'UstId_1' => '',
|
|
'UstId_2' => '',
|
|
'ErrorCode' => '',
|
|
'Druck' => '',
|
|
'Erg_PLZ' => '',
|
|
'Ort' => '',
|
|
'Datum' => '',
|
|
'PLZ' => '',
|
|
'Erg_Ort' => '',
|
|
'Uhrzeit' => '',
|
|
'Erg_Name' => '',
|
|
'Gueltig_ab' => '',
|
|
'Gueltig_bis' => '',
|
|
'Strasse' => '',
|
|
'Firmenname' => '',
|
|
'Erg_Str' => '',
|
|
'ErrorMSG' => '',
|
|
'OK' => '',
|
|
);
|
|
|
|
/**
|
|
* the required GET-parameter,
|
|
* leave blank if field is not necessary,
|
|
* but don't remove it from the request.
|
|
* @see: https://evatr.bff-online.de/eVatR/xmlrpc/schnittstelle
|
|
*
|
|
* @var array
|
|
*/
|
|
private $_data = array(
|
|
'UstId_1' => '', // Ihre deutsche USt-IdNr.
|
|
'UstId_2' => '', // Anzufragende ausländische USt-IdNr.
|
|
'Firmenname' => '', // Name der anzufragenden Firma einschl. Rechtsform
|
|
'Ort' => '', // Ort der anzufragenden Firma
|
|
'PLZ' => '', // Postleitzahl der anzufragenden Firma
|
|
'Strasse' => '', // Strasse und Hausnummer der anzufragenden Firma,
|
|
'Druck' => 'nein' // ja = mit amtlicher Bestätigungsmitteilung
|
|
);
|
|
|
|
/**
|
|
* @see https://evatr.bff-online.de/eVatR/xmlrpc/codes
|
|
* @var array $_errorCodes
|
|
*/
|
|
private $_errorCodes = array(
|
|
/*
|
|
* my own
|
|
* Kein gültiger Fehlercode vom Finanzamt-Server erhalten.
|
|
*/
|
|
100 => 'Es wurde keine Antwort vom Finanzamt-Server erhalten. Nur in der Zeit zwischen 05:00 Uhr und 23:00 Uhr sind abfragen möglich',
|
|
101 => 'Es wurde eine Fehlerhafte Antwort vom Finanzamt-Server erhalten',
|
|
/*
|
|
* API's status/error codes
|
|
*/
|
|
200 => 'Die angefragte USt-IdNr. ist gültig.',
|
|
201 => 'Die angefragte USt-IdNr. ist ungültig.',
|
|
202 => 'Die angefragte USt-IdNr. ist ungültig. Sie ist nicht in der Unternehmerdatei des betreffenden EU-Mitgliedstaates registriert. (Hinweis: Ihr Geschäftspartner kann seine gültige USt-IdNr. bei der für ihn zuständigen Finanzbehörde in Erfahrung bringen. Möglicherweise muss er einen Antrag stellen, damit seine USt-IdNr. in die Datenbank aufgenommen wird.)',
|
|
203 => 'Die angefragte USt-IdNr. ist ungültig. Sie ist erst ab dem {{Gueltig_ab}} gültig.',
|
|
204 => 'Die angefragte USt-IdNr. ist ungültig. Sie war im Zeitraum von {{Gueltig_ab}} bis {{Gueltig_bis}} gültig.',
|
|
205 => 'Ihre Anfrage kann derzeit durch den angefragten EU-Mitgliedstaat oder aus anderen Gründen nicht beantwortet werden.',
|
|
206 => 'Ihre deutsche USt-IdNr. ist ungültig. Eine Bestätigungsanfrage ist daher nicht möglich.',
|
|
207 => 'Ihnen wurde die deutsche USt-IdNr. ausschliesslich zu Zwecken der Besteuerung des innergemeinschaftlichen Erwerbs erteilt. Sie sind somit nicht berechtigt, Bestätigungsanfragen zu stellen.',
|
|
208 => 'Für die von Ihnen angefragte USt-IdNr. läuft gerade eine Anfrage von einem anderen Nutzer. Eine Bearbeitung ist daher nicht möglich. Bitte versuchen Sie es später noch einmal.',
|
|
209 => 'Die angefragte USt-IdNr. ist ungültig. Sie entspricht nicht dem Aufbau der für diesen EU-Mitgliedstaat gilt.',
|
|
210 => 'Die angefragte USt-IdNr. ist ungültig. Sie entspricht nicht den Prüfziffernregeln die für diesen EU-Mitgliedstaat gelten.',
|
|
211 => 'Die angefragte USt-IdNr. ist ungültig. Sie enthält unzulässige Zeichen (wie z.B. Leerzeichen oder Punkt oder Bindestrich usw.).',
|
|
212 => 'Die angefragte USt-IdNr. ist ungültig. Sie enthält ein unzulässiges Länderkennzeichen.',
|
|
213 => 'Die Abfrage einer deutschen USt-IdNr. ist nicht möglich.',
|
|
214 => 'Ihre deutsche USt-IdNr. ist fehlerhaft. Sie beginnt mit \'DE\' gefolgt von 9 Ziffern.',
|
|
215 => 'Ihre Anfrage enthält nicht alle notwendigen Angaben für eine einfache Bestätigungsanfrage (Ihre deutsche USt-IdNr. und die ausl. USt-IdNr.).',
|
|
216 => 'Ihre Anfrage enthält nicht alle notwendigen Angaben für eine qualifizierte Bestätigungsanfrage. (Ihre deutsche USt-IdNr., die ausl. USt-IdNr., Firmenname einschl. Rechtsform und Ort).',
|
|
217 => 'Bei der Verarbeitung der Daten aus dem angefragten EU-Mitgliedstaat ist ein Fehler aufgetreten.',
|
|
218 => 'Eine qualifizierte Bestätigung ist zur Zeit nicht möglich.',
|
|
219 => 'Bei der Durchführung der qualifizierten Bestätigungsanfrage ist ein Fehler aufgetreten. Die angefragte USt-IdNr. ist gültig.',
|
|
220 => 'Bei der Anforderung der amtlichen Bestätigungsmitteilung ist ein Fehler aufgetreten. Sie werden kein Schreiben erhalten.',
|
|
221 => 'Die Anfragedaten enthalten nicht alle notwendigen Parameter oder einen ungültigen Datentyp.',
|
|
222 => 'Die angefragte USt-IdNr. ist gültig.',
|
|
999 => 'Eine Bearbeitung Ihrer Anfrage ist zurzeit nicht möglich. Bitte versuchen Sie es später noch einmal.'
|
|
);
|
|
|
|
/**
|
|
* the last error message
|
|
* @var string
|
|
*/
|
|
private $_error = '';
|
|
|
|
/**
|
|
* @param string $ust1 your UST ID
|
|
* @param string $ust2 requested UST ID
|
|
* @param string $firmenname
|
|
* @param string $ort
|
|
* @param string $strasse
|
|
* @param string $plz
|
|
* @param string $druck
|
|
* @param $onlinefehler
|
|
* @return int
|
|
*
|
|
* return values:
|
|
* 1: ok,
|
|
* -1: invalid/wrong response,
|
|
* -2: curl error
|
|
*/
|
|
function check($ust1, $ust2, $firmenname, $ort, $strasse, $plz, $druck = "nein", &$onlinefehler)
|
|
{
|
|
$this->set('UstId_1', $ust1);
|
|
$this->set('UstId_2', $ust2);
|
|
$this->set('Firmenname', $firmenname);
|
|
$this->set('Ort', $ort);
|
|
$this->set('Strasse', $strasse);
|
|
$this->set('PLZ', $plz);
|
|
$this->set('Druck', $druck);
|
|
|
|
/*
|
|
* we could use $response = $this->send();
|
|
* send() returns only false or the response.
|
|
* To detect the reason of an error, don't use it.
|
|
*/
|
|
|
|
$url = $this->buildQueryURL();
|
|
$response = $this->curl($url);
|
|
|
|
if ($response == false) {
|
|
/**
|
|
* curl failed,
|
|
*
|
|
*/
|
|
$onlinefehler = $this->_error;
|
|
return -2;
|
|
}
|
|
|
|
$response = $this->convertXML2Array($response);
|
|
$this->answer = $response;
|
|
$this->_error = $response['ErrorMSG'];
|
|
$onlinefehler = $this->_error;
|
|
|
|
return $response['OK'] == true ? 1 : -1;
|
|
}
|
|
|
|
/**
|
|
* from erpAPI (class.erpapi.php) to simplify that class
|
|
* from first version of USTID, still unused?
|
|
*/
|
|
function checkAndSendMailIfWrong()
|
|
{
|
|
// Job: implement
|
|
// this method was empty
|
|
}
|
|
|
|
/**
|
|
* just for downward compatibility,
|
|
* the class 'Adresse' use that method
|
|
* to get the error code message
|
|
*
|
|
* @param $code
|
|
* @return string
|
|
*/
|
|
function errormessages($code)
|
|
{
|
|
return $this->generateErrorMsg($code);
|
|
}
|
|
|
|
/**
|
|
* return the error code message
|
|
* alias for errormessages
|
|
*
|
|
* @param $code
|
|
* @return string
|
|
*/
|
|
public function getErrorMsg($code)
|
|
{
|
|
return $this->generateErrorMsg($code);
|
|
}
|
|
|
|
/**
|
|
* create the full url and send the request
|
|
*
|
|
* @param array $args
|
|
* @return mixed
|
|
*/
|
|
public function send($args = array())
|
|
{
|
|
if (is_array($args) && !empty($args)) {
|
|
foreach ($args as $key => $value) {
|
|
$this->set($key, $value);
|
|
}
|
|
}
|
|
if (!empty($this->_error)) {
|
|
/**
|
|
* on error, don't send the request,
|
|
* let the user change the settings
|
|
*/
|
|
return false;
|
|
}
|
|
|
|
$url = $this->buildQueryURL();
|
|
$response = $this->curl($url);
|
|
if ($response == false) {
|
|
/**
|
|
* curl failed,
|
|
*
|
|
*/
|
|
return false;
|
|
}
|
|
$response = $this->convertXML2Array($response);
|
|
$this->answer = $response;
|
|
$this->_error = $response['ErrorMSG'];
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* just a little helper to reset the request
|
|
*/
|
|
public function reset()
|
|
{
|
|
foreach ($this->_data as $key => $_) {
|
|
$this->_data[$key] = '';
|
|
}
|
|
}
|
|
|
|
/**
|
|
* set an GET parameter,
|
|
* only the keys from $_data array are allowed,
|
|
* only string values are allowed
|
|
*
|
|
* @param string $key
|
|
* @param string $value
|
|
* @return bool
|
|
*/
|
|
public function set($key, $value)
|
|
{
|
|
if (!is_string($value)) {
|
|
$this->_error = 'Invalid value type given';
|
|
return false;
|
|
}
|
|
if (!is_string($key)) {
|
|
$this->_error = 'Invalid key type given';
|
|
return false;
|
|
}
|
|
$keys = array_keys($this->_data);
|
|
if (!in_array($key, $keys)) {
|
|
$this->_error = "Key [{$key}] is not allowed";
|
|
return false;
|
|
}
|
|
$this->_data[$key] = $value;
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* build the query url and reset the config
|
|
*
|
|
* @return string
|
|
*/
|
|
private function buildQueryURL()
|
|
{
|
|
$args = $this->formatData($this->_data);
|
|
$args = http_build_query($args);
|
|
|
|
$url = self::BASE . '/' . self::ENDPOINT . '?' . $args;
|
|
$this->reset();
|
|
|
|
return $url;
|
|
}
|
|
|
|
/**
|
|
* before curl, reformat the data.
|
|
* validate the 'Druck' value,
|
|
* remove blanks from the UstId
|
|
*
|
|
* @param array $data
|
|
* @return array
|
|
*/
|
|
private function formatData($data)
|
|
{
|
|
if (!$data['Druck'] == 'ja') {
|
|
/*
|
|
* the 'Druck' value was not set to 'ja',
|
|
* so empty that value
|
|
*/
|
|
$data['Druck'] = '';
|
|
}
|
|
foreach (array('UstId_1', 'UstId_2') as $key) {
|
|
/*
|
|
* unzulässige Zeichen (wie z.B. Leerzeichen oder Punkt oder Bindestrich usw.)
|
|
*/
|
|
$value = $data[$key];
|
|
$value = str_replace(" ", "", $value);
|
|
$value = str_replace(".", "", $value);
|
|
$value = str_replace("-", "", $value);
|
|
$data[$key] = $value;
|
|
}
|
|
|
|
return $data;
|
|
}
|
|
|
|
/**
|
|
* curl the given url
|
|
* @param $url
|
|
* @return mixed
|
|
*/
|
|
private function curl($url)
|
|
{
|
|
$ch = curl_init($url);
|
|
|
|
curl_setopt_array($ch, array(
|
|
CURLOPT_SSL_VERIFYPEER => false,
|
|
CURLOPT_SSLVERSION => 6,
|
|
CURLOPT_POST => false,
|
|
CURLOPT_RETURNTRANSFER => true,
|
|
));
|
|
|
|
$result = curl_exec($ch);
|
|
if (!$result) {
|
|
$this->_error = 'curl error: ' . curl_error($ch);
|
|
return false;
|
|
}
|
|
|
|
$info = curl_getinfo($ch);
|
|
|
|
curl_close($ch);
|
|
|
|
$httpCode = $info['http_code'];
|
|
if ($httpCode != 200 && $httpCode != 222) {
|
|
$this->_error = 'Konnte Finanzamt-Server nicht erreichen (HTTP status code: ' . $httpCode . ')';
|
|
return false;
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
|
|
/**
|
|
* convert the xml to array
|
|
*
|
|
* @param $xml
|
|
* @return array
|
|
*/
|
|
private function convertXML2Array($xml)
|
|
{
|
|
if (!$xml) {
|
|
return array(
|
|
'ErrorMSG' => $this->_errorCodes[100],
|
|
'ErrorCode' => 100
|
|
);
|
|
}
|
|
|
|
$xml = simplexml_load_string($xml);
|
|
if ($xml === false) {
|
|
return array(
|
|
'ErrorMSG' => $this->_errorCodes[101],
|
|
'ErrorCode' => 101
|
|
);
|
|
}
|
|
|
|
$response = array();
|
|
if (isset($xml->param)) {
|
|
foreach ($xml->param as $param) {
|
|
if (!isset($param->value->array->data->value)) {
|
|
continue;
|
|
}
|
|
$key = (string)$param->value->array->data->value[0]->string;
|
|
$value = (string)$param->value->array->data->value[1]->string;
|
|
$response[$key] = is_numeric($value) ? ((int)$value) : $value;
|
|
}
|
|
}
|
|
$response = $this->extendResponse($response);
|
|
return $response;
|
|
}
|
|
|
|
/**
|
|
* extend the 'ErrorMSG' and the 'OK' field
|
|
*
|
|
* @param array $response
|
|
* @return array
|
|
*/
|
|
private function extendResponse($response)
|
|
{
|
|
$response = (array)$response;
|
|
|
|
/**
|
|
* let's add the error msg from the error codes 'table'
|
|
*/
|
|
$response['ErrorMSG'] = 'Kein gültiger Fehlercode vom Finanzamt-Server erhalten.';
|
|
|
|
if (array_key_exists('ErrorCode', $response)) {
|
|
$error = (int)$response['ErrorCode'];
|
|
$response['ErrorMSG'] = $this->generateErrorMsg($error, $response);
|
|
$response['OK'] = $response['ErrorCode'] == 200;
|
|
} else {
|
|
$response['ErrorMSG'] = 'Kein gültiger Fehlercode vom Finanzamt-Server erhalten.';
|
|
$response['OK'] = false;
|
|
}
|
|
return $response;
|
|
}
|
|
|
|
/**
|
|
* @param $code
|
|
* @param $env
|
|
* @return string
|
|
*/
|
|
private function generateErrorMsg($code, $env = array())
|
|
{
|
|
if (!array_key_exists($code, $this->_errorCodes)) {
|
|
return 'Kein gültiger Fehlercode vom Finanzamt-Server erhalten.';
|
|
}
|
|
|
|
$msg = $this->_errorCodes[$code];
|
|
|
|
if (empty($env) || !is_array($env)) {
|
|
$rep = array('{{' => '(Siehe Feld: ', '}}' => ')');
|
|
return strtr($msg, $rep);
|
|
}
|
|
|
|
/**
|
|
* extracts the placeholders in the msg string.
|
|
* E.g.: Sie war im Zeitraum von {{Gueltig_ab}} bis {{Gueltig_bis}} gültig.
|
|
* converts to:
|
|
* array(
|
|
* 0 => array(
|
|
* 0 => '{{Gueltig_ab}}'
|
|
* 1 => '{{Gueltig_bis}}'
|
|
* )
|
|
* 1 => array(
|
|
* 0 => Gueltig_ab
|
|
* 1 => Gueltig_bis
|
|
* )
|
|
* )
|
|
*/
|
|
preg_match_all("/\{\{([^}]+)\}\}/", $msg, $keys);
|
|
/*
|
|
* filter empty results:
|
|
*/
|
|
$keys = array_filter($keys);
|
|
|
|
/*
|
|
* for each key in the extract, replace the string
|
|
*/
|
|
$rep = array();
|
|
if (is_array($keys) && (!empty($keys)?count($keys):0) == 2) {
|
|
foreach ($keys[1] as $key) {
|
|
if (!array_key_exists($key, $env)) {
|
|
continue;
|
|
}
|
|
$rep['{{' . $key . '}}'] = $env[$key];
|
|
}
|
|
$msg = strtr($msg, $rep);
|
|
}
|
|
return $msg;
|
|
}
|
|
}
|