GeoIP/src/Application/Model/d3geoip.php

526 lines
15 KiB
PHP
Raw Normal View History

2013-04-22 11:21:42 +02:00
<?php
2013-04-22 11:40:11 +02:00
/**
* This Software is the property of Data Development and is protected
* by copyright law - it is NOT Freeware.
*
* Any unauthorized use of this software without a valid license
* is a violation of the license agreement and will be prosecuted by
* civil and criminal law.
*
* http://www.shopmodule.com
*
* @copyright (C) D3 Data Development (Inh. Thomas Dartsch)
* @author D3 Data Development - Daniel Seifert <ds@shopmodule.com>
* @link http://www.oxidmodule.com
*/
2012-02-15 14:17:42 +01:00
2020-04-17 09:29:09 +02:00
namespace D3\GeoIp\Application\Model;
use D3\ModCfg\Application\Model\Configuration\d3_cfg_mod;
2020-06-10 10:28:16 +02:00
use D3\ModCfg\Application\Model\d3str;
use D3\ModCfg\Application\Model\Exception\d3_cfg_mod_exception;
use D3\ModCfg\Application\Model\Exception\d3ShopCompatibilityAdapterException;
2020-04-17 09:29:09 +02:00
use D3\ModCfg\Application\Model\Log\d3log;
2020-06-10 10:28:16 +02:00
use Doctrine\DBAL\DBALException;
2020-04-17 09:29:09 +02:00
use OxidEsales\Eshop\Application\Model\Country;
2020-06-10 10:28:16 +02:00
use OxidEsales\Eshop\Core\Config;
2020-04-17 09:29:09 +02:00
use OxidEsales\Eshop\Core\DatabaseProvider;
2020-06-10 10:28:16 +02:00
use OxidEsales\Eshop\Core\Exception\DatabaseConnectionException;
use OxidEsales\Eshop\Core\Exception\DatabaseErrorException;
use OxidEsales\Eshop\Core\Exception\StandardException;
2020-04-17 09:29:09 +02:00
use OxidEsales\Eshop\Core\Model\BaseModel;
use OxidEsales\Eshop\Core\Registry;
class d3geoip extends BaseModel
2013-04-22 11:21:42 +02:00
{
protected $_sClassName = 'd3geoip';
private $_sModId = 'd3_geoip';
2013-04-22 11:40:11 +02:00
public $oCountry;
public $oD3Log;
2013-04-22 11:21:42 +02:00
/**
* Class constructor, initiates parent constructor (parent::oxI18n()).
*/
public function __construct()
{
parent::__construct();
$this->init('d3geoip');
}
2020-04-17 09:29:09 +02:00
/**
* get oxcountry object by given IP address (optional)
*
* @param string $sIP optional
*
2020-06-10 10:28:16 +02:00
* @return Country
* @throws d3ShopCompatibilityAdapterException
* @throws d3_cfg_mod_exception
* @throws DBALException
* @throws DatabaseConnectionException
* @throws DatabaseErrorException
* @throws StandardException
2020-04-17 09:29:09 +02:00
*/
2013-04-22 11:40:11 +02:00
public function getUserLocationCountryObject($sIP = null)
2013-04-22 11:21:42 +02:00
{
2015-07-19 20:42:40 +02:00
if (!$this->oCountry) {
startProfile(__METHOD__);
2015-07-19 20:42:40 +02:00
if (!$sIP) {
2013-04-22 11:21:42 +02:00
$sIP = $this->getIP();
2013-04-22 11:40:11 +02:00
}
2013-04-22 11:21:42 +02:00
2020-04-17 09:29:09 +02:00
$sISOAlpha = $this->loadByIP(Registry::getConfig()->checkParamSpecialChars($sIP));
2013-04-22 11:21:42 +02:00
2015-07-19 20:42:40 +02:00
if (!$sISOAlpha) {
2015-12-21 15:41:40 +01:00
$this->_getLog()->log(
d3log::ERROR,
__CLASS__,
__FUNCTION__,
__LINE__,
'get ISO by IP failed',
$sIP
);
2012-02-15 14:17:42 +01:00
$this->oCountry = $this->getCountryFallBackObject();
2015-07-19 20:42:40 +02:00
} else {
2015-12-21 15:41:40 +01:00
$this->_getLog()->log(
d3log::INFO,
__CLASS__,
__FUNCTION__,
__LINE__,
'get ISO by IP',
$sIP." => ".$sISOAlpha
);
2013-04-22 11:21:42 +02:00
$this->oCountry = $this->getCountryObject($sISOAlpha);
}
stopProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
}
return $this->oCountry;
}
2020-04-17 09:29:09 +02:00
/**
* get IP address from client or set test IP address
*
* @return string
2020-06-10 10:28:16 +02:00
* @throws DBALException
* @throws DatabaseConnectionException
* @throws DatabaseErrorException
2020-04-17 09:29:09 +02:00
*/
2013-04-22 11:21:42 +02:00
public function getIP()
{
startProfile(__METHOD__);
2015-12-21 15:41:40 +01:00
if ($this->_getModConfig()->getValue('blUseTestIp')
&& $this->_getModConfig()->getValue('sTestIp')
) {
$sIP = $this->_getModConfig()->getValue('sTestIp');
2015-12-21 15:41:40 +01:00
} elseif ($this->_getModConfig()->getValue('blUseTestCountry')
&& $this->_getModConfig()->getValue('sTestCountryIp')
) {
$sIP = $this->_getModConfig()->getValue('sTestCountryIp');
2015-07-19 20:42:40 +02:00
} else {
if(isset($_SERVER['HTTP_CF_CONNECTING_IP'])) {
$sIP = $_SERVER['HTTP_CF_CONNECTING_IP'];
} else if (isset($_SERVER['HTTP_X_REAL_IP'])) {
$sIP = $_SERVER['HTTP_X_REAL_IP'];
} else if (isset($_SERVER['HTTP_CLIENT_IP'])) {
$sIP = $_SERVER['HTTP_CLIENT_IP'];
} else if(isset($_SERVER['HTTP_X_FORWARDED_FOR'])) {
$sIP = $_SERVER['HTTP_X_FORWARDED_FOR'];
} else if(isset($_SERVER['HTTP_X_FORWARDED'])) {
$sIP = $_SERVER['HTTP_X_FORWARDED'];
} else if(isset($_SERVER['HTTP_FORWARDED_FOR'])) {
$sIP = $_SERVER['HTTP_FORWARDED_FOR'];
} else if(isset($_SERVER['HTTP_FORWARDED'])) {
$sIP = $_SERVER['HTTP_FORWARDED'];
} else if(isset($_SERVER['REMOTE_ADDR'])) {
$sIP = $_SERVER['REMOTE_ADDR'];
} else {
$sIP = 'UNKNOWN';
}
2013-04-22 11:40:11 +02:00
}
2016-11-21 16:34:05 +01:00
$sIP = str_replace(' ', '', $sIP);
stopProfile(__METHOD__);
2020-04-17 09:29:09 +02:00
return Registry::getConfig()->checkParamSpecialChars($sIP);
2013-04-22 11:21:42 +02:00
}
2020-04-17 09:29:09 +02:00
/**
* get ISO alpha 2 ID by IP address
*
* @param int $sIP IP address
*
* @return string
2020-06-10 10:28:16 +02:00
* @throws DatabaseConnectionException
2020-04-17 09:29:09 +02:00
*/
public function loadByIP($sIP)
2013-04-22 11:21:42 +02:00
{
startProfile(__METHOD__);
$sSelect = "
SELECT
d3iso
FROM
".$this->_sClassName."
WHERE
LPAD(
BINARY(
if(
IS_IPV4('" . $sIP . "'),
INET_ATON('" . $sIP . "'),
INET6_ATON('" . $sIP . "')
)
),
16,
0
) BETWEEN D3STARTIPBIN AND D3ENDIPBIN";
2020-04-17 09:29:09 +02:00
$oDB = DatabaseProvider::getDb(DatabaseProvider::FETCH_MODE_ASSOC);
$sISO = $oDB->getOne($sSelect);
stopProfile(__METHOD__);
return $sISO;
2013-04-22 11:21:42 +02:00
}
2020-04-17 09:29:09 +02:00
/**
* get Country object by ISO alpha 2 ID
*
* @param string $sISOAlpha
*
* @return Country
2020-06-10 10:28:16 +02:00
* @throws DatabaseConnectionException
2020-04-17 09:29:09 +02:00
*/
2013-04-22 11:21:42 +02:00
public function getCountryObject($sISOAlpha)
{
startProfile(__METHOD__);
2020-04-17 09:29:09 +02:00
/** @var Country $oCountry */
$oCountry = oxNew(Country::class);
2015-12-21 15:41:40 +01:00
$sSelect = "SELECT oxid FROM ".$oCountry->getViewName().
" WHERE OXISOALPHA2 = '".$sISOAlpha."' AND OXACTIVE = '1'";
2013-04-22 11:25:27 +02:00
2020-04-17 09:29:09 +02:00
$oDB = DatabaseProvider::getDb();
$oCountry->load($oDB->getOne($sSelect));
2013-04-22 11:21:42 +02:00
stopProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
return $oCountry;
}
2020-04-17 09:29:09 +02:00
/**
* get Country object for fallback, if set
*
2020-06-10 10:28:16 +02:00
* @return Country
* @throws DBALException
* @throws DatabaseConnectionException
* @throws DatabaseErrorException
2020-04-17 09:29:09 +02:00
*/
2012-02-15 14:17:42 +01:00
public function getCountryFallBackObject()
2012-02-15 14:15:28 +01:00
{
startProfile(__METHOD__);
2012-02-15 14:17:42 +01:00
$oCountry = oxNew('oxcountry');
2013-04-22 11:21:42 +02:00
2015-12-21 15:41:40 +01:00
if ($this->_getModConfig()->getValue('blUseFallback')
&& $this->_getModConfig()->getValue('sFallbackCountryId')
) {
2020-06-10 10:28:16 +02:00
$oCountry->load($this->_getModConfig()->getValue('sFallbackCountryId'));
2012-02-15 14:17:42 +01:00
}
2013-04-22 11:21:42 +02:00
stopProfile(__METHOD__);
2012-02-15 14:15:28 +01:00
return $oCountry;
}
2020-04-17 09:29:09 +02:00
/**
* check module active state and set user country specific language
*
2020-06-10 10:28:16 +02:00
* @throws DBALException
* @throws DatabaseConnectionException
* @throws DatabaseErrorException
* @throws StandardException
* @throws d3ShopCompatibilityAdapterException
* @throws d3_cfg_mod_exception
2020-04-17 09:29:09 +02:00
*/
2013-04-22 11:21:42 +02:00
public function setCountryLanguage()
{
startProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
$this->performURLSwitch();
$this->performShopSwitch();
2015-12-21 15:41:40 +01:00
if (!$this->_getModConfig()->isActive()
|| false == $this->_getModConfig()->getValue('blChangeLang')) {
stopProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
return;
}
2013-04-22 11:21:42 +02:00
$oCountry = $this->getUserLocationCountryObject();
2015-07-19 20:42:40 +02:00
if (!$this->isAdmin()
2020-04-17 09:29:09 +02:00
&& Registry::getUtils()->isSearchEngine() === false
&& Registry::getSession()->getId()
&& Registry::getSession()->getVariable('d3isSetLang') === null
2015-12-21 15:41:40 +01:00
&& $oCountry->getId() && $oCountry->getFieldData('d3geoiplang') > -1
2015-07-19 20:42:40 +02:00
) {
2015-12-21 15:41:40 +01:00
$this->_getLog()->log(
d3log::INFO,
__CLASS__,
__FUNCTION__,
__LINE__,
'set language',
$this->getIP().' => '.$oCountry->getFieldData('d3geoiplang')
);
2020-04-17 09:29:09 +02:00
Registry::getLang()->setTplLanguage((int) $oCountry->getFieldData('d3geoiplang'));
Registry::getLang()->setBaseLanguage((int) $oCountry->getFieldData('d3geoiplang'));
Registry::getSession()->setVariable('d3isSetLang', true);
2013-04-22 11:21:42 +02:00
}
stopProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
}
2020-04-17 09:29:09 +02:00
/**
* check module active state and set user country specific currency
*
2020-06-10 10:28:16 +02:00
* @throws DBALException
* @throws DatabaseConnectionException
* @throws DatabaseErrorException
* @throws StandardException
* @throws d3ShopCompatibilityAdapterException
* @throws d3_cfg_mod_exception
2020-04-17 09:29:09 +02:00
*/
2013-04-22 11:21:42 +02:00
public function setCountryCurrency()
{
2015-12-21 15:41:40 +01:00
if (!$this->_getModConfig()->isActive()
|| false == $this->_getModConfig()->getValue('blChangeCurr')
) {
2013-04-22 11:21:42 +02:00
return;
2015-07-19 20:42:40 +02:00
}
2013-04-22 11:21:42 +02:00
startProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
$oCountry = $this->getUserLocationCountryObject();
2015-07-19 20:42:40 +02:00
if (!$this->isAdmin()
2020-04-17 09:29:09 +02:00
&& Registry::getUtils()->isSearchEngine() === false
&& !Registry::getSession()->getVariable('d3isSetCurr')
2015-07-19 20:42:40 +02:00
&& $oCountry->getId()
&& $oCountry->getFieldData('d3geoipcur') > -1
) {
2015-12-21 15:41:40 +01:00
$this->_getLog()->log(
d3log::INFO,
__CLASS__,
__FUNCTION__,
__LINE__,
'set currency',
$this->getIP().' => '.$oCountry->getFieldData('d3geoipcur')
);
2020-04-17 09:29:09 +02:00
Registry::getConfig()->setActShopCurrency((int) $oCountry->getFieldData('d3geoipcur'));
Registry::getSession()->setVariable('d3isSetCurr', true);
2013-04-22 11:21:42 +02:00
}
stopProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
}
2020-06-10 10:28:16 +02:00
/**
* @param $oCurr
*
* @return bool
* @throws DBALException
* @throws DatabaseConnectionException
* @throws DatabaseErrorException
* @throws StandardException
* @throws d3ShopCompatibilityAdapterException
* @throws d3_cfg_mod_exception
*/
public function hasNotSetCurrency($oCurr)
{
$oCountry = $this->getUserLocationCountryObject();
if ($oCountry->getFieldData('d3geoipcur') > 0
&& $oCurr->id != $oCountry->getFieldData('d3geoipcur')
) {
return true;
}
return false;
}
2020-04-17 09:29:09 +02:00
/**
* check module active state and perform switching to user country specific shop (EE only)
*
2020-06-10 10:28:16 +02:00
* @throws DBALException
* @throws DatabaseConnectionException
* @throws DatabaseErrorException
* @throws StandardException
* @throws d3ShopCompatibilityAdapterException
* @throws d3_cfg_mod_exception
2020-04-17 09:29:09 +02:00
*/
2013-04-22 11:21:42 +02:00
public function performShopSwitch()
{
2015-07-19 20:42:40 +02:00
if (!$this->_getModConfig()->isActive() || !$this->_getModConfig()->getValue('blChangeShop')) {
2013-04-22 11:21:42 +02:00
return;
2015-07-19 20:42:40 +02:00
}
2013-04-22 11:21:42 +02:00
startProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
$oCountry = $this->getUserLocationCountryObject();
2013-04-22 11:21:42 +02:00
$iNewShop = $oCountry->getFieldData('d3geoipshop');
2020-06-10 10:28:16 +02:00
if (Registry::getRequest()->getRequestEscapedParameter('d3redirect') != 1
2015-07-19 20:42:40 +02:00
&& false == $this->isAdmin()
2020-04-17 09:29:09 +02:00
&& Registry::getUtils()->isSearchEngine() === false
2015-07-19 20:42:40 +02:00
&& $oCountry->getId()
2015-12-21 15:41:40 +01:00
&& $this->getConfig()->isMall()
&& $iNewShop > -1 &&
(
2015-07-19 20:42:40 +02:00
$iNewShop != $this->getConfig()->getShopId()
2020-04-17 09:29:09 +02:00
|| strtolower($this->getConfig()->getActiveView()->getClassKey()) == 'mallstart'
)
2015-07-19 20:42:40 +02:00
) {
2020-06-10 10:28:16 +02:00
$oNewConf = new Config();
2013-04-22 11:21:42 +02:00
$oNewConf->setShopId($iNewShop);
$oNewConf->init();
$this->getConfig()->onShopChange();
2020-04-17 09:29:09 +02:00
if (!Registry::getSession()->getVariable('d3isSetLang')
2015-12-21 15:41:40 +01:00
&& $this->_getModConfig()->getValue('blChangeLang')
&& $oCountry->getFieldData('d3geoiplang') > -1
) {
2013-04-22 11:21:42 +02:00
$sLangId = $oCountry->getFieldData('d3geoiplang');
2015-07-19 20:42:40 +02:00
} else {
2013-04-22 11:21:42 +02:00
$sLangId = '';
2013-04-22 11:40:11 +02:00
}
2013-04-22 11:21:42 +02:00
/** @var $oStr d3str */
2020-06-10 10:28:16 +02:00
$oStr = Registry::get(d3str::class);
$aParams = array(
'd3redirect' => '1',
2020-06-10 10:28:16 +02:00
'fnc' => Registry::getRequest()->getRequestEscapedParameter('fnc'),
'shp' => $iNewShop
);
2015-12-21 15:41:40 +01:00
$sUrl = str_replace(
'&amp;',
'&',
$oStr->generateParameterUrl($oNewConf->getShopHomeUrl($sLangId), $aParams)
);
2015-12-21 15:41:40 +01:00
$this->_getLog()->log(
d3log::INFO,
__CLASS__,
__FUNCTION__,
__LINE__,
'change shop',
$this->getIP().' => '.$sUrl
);
2013-04-22 11:21:42 +02:00
header("Location: ".$sUrl);
2013-04-22 11:21:42 +02:00
exit();
}
stopProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
}
2020-04-17 09:29:09 +02:00
/**
* check module active state and perform switching to user country specific url
*
2020-06-10 10:28:16 +02:00
* @throws DBALException
* @throws DatabaseConnectionException
* @throws DatabaseErrorException
* @throws StandardException
* @throws d3ShopCompatibilityAdapterException
* @throws d3_cfg_mod_exception
2020-04-17 09:29:09 +02:00
*/
2013-04-22 11:21:42 +02:00
public function performURLSwitch()
{
2015-12-21 15:41:40 +01:00
if (!$this->_getModConfig()->isActive()
|| false == $this->_getModConfig()->getValue('blChangeURL')) {
2013-04-22 11:21:42 +02:00
return;
2013-04-22 11:40:11 +02:00
}
2013-04-22 11:21:42 +02:00
startProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
$oCountry = $this->getUserLocationCountryObject();
2021-07-22 10:54:03 +02:00
if (false == (bool) Registry::getRequest()->getRequestEscapedParameter('forceUrl')
&& false == $this->isAdmin()
2020-04-17 09:29:09 +02:00
&& Registry::getUtils()->isSearchEngine() === false
2015-12-21 15:41:40 +01:00
&& $oCountry->getId()
&& $oCountry->getFieldData('d3geoipurl')
&& strlen(trim($oCountry->getFieldData('d3geoipurl'))) > 0
2015-07-19 20:42:40 +02:00
) {
2013-04-22 11:21:42 +02:00
$sNewUrl = $oCountry->getFieldData('d3geoipurl');
2015-12-21 15:41:40 +01:00
$this->_getLog()->log(
d3log::INFO,
__CLASS__,
__FUNCTION__,
__LINE__,
'change url',
$this->getIP().' => '.$oCountry->getFieldData('d3geoipurl')
);
2013-04-22 11:21:42 +02:00
header("Location: ".$sNewUrl);
exit();
}
stopProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
}
/**
* get all shop urls
*
2020-06-10 10:28:16 +02:00
* @return array
*/
2013-04-22 11:21:42 +02:00
public function getShopUrls()
{
startProfile(__METHOD__);
2015-12-21 15:41:40 +01:00
$oShoplist = oxNew('oxshoplist');
2013-04-22 11:21:42 +02:00
$oShoplist->getList();
$aShopUrls = array();
2015-12-21 15:41:40 +01:00
foreach ($oShoplist->arrayKeys() as $sId) {
$aShopUrls[$sId] = $this->getConfig()->getShopConfVar('sMallShopURL', $sId);
2013-04-22 11:21:42 +02:00
}
stopProfile(__METHOD__);
2013-04-22 11:21:42 +02:00
return $aShopUrls;
}
2020-04-17 09:29:09 +02:00
/**
* get modcfg instance
*
* @return d3_cfg_mod
2020-06-10 10:28:16 +02:00
* @throws DBALException
* @throws DatabaseConnectionException
* @throws DatabaseErrorException
2020-04-17 09:29:09 +02:00
*/
2013-04-22 11:21:42 +02:00
protected function _getModConfig()
{
return d3_cfg_mod::get($this->_sModId);
}
2020-04-17 09:29:09 +02:00
/**
* get d3log instance
*
* @return d3log
2020-06-10 10:28:16 +02:00
* @throws DBALException
* @throws DatabaseConnectionException
* @throws DatabaseErrorException
2020-04-17 09:29:09 +02:00
*/
2013-04-22 11:21:42 +02:00
protected function _getLog()
{
2015-07-19 20:42:40 +02:00
if (!$this->oD3Log) {
2015-12-21 15:41:40 +01:00
$this->oD3Log = $this->_getModConfig()->d3getLog();
2013-04-22 11:40:11 +02:00
}
2013-04-22 11:21:42 +02:00
return $this->oD3Log;
}
2015-07-19 20:42:40 +02:00
}
2015-12-21 15:41:40 +01:00