690 lines
27 KiB
PHP
690 lines
27 KiB
PHP
<?php
|
|
/**
|
|
*--------------------------------------------------------------------
|
|
*
|
|
* Calculate the GS1-128 based on the Code-128 encoding.
|
|
*
|
|
*--------------------------------------------------------------------
|
|
* @author Akhtar Khan <er.akhtarkhan@gmail.com>
|
|
* @link http://www.codeitnow.in
|
|
* @package https://github.com/codeitnowin/barcode-generator
|
|
*/
|
|
namespace CodeItNow\BarcodeBundle\Generator;
|
|
use CodeItNow\BarcodeBundle\Generator\CINParseException;
|
|
use CodeItNow\BarcodeBundle\Generator\CINcode128;
|
|
|
|
class CINgs1128 extends CINcode128 {
|
|
const KIND_OF_DATA = 0;
|
|
const MINLENGTH = 1;
|
|
const MAXLENGTH = 2;
|
|
const CHECKSUM = 3;
|
|
const NUMERIC = 0;
|
|
const ALPHA_NUMERIC = 1;
|
|
const DATE_YYMMDD = 2;
|
|
const ID = 0;
|
|
const CONTENT = 1;
|
|
const MAX_ID_FORMATED = 6;
|
|
const MAX_ID_NOT_FORMATED = 4;
|
|
const MAX_GS1128_CHARS = 48;
|
|
|
|
private $strictMode;
|
|
private $allowsUnknownIdentifier;
|
|
private $noLengthLimit;
|
|
private $identifiersId = array();
|
|
private $identifiersContent = array();
|
|
private $identifiersAi = array();
|
|
private $customLabelText=false;
|
|
|
|
/**
|
|
* Constructors
|
|
*
|
|
* @param char $start
|
|
*/
|
|
public function __construct($start = null) {
|
|
if ($start === null) {
|
|
$start = 'C';
|
|
}
|
|
|
|
parent::__construct($start);
|
|
|
|
/* Application Identifiers (AIs) */
|
|
/*
|
|
array ( KIND_OF_DATA , MINLENGTH , MAXLENGTH , CHECKSUM )
|
|
KIND_OF_DATA: NUMERIC , ALPHA_NUMERIC or DATE_YYMMDD
|
|
CHECKSUM: bool (true / false)
|
|
*/
|
|
$this->identifiersAi = array(
|
|
'00' => array(self::NUMERIC, 18, 18, true),
|
|
'01' => array(self::NUMERIC, 14, 14, true),
|
|
'02' => array(self::NUMERIC, 14, 14, true),
|
|
'10' => array(self::ALPHA_NUMERIC, 1, 20, false),
|
|
'11' => array(self::DATE_YYMMDD, 6, 6, false),
|
|
'12' => array(self::DATE_YYMMDD, 6, 6, false),
|
|
'13' => array(self::DATE_YYMMDD, 6, 6, false),
|
|
'15' => array(self::DATE_YYMMDD, 6, 6, false),
|
|
'17' => array(self::DATE_YYMMDD, 6, 6, false),
|
|
'20' => array(self::NUMERIC, 2, 2, false),
|
|
'21' => array(self::ALPHA_NUMERIC, 1, 20, false),
|
|
'240' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'241' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'250' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'251' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'253' => array(self::NUMERIC, 14, 30, false),
|
|
'30' => array(self::NUMERIC, 1, 8, false),
|
|
'310y' => array(self::NUMERIC, 6, 6, false),
|
|
'311y' => array(self::NUMERIC, 6, 6, false),
|
|
'312y' => array(self::NUMERIC, 6, 6, false),
|
|
'313y' => array(self::NUMERIC, 6, 6, false),
|
|
'314y' => array(self::NUMERIC, 6, 6, false),
|
|
'315y' => array(self::NUMERIC, 6, 6, false),
|
|
'316y' => array(self::NUMERIC, 6, 6, false),
|
|
'320y' => array(self::NUMERIC, 6, 6, false),
|
|
'321y' => array(self::NUMERIC, 6, 6, false),
|
|
'322y' => array(self::NUMERIC, 6, 6, false),
|
|
'323y' => array(self::NUMERIC, 6, 6, false),
|
|
'324y' => array(self::NUMERIC, 6, 6, false),
|
|
'325y' => array(self::NUMERIC, 6, 6, false),
|
|
'326y' => array(self::NUMERIC, 6, 6, false),
|
|
'327y' => array(self::NUMERIC, 6, 6, false),
|
|
'328y' => array(self::NUMERIC, 6, 6, false),
|
|
'329y' => array(self::NUMERIC, 6, 6, false),
|
|
'330y' => array(self::NUMERIC, 6, 6, false),
|
|
'331y' => array(self::NUMERIC, 6, 6, false),
|
|
'332y' => array(self::NUMERIC, 6, 6, false),
|
|
'333y' => array(self::NUMERIC, 6, 6, false),
|
|
'334y' => array(self::NUMERIC, 6, 6, false),
|
|
'335y' => array(self::NUMERIC, 6, 6, false),
|
|
'336y' => array(self::NUMERIC, 6, 6, false),
|
|
'337y' => array(self::NUMERIC, 6, 6, false),
|
|
'340y' => array(self::NUMERIC, 6, 6, false),
|
|
'341y' => array(self::NUMERIC, 6, 6, false),
|
|
'342y' => array(self::NUMERIC, 6, 6, false),
|
|
'343y' => array(self::NUMERIC, 6, 6, false),
|
|
'344y' => array(self::NUMERIC, 6, 6, false),
|
|
'345y' => array(self::NUMERIC, 6, 6, false),
|
|
'346y' => array(self::NUMERIC, 6, 6, false),
|
|
'347y' => array(self::NUMERIC, 6, 6, false),
|
|
'348y' => array(self::NUMERIC, 6, 6, false),
|
|
'349y' => array(self::NUMERIC, 6, 6, false),
|
|
'350y' => array(self::NUMERIC, 6, 6, false),
|
|
'351y' => array(self::NUMERIC, 6, 6, false),
|
|
'352y' => array(self::NUMERIC, 6, 6, false),
|
|
'353y' => array(self::NUMERIC, 6, 6, false),
|
|
'354y' => array(self::NUMERIC, 6, 6, false),
|
|
'355y' => array(self::NUMERIC, 6, 6, false),
|
|
'356y' => array(self::NUMERIC, 6, 6, false),
|
|
'357y' => array(self::NUMERIC, 6, 6, false),
|
|
'360y' => array(self::NUMERIC, 6, 6, false),
|
|
'361y' => array(self::NUMERIC, 6, 6, false),
|
|
'362y' => array(self::NUMERIC, 6, 6, false),
|
|
'363y' => array(self::NUMERIC, 6, 6, false),
|
|
'364y' => array(self::NUMERIC, 6, 6, false),
|
|
'365y' => array(self::NUMERIC, 6, 6, false),
|
|
'366y' => array(self::NUMERIC, 6, 6, false),
|
|
'367y' => array(self::NUMERIC, 6, 6, false),
|
|
'368y' => array(self::NUMERIC, 6, 6, false),
|
|
'369y' => array(self::NUMERIC, 6, 6, false),
|
|
'37' => array(self::NUMERIC, 1, 8, false),
|
|
'390y' => array(self::NUMERIC, 1, 15, false),
|
|
'391y' => array(self::NUMERIC, 4, 18, false),
|
|
'392y' => array(self::NUMERIC, 1, 15, false),
|
|
'393y' => array(self::NUMERIC, 4, 18, false),
|
|
'400' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'401' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'402' => array(self::NUMERIC, 17, 17, false),
|
|
'403' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'410' => array(self::NUMERIC, 13, 13, true),
|
|
'411' => array(self::NUMERIC, 13, 13, true),
|
|
'412' => array(self::NUMERIC, 13, 13, true),
|
|
'413' => array(self::NUMERIC, 13, 13, true),
|
|
'414' => array(self::NUMERIC, 13, 13, true),
|
|
'415' => array(self::NUMERIC, 13, 13, true),
|
|
'420' => array(self::ALPHA_NUMERIC, 1, 20, false),
|
|
'421' => array(self::ALPHA_NUMERIC, 4, 12, false),
|
|
'422' => array(self::NUMERIC, 3, 3, false),
|
|
'8001' => array(self::NUMERIC, 14, 14, false),
|
|
'8002' => array(self::ALPHA_NUMERIC, 1, 20, false),
|
|
'8003' => array(self::ALPHA_NUMERIC, 15, 30, false),
|
|
'8004' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'8005' => array(self::NUMERIC, 6, 6, false),
|
|
'8006' => array(self::NUMERIC, 18, 18, false),
|
|
'8007' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'8018' => array(self::NUMERIC, 18, 18, false),
|
|
'8020' => array(self::ALPHA_NUMERIC, 1, 25, false),
|
|
'8100' => array(self::NUMERIC, 6, 6, false),
|
|
'8101' => array(self::NUMERIC, 10, 10, false),
|
|
'8102' => array(self::NUMERIC, 2, 2, false),
|
|
'90' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'91' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'92' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'93' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'94' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'95' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'96' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'97' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'98' => array(self::ALPHA_NUMERIC, 1, 30, false),
|
|
'99' => array(self::ALPHA_NUMERIC, 1, 30, false)
|
|
);
|
|
|
|
$this->setStrictMode(true);
|
|
$this->setTilde(true);
|
|
$this->setAllowsUnknownIdentifier(false);
|
|
$this->setNoLengthLimit(false);
|
|
}
|
|
|
|
/**
|
|
* Gets the content checksum for an identifier.
|
|
* Do not pass the identifier code.
|
|
*
|
|
* @param string $content
|
|
* @return int
|
|
*/
|
|
public static function getAiContentChecksum($content) {
|
|
return self::calculateChecksumMod10($content);
|
|
}
|
|
|
|
/**
|
|
* Enables or disables the strict mode.
|
|
*
|
|
* @param bool $strictMode
|
|
*/
|
|
public function setStrictMode($strictMode) {
|
|
$this->strictMode = $strictMode;
|
|
}
|
|
|
|
/**
|
|
* Gets if the strict mode is activated.
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function getStrictMode() {
|
|
return $this->strictMode;
|
|
}
|
|
|
|
/**
|
|
* Allows unknown identifiers.
|
|
*
|
|
* @param bool $allow
|
|
*/
|
|
public function setAllowsUnknownIdentifier($allow) {
|
|
$this->allowsUnknownIdentifier = (bool)$allow;
|
|
}
|
|
|
|
/**
|
|
* Gets if unkmown identifiers are allowed.
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function getAllowsUnknownIdentifier() {
|
|
return $this->allowsUnknownIdentifier;
|
|
}
|
|
|
|
/**
|
|
* Removes the limit of 48 characters.
|
|
*
|
|
* @param bool $noLengthLimit
|
|
*/
|
|
public function setNoLengthLimit($noLengthLimit) {
|
|
$this->noLengthLimit = (bool)$noLengthLimit;
|
|
}
|
|
|
|
/**
|
|
* Gets if the limit of 48 characters is removed.
|
|
*
|
|
* @return bool
|
|
*/
|
|
public function getNoLengthLimit() {
|
|
return $this->noLengthLimit;
|
|
}
|
|
|
|
public function setLabel($label){
|
|
$this->customLabelText = $label;
|
|
}
|
|
|
|
/**
|
|
* Parses Text.
|
|
*
|
|
* @param string $text
|
|
*/
|
|
public function parse($text) {
|
|
$parsedText = $this->parseGs1128($text);
|
|
if($this->customLabelText!==false and CINBarcode1D::AUTO_LABEL!=$this->customLabelText){
|
|
parent::setLabel($this->customLabelText);
|
|
}
|
|
parent::parse($parsedText);
|
|
}
|
|
|
|
/**
|
|
* Formats data for gs1-128.
|
|
*
|
|
* @return string
|
|
*/
|
|
private function formatGs1128() {
|
|
$formatedText = '~F1';
|
|
$formatedLabel = '';
|
|
$c = count($this->identifiersId);
|
|
|
|
for ($i = 0; $i < $c; $i++) {
|
|
if ($i > 0) {
|
|
$formatedLabel .= ' ';
|
|
}
|
|
|
|
if ($this->identifiersId[$i] !== null) {
|
|
$formatedLabel .= '(' . $this->identifiersId[$i] . ')';
|
|
}
|
|
|
|
$formatedText .= $this->identifiersId[$i];
|
|
|
|
$formatedLabel .= $this->identifiersContent[$i];
|
|
$formatedText .= $this->identifiersContent[$i];
|
|
|
|
if (isset($this->identifiersAi[$this->identifiersId[$i]])) {
|
|
$ai_data = $this->identifiersAi[$this->identifiersId[$i]];
|
|
} elseif (isset($this->identifiersId[$i][3])) {
|
|
$identifierWithVar = substr($this->identifiersId[$i], 0, -1) . 'y';
|
|
$ai_data = isset($this->identifiersAi[$identifierWithVar]) ? $this->identifiersAi[$identifierWithVar] : null;
|
|
} else {
|
|
$ai_data = null;
|
|
}
|
|
|
|
/* We'll check if we need to add a ~F1 (<GS>) char */
|
|
/* If we use the legacy mode, we always add a ~F1 (<GS>) char between AIs */
|
|
if ($ai_data !== null) {
|
|
if ((strlen($this->identifiersContent[$i]) < $ai_data[self::MAXLENGTH] && ($i + 1) !== $c) || (!$this->strictMode && ($i + 1) !== $c)) {
|
|
$formatedText .= '~F1';
|
|
}
|
|
} elseif ($this->allowsUnknownIdentifier && $this->identifiersId[$i] === null && ($i + 1) !== $c) {
|
|
/* If this id is unknown, we add a ~F1 (<GS>) char */
|
|
$formatedText .= '~F1';
|
|
}
|
|
}
|
|
|
|
if ($this->noLengthLimit === false && (strlen(str_replace('~F1', chr(29), $formatedText)) - 1) > self::MAX_GS1128_CHARS) {
|
|
throw new CINParseException('gs1128', 'The barcode can\'t contain more than ' . self::MAX_GS1128_CHARS . ' characters.');
|
|
}
|
|
|
|
$this->label = $formatedLabel;
|
|
return $formatedText;
|
|
}
|
|
|
|
/**
|
|
* Parses the text to gs1-128.
|
|
*
|
|
* @param mixed $text
|
|
* @return mixed
|
|
*/
|
|
private function parseGs1128($text) {
|
|
/* We format correctly what the user gives */
|
|
if (is_array($text)) {
|
|
$formatArray = array();
|
|
foreach ($text as $content) {
|
|
if (is_array($content)) { /* double array */
|
|
if (count($content) === 2) {
|
|
if (is_array($content[self::ID]) || is_array($content[self::CONTENT])) {
|
|
throw new CINParseException('gs1128', 'Double arrays can\'t contain arrays.');
|
|
} else {
|
|
$formatArray[] = '(' . $content[self::ID] . ')' . $content[self::CONTENT];
|
|
}
|
|
} else {
|
|
throw new CINParseException('gs1128', 'Double arrays must contain 2 values.');
|
|
}
|
|
} else { /* simple array */
|
|
$formatArray[] = $content;
|
|
}
|
|
}
|
|
|
|
unset($text);
|
|
$text = $formatArray;
|
|
} else { /* string */
|
|
$text = array($text);
|
|
}
|
|
|
|
$textCount = count($text);
|
|
for ($cmpt = 0; $cmpt < $textCount; $cmpt++) {
|
|
/* We parse the content of the array */
|
|
if (!$this->parseContent($text[$cmpt])) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
return $this->formatGs1128();
|
|
}
|
|
|
|
/**
|
|
* Splits the id and the content for each application identifiers (AIs).
|
|
*
|
|
* @param string $text
|
|
* @param int $cmpt
|
|
* @return bool
|
|
*/
|
|
private function parseContent($text) {
|
|
/* $yAlreadySet has 3 states: */
|
|
/* null: There is no variable in the ID; true: the variable is already set; false: the variable is not set yet; */
|
|
$content = null;
|
|
$yAlreadySet = null;
|
|
$realNameId = null;
|
|
$separatorsFound = 0;
|
|
$checksumAdded = 0;
|
|
$decimalPointRemoved = 0;
|
|
$toParse = str_replace('~F1', chr(29), $text);
|
|
$nbCharToParse = strlen($toParse);
|
|
$nbCharId = 0;
|
|
$isFormated = $toParse[0] === '(' ? true : false;
|
|
$maxCharId = $isFormated ? self::MAX_ID_FORMATED : self::MAX_ID_NOT_FORMATED;
|
|
$id = strtolower(substr($toParse, 0, min($maxCharId, $nbCharToParse)));
|
|
$id = $isFormated ? $this->findIdFormated($id, $yAlreadySet, $realNameId) : $this->findIdNotFormated($id, $yAlreadySet, $realNameId);
|
|
|
|
if ($id === false) {
|
|
if ($this->allowsUnknownIdentifier === false) {
|
|
return false;
|
|
}
|
|
|
|
$id = null;
|
|
$nbCharId = 0;
|
|
$content = $toParse;
|
|
} else {
|
|
$nbCharId = strlen($id) + ($isFormated ? 2 : 0);
|
|
$n = min($this->identifiersAi[$realNameId][self::MAXLENGTH], $nbCharToParse);
|
|
$content = substr($toParse, $nbCharId, $n);
|
|
}
|
|
|
|
if ($id !== null) {
|
|
/* If we have an AI with an "y" var, we check if there is a decimal point in the next *MAXLENGTH* characters */
|
|
/* if there is one, we take an extra character */
|
|
if ($yAlreadySet !== null) {
|
|
if (strpos($content, '.') !== false || strpos($content, ',') !== false) {
|
|
$n++;
|
|
if ($n <= $nbCharToParse) {
|
|
/* We take an extra char */
|
|
$content = substr($toParse, $nbCharId, $n);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* We check for separator */
|
|
$separator = strpos($content, chr(29));
|
|
if ($separator !== false) {
|
|
$content = substr($content, 0, $separator);
|
|
$separatorsFound++;
|
|
}
|
|
|
|
if ($id !== null) {
|
|
/* We check the conformity */
|
|
if (!$this->checkConformity($content, $id, $realNameId)) {
|
|
return false;
|
|
}
|
|
|
|
/* We check the checksum */
|
|
if (!$this->checkChecksum($content, $id, $realNameId, $checksumAdded)) {
|
|
return false;
|
|
}
|
|
|
|
/* We check the vars */
|
|
if (!$this->checkVars($content, $id, $yAlreadySet, $decimalPointRemoved)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
$this->identifiersId[] = $id;
|
|
$this->identifiersContent[] = $content;
|
|
|
|
$nbCharLastContent = (((strlen($content) + $nbCharId) - $checksumAdded) + $decimalPointRemoved) + $separatorsFound;
|
|
if ($nbCharToParse - $nbCharLastContent > 0) {
|
|
/* If there is more than one content in this array, we parse again */
|
|
$otherContent = substr($toParse, $nbCharLastContent, $nbCharToParse);
|
|
$nbCharOtherContent = strlen($otherContent);
|
|
|
|
if ($otherContent[0] === chr(29)) {
|
|
$otherContent = substr($otherContent, 1);
|
|
$nbCharOtherContent--;
|
|
}
|
|
|
|
if ($nbCharOtherContent > 0) {
|
|
$text = $otherContent;
|
|
return $this->parseContent($text);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Checks if an id exists.
|
|
*
|
|
* @param string $id
|
|
* @param bool $yAlreadySet
|
|
* @param string $realNameId
|
|
* @return bool
|
|
*/
|
|
private function idExists($id, &$yAlreadySet, &$realNameId) {
|
|
$yFound = isset($id[3]) && $id[3] === 'y';
|
|
$idVarAdded = substr($id, 0, -1) . 'y';
|
|
|
|
if (isset($this->identifiersAi[$id])) {
|
|
if ($yFound) {
|
|
$yAlreadySet = false;
|
|
}
|
|
|
|
$realNameId = $id;
|
|
return true;
|
|
} elseif (!$yFound && isset($this->identifiersAi[$idVarAdded])) {
|
|
/* if the id don't exist, we try to find this id with "y" at the last char */
|
|
$yAlreadySet = true;
|
|
$realNameId = $idVarAdded;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Finds ID with formated content.
|
|
*
|
|
* @param string $id
|
|
* @param bool $yAlreadySet
|
|
* @param string $realNameId
|
|
* @return mixed
|
|
*/
|
|
private function findIdFormated($id, &$yAlreadySet, &$realNameId) {
|
|
$pos = strpos($id, ')');
|
|
if ($pos === false) {
|
|
throw new CINParseException('gs1128', 'Identifiers must have no more than 4 characters.');
|
|
} else {
|
|
if ($pos < 3) {
|
|
throw new CINParseException('gs1128', 'Identifiers must have at least 2 characters.');
|
|
}
|
|
|
|
$id = substr($id, 1, $pos - 1);
|
|
if ($this->idExists($id, $yAlreadySet, $realNameId)) {
|
|
return $id;
|
|
}
|
|
|
|
if ($this->allowsUnknownIdentifier === false) {
|
|
throw new CINParseException('gs1128', 'The identifier ' . $id . ' doesn\'t exist.');
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Finds ID with non-formated content.
|
|
*
|
|
* @param string $id
|
|
* @param bool $yAlreadySet
|
|
* @param string $realNameId
|
|
* @return mixed
|
|
*/
|
|
private function findIdNotFormated($id, &$yAlreadySet, &$realNameId) {
|
|
$tofind = $id;
|
|
|
|
while (strlen($tofind) >= 2) {
|
|
if ($this->idExists($tofind, $yAlreadySet, $realNameId)) {
|
|
return $tofind;
|
|
} else {
|
|
$tofind = substr($tofind, 0, -1);
|
|
}
|
|
}
|
|
|
|
if ($this->allowsUnknownIdentifier === false) {
|
|
throw new CINParseException('gs1128', 'Error in formatting, can\'t find an identifier.');
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Checks confirmity of the content.
|
|
*
|
|
* @param string $content
|
|
* @param string $id
|
|
* @param string $realNameId
|
|
* @return bool
|
|
*/
|
|
private function checkConformity(&$content, $id, $realNameId) {
|
|
switch ($this->identifiersAi[$realNameId][self::KIND_OF_DATA]) {
|
|
case self::NUMERIC:
|
|
$content = str_replace(',', '.', $content);
|
|
if (!preg_match("/^[0-9.]+$/", $content)) {
|
|
throw new CINParseException('gs1128', 'The value of "' . $id . '" must be numerical.');
|
|
}
|
|
|
|
break;
|
|
case self::DATE_YYMMDD:
|
|
$valid_date = true;
|
|
if (preg_match("/^[0-9]{6}$/", $content)) {
|
|
$year = substr($content, 0, 2);
|
|
$month = substr($content, 2, 2);
|
|
$day = substr($content, 4, 2);
|
|
|
|
/* day can be 00 if we only need month and year */
|
|
if (intval($month) < 1 || intval($month) > 12 || intval($day) < 0 || intval($day) > 31) {
|
|
$valid_date = false;
|
|
}
|
|
} else {
|
|
$valid_date = false;
|
|
}
|
|
|
|
if (!$valid_date) {
|
|
throw new CINParseException('gs1128', 'The value of "' . $id . '" must be in YYMMDD format.');
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
// We check the length of the content
|
|
$nbCharContent = strlen($content);
|
|
$checksumChar = 0;
|
|
$minlengthContent = $this->identifiersAi[$realNameId][self::MINLENGTH];
|
|
$maxlengthContent = $this->identifiersAi[$realNameId][self::MAXLENGTH];
|
|
|
|
if ($this->identifiersAi[$realNameId][self::CHECKSUM]) {
|
|
$checksumChar++;
|
|
}
|
|
|
|
if ($nbCharContent < ($minlengthContent - $checksumChar)) {
|
|
if ($minlengthContent === $maxlengthContent) {
|
|
throw new CINParseException('gs1128', 'The value of "' . $id . '" must contain ' . $minlengthContent . ' character(s).');
|
|
} else {
|
|
throw new CINParseException('gs1128', 'The value of "' . $id . '" must contain between ' . $minlengthContent . ' and ' . $maxlengthContent . ' character(s).');
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Verifies the checksum.
|
|
*
|
|
* @param string $content
|
|
* @param string $id
|
|
* @param int $realNameId
|
|
* @param int $checksumAdded
|
|
* @return bool
|
|
*/
|
|
private function checkChecksum(&$content, $id, $realNameId, &$checksumAdded) {
|
|
if ($this->identifiersAi[$realNameId][self::CHECKSUM]) {
|
|
$nbCharContent = strlen($content);
|
|
$minlengthContent = $this->identifiersAi[$realNameId][self::MINLENGTH];
|
|
if ($nbCharContent === ($minlengthContent - 1)) {
|
|
/* we need to calculate the checksum */
|
|
$content .= self::getAiContentChecksum($content);
|
|
$checksumAdded++;
|
|
} elseif ($nbCharContent === $minlengthContent) {
|
|
/* we need to check the checksum */
|
|
$checksum = self::getAiContentChecksum(substr($content, 0, -1));
|
|
if (intval($content[$nbCharContent - 1]) !== $checksum) {
|
|
throw new CINParseException('gs1128', 'The checksum of "(' . $id . ') ' . $content . '" must be: ' . $checksum);
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Checks vars "y".
|
|
*
|
|
* @param string $content
|
|
* @param string $id
|
|
* @param bool $yAlreadySet
|
|
* @param int $decimalPointRemoved
|
|
* @return bool
|
|
*/
|
|
private function checkVars(&$content, &$id, $yAlreadySet, &$decimalPointRemoved) {
|
|
$nbCharContent = strlen($content);
|
|
/* We check for "y" var in AI */
|
|
if ($yAlreadySet) {
|
|
/* We'll check if we have a decimal point */
|
|
if (strpos($content, '.') !== false) {
|
|
throw new CINParseException('gs1128', 'If you do not use any "y" variable, you have to insert a whole number.');
|
|
}
|
|
} elseif ($yAlreadySet !== null) {
|
|
/* We need to replace the "y" var with the position of the decimal point */
|
|
$pos = strpos($content, '.');
|
|
if ($pos === false) {
|
|
$pos = $nbCharContent - 1;
|
|
}
|
|
|
|
$id = str_replace('y', $nbCharContent - ($pos + 1), strtolower($id));
|
|
$content = str_replace('.', '', $content);
|
|
$decimalPointRemoved++;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Checksum Mod10.
|
|
*
|
|
* @param int $content
|
|
* @return int
|
|
*/
|
|
private static function calculateChecksumMod10($content) {
|
|
// Calculating Checksum
|
|
// Consider the right-most digit of the message to be in an "odd" position,
|
|
// and assign odd/even to each character moving from right to left
|
|
// Odd Position = 3, Even Position = 1
|
|
// Multiply it by the number
|
|
// Add all of that and do 10-(?mod10)
|
|
$odd = true;
|
|
$checksumValue = 0;
|
|
$c = strlen($content);
|
|
|
|
for ($i = $c; $i > 0; $i--) {
|
|
if ($odd === true) {
|
|
$multiplier = 3;
|
|
$odd = false;
|
|
} else {
|
|
$multiplier = 1;
|
|
$odd = true;
|
|
}
|
|
|
|
$checksumValue += ($content[$i - 1] * $multiplier);
|
|
}
|
|
|
|
return (10 - $checksumValue % 10) % 10;
|
|
}
|
|
}
|
|
?>
|