class-wc-geo-ip.php 30 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814
  1. <?php
  2. /**
  3. * Geo IP class
  4. *
  5. * This class is a fork of GeoIP class from MaxMind LLC.
  6. *
  7. * @package WooCommerce/Classes
  8. * @version 2.4.0
  9. * @deprecated 3.4.0
  10. */
  11. if ( ! defined( 'ABSPATH' ) ) {
  12. exit;
  13. }
  14. /**
  15. * WC_Geo_IP Class.
  16. *
  17. * @deprecated 3.4.0
  18. */
  19. class WC_Geo_IP {
  20. const GEOIP_COUNTRY_BEGIN = 16776960;
  21. const GEOIP_STATE_BEGIN_REV0 = 16700000;
  22. const GEOIP_STATE_BEGIN_REV1 = 16000000;
  23. const GEOIP_MEMORY_CACHE = 1;
  24. const GEOIP_SHARED_MEMORY = 2;
  25. const STRUCTURE_INFO_MAX_SIZE = 20;
  26. const GEOIP_COUNTRY_EDITION = 1;
  27. const GEOIP_PROXY_EDITION = 8;
  28. const GEOIP_ASNUM_EDITION = 9;
  29. const GEOIP_NETSPEED_EDITION = 10;
  30. const GEOIP_REGION_EDITION_REV0 = 7;
  31. const GEOIP_REGION_EDITION_REV1 = 3;
  32. const GEOIP_CITY_EDITION_REV0 = 6;
  33. const GEOIP_CITY_EDITION_REV1 = 2;
  34. const GEOIP_ORG_EDITION = 5;
  35. const GEOIP_ISP_EDITION = 4;
  36. const SEGMENT_RECORD_LENGTH = 3;
  37. const STANDARD_RECORD_LENGTH = 3;
  38. const ORG_RECORD_LENGTH = 4;
  39. const GEOIP_SHM_KEY = 0x4f415401;
  40. const GEOIP_DOMAIN_EDITION = 11;
  41. const GEOIP_COUNTRY_EDITION_V6 = 12;
  42. const GEOIP_LOCATIONA_EDITION = 13;
  43. const GEOIP_ACCURACYRADIUS_EDITION = 14;
  44. const GEOIP_CITY_EDITION_REV1_V6 = 30;
  45. const GEOIP_CITY_EDITION_REV0_V6 = 31;
  46. const GEOIP_NETSPEED_EDITION_REV1 = 32;
  47. const GEOIP_NETSPEED_EDITION_REV1_V6 = 33;
  48. const GEOIP_USERTYPE_EDITION = 28;
  49. const GEOIP_USERTYPE_EDITION_V6 = 29;
  50. const GEOIP_ASNUM_EDITION_V6 = 21;
  51. const GEOIP_ISP_EDITION_V6 = 22;
  52. const GEOIP_ORG_EDITION_V6 = 23;
  53. const GEOIP_DOMAIN_EDITION_V6 = 24;
  54. /**
  55. * Flags.
  56. *
  57. * @var int
  58. */
  59. public $flags;
  60. /**
  61. * File handler.
  62. *
  63. * @var resource
  64. */
  65. public $filehandle;
  66. /**
  67. * Memory buffer.
  68. *
  69. * @var string
  70. */
  71. public $memory_buffer;
  72. /**
  73. * Database type.
  74. *
  75. * @var int
  76. */
  77. public $databaseType;
  78. /**
  79. * Database segments.
  80. *
  81. * @var int
  82. */
  83. public $databaseSegments;
  84. /**
  85. * Record length.
  86. *
  87. * @var int
  88. */
  89. public $record_length;
  90. /**
  91. * Shmid.
  92. *
  93. * @var string
  94. */
  95. public $shmid;
  96. /**
  97. * Two letters country codes.
  98. *
  99. * @var array
  100. */
  101. public $GEOIP_COUNTRY_CODES = array(
  102. '',
  103. 'AP',
  104. 'EU',
  105. 'AD',
  106. 'AE',
  107. 'AF',
  108. 'AG',
  109. 'AI',
  110. 'AL',
  111. 'AM',
  112. 'CW',
  113. 'AO',
  114. 'AQ',
  115. 'AR',
  116. 'AS',
  117. 'AT',
  118. 'AU',
  119. 'AW',
  120. 'AZ',
  121. 'BA',
  122. 'BB',
  123. 'BD',
  124. 'BE',
  125. 'BF',
  126. 'BG',
  127. 'BH',
  128. 'BI',
  129. 'BJ',
  130. 'BM',
  131. 'BN',
  132. 'BO',
  133. 'BR',
  134. 'BS',
  135. 'BT',
  136. 'BV',
  137. 'BW',
  138. 'BY',
  139. 'BZ',
  140. 'CA',
  141. 'CC',
  142. 'CD',
  143. 'CF',
  144. 'CG',
  145. 'CH',
  146. 'CI',
  147. 'CK',
  148. 'CL',
  149. 'CM',
  150. 'CN',
  151. 'CO',
  152. 'CR',
  153. 'CU',
  154. 'CV',
  155. 'CX',
  156. 'CY',
  157. 'CZ',
  158. 'DE',
  159. 'DJ',
  160. 'DK',
  161. 'DM',
  162. 'DO',
  163. 'DZ',
  164. 'EC',
  165. 'EE',
  166. 'EG',
  167. 'EH',
  168. 'ER',
  169. 'ES',
  170. 'ET',
  171. 'FI',
  172. 'FJ',
  173. 'FK',
  174. 'FM',
  175. 'FO',
  176. 'FR',
  177. 'SX',
  178. 'GA',
  179. 'GB',
  180. 'GD',
  181. 'GE',
  182. 'GF',
  183. 'GH',
  184. 'GI',
  185. 'GL',
  186. 'GM',
  187. 'GN',
  188. 'GP',
  189. 'GQ',
  190. 'GR',
  191. 'GS',
  192. 'GT',
  193. 'GU',
  194. 'GW',
  195. 'GY',
  196. 'HK',
  197. 'HM',
  198. 'HN',
  199. 'HR',
  200. 'HT',
  201. 'HU',
  202. 'ID',
  203. 'IE',
  204. 'IL',
  205. 'IN',
  206. 'IO',
  207. 'IQ',
  208. 'IR',
  209. 'IS',
  210. 'IT',
  211. 'JM',
  212. 'JO',
  213. 'JP',
  214. 'KE',
  215. 'KG',
  216. 'KH',
  217. 'KI',
  218. 'KM',
  219. 'KN',
  220. 'KP',
  221. 'KR',
  222. 'KW',
  223. 'KY',
  224. 'KZ',
  225. 'LA',
  226. 'LB',
  227. 'LC',
  228. 'LI',
  229. 'LK',
  230. 'LR',
  231. 'LS',
  232. 'LT',
  233. 'LU',
  234. 'LV',
  235. 'LY',
  236. 'MA',
  237. 'MC',
  238. 'MD',
  239. 'MG',
  240. 'MH',
  241. 'MK',
  242. 'ML',
  243. 'MM',
  244. 'MN',
  245. 'MO',
  246. 'MP',
  247. 'MQ',
  248. 'MR',
  249. 'MS',
  250. 'MT',
  251. 'MU',
  252. 'MV',
  253. 'MW',
  254. 'MX',
  255. 'MY',
  256. 'MZ',
  257. 'NA',
  258. 'NC',
  259. 'NE',
  260. 'NF',
  261. 'NG',
  262. 'NI',
  263. 'NL',
  264. 'NO',
  265. 'NP',
  266. 'NR',
  267. 'NU',
  268. 'NZ',
  269. 'OM',
  270. 'PA',
  271. 'PE',
  272. 'PF',
  273. 'PG',
  274. 'PH',
  275. 'PK',
  276. 'PL',
  277. 'PM',
  278. 'PN',
  279. 'PR',
  280. 'PS',
  281. 'PT',
  282. 'PW',
  283. 'PY',
  284. 'QA',
  285. 'RE',
  286. 'RO',
  287. 'RU',
  288. 'RW',
  289. 'SA',
  290. 'SB',
  291. 'SC',
  292. 'SD',
  293. 'SE',
  294. 'SG',
  295. 'SH',
  296. 'SI',
  297. 'SJ',
  298. 'SK',
  299. 'SL',
  300. 'SM',
  301. 'SN',
  302. 'SO',
  303. 'SR',
  304. 'ST',
  305. 'SV',
  306. 'SY',
  307. 'SZ',
  308. 'TC',
  309. 'TD',
  310. 'TF',
  311. 'TG',
  312. 'TH',
  313. 'TJ',
  314. 'TK',
  315. 'TM',
  316. 'TN',
  317. 'TO',
  318. 'TL',
  319. 'TR',
  320. 'TT',
  321. 'TV',
  322. 'TW',
  323. 'TZ',
  324. 'UA',
  325. 'UG',
  326. 'UM',
  327. 'US',
  328. 'UY',
  329. 'UZ',
  330. 'VA',
  331. 'VC',
  332. 'VE',
  333. 'VG',
  334. 'VI',
  335. 'VN',
  336. 'VU',
  337. 'WF',
  338. 'WS',
  339. 'YE',
  340. 'YT',
  341. 'RS',
  342. 'ZA',
  343. 'ZM',
  344. 'ME',
  345. 'ZW',
  346. 'A1',
  347. 'A2',
  348. 'O1',
  349. 'AX',
  350. 'GG',
  351. 'IM',
  352. 'JE',
  353. 'BL',
  354. 'MF',
  355. 'BQ',
  356. 'SS',
  357. 'O1',
  358. );
  359. /**
  360. * 3 letters country codes.
  361. *
  362. * @var array
  363. */
  364. public $GEOIP_COUNTRY_CODES3 = array(
  365. '',
  366. 'AP',
  367. 'EU',
  368. 'AND',
  369. 'ARE',
  370. 'AFG',
  371. 'ATG',
  372. 'AIA',
  373. 'ALB',
  374. 'ARM',
  375. 'CUW',
  376. 'AGO',
  377. 'ATA',
  378. 'ARG',
  379. 'ASM',
  380. 'AUT',
  381. 'AUS',
  382. 'ABW',
  383. 'AZE',
  384. 'BIH',
  385. 'BRB',
  386. 'BGD',
  387. 'BEL',
  388. 'BFA',
  389. 'BGR',
  390. 'BHR',
  391. 'BDI',
  392. 'BEN',
  393. 'BMU',
  394. 'BRN',
  395. 'BOL',
  396. 'BRA',
  397. 'BHS',
  398. 'BTN',
  399. 'BVT',
  400. 'BWA',
  401. 'BLR',
  402. 'BLZ',
  403. 'CAN',
  404. 'CCK',
  405. 'COD',
  406. 'CAF',
  407. 'COG',
  408. 'CHE',
  409. 'CIV',
  410. 'COK',
  411. 'CHL',
  412. 'CMR',
  413. 'CHN',
  414. 'COL',
  415. 'CRI',
  416. 'CUB',
  417. 'CPV',
  418. 'CXR',
  419. 'CYP',
  420. 'CZE',
  421. 'DEU',
  422. 'DJI',
  423. 'DNK',
  424. 'DMA',
  425. 'DOM',
  426. 'DZA',
  427. 'ECU',
  428. 'EST',
  429. 'EGY',
  430. 'ESH',
  431. 'ERI',
  432. 'ESP',
  433. 'ETH',
  434. 'FIN',
  435. 'FJI',
  436. 'FLK',
  437. 'FSM',
  438. 'FRO',
  439. 'FRA',
  440. 'SXM',
  441. 'GAB',
  442. 'GBR',
  443. 'GRD',
  444. 'GEO',
  445. 'GUF',
  446. 'GHA',
  447. 'GIB',
  448. 'GRL',
  449. 'GMB',
  450. 'GIN',
  451. 'GLP',
  452. 'GNQ',
  453. 'GRC',
  454. 'SGS',
  455. 'GTM',
  456. 'GUM',
  457. 'GNB',
  458. 'GUY',
  459. 'HKG',
  460. 'HMD',
  461. 'HND',
  462. 'HRV',
  463. 'HTI',
  464. 'HUN',
  465. 'IDN',
  466. 'IRL',
  467. 'ISR',
  468. 'IND',
  469. 'IOT',
  470. 'IRQ',
  471. 'IRN',
  472. 'ISL',
  473. 'ITA',
  474. 'JAM',
  475. 'JOR',
  476. 'JPN',
  477. 'KEN',
  478. 'KGZ',
  479. 'KHM',
  480. 'KIR',
  481. 'COM',
  482. 'KNA',
  483. 'PRK',
  484. 'KOR',
  485. 'KWT',
  486. 'CYM',
  487. 'KAZ',
  488. 'LAO',
  489. 'LBN',
  490. 'LCA',
  491. 'LIE',
  492. 'LKA',
  493. 'LBR',
  494. 'LSO',
  495. 'LTU',
  496. 'LUX',
  497. 'LVA',
  498. 'LBY',
  499. 'MAR',
  500. 'MCO',
  501. 'MDA',
  502. 'MDG',
  503. 'MHL',
  504. 'MKD',
  505. 'MLI',
  506. 'MMR',
  507. 'MNG',
  508. 'MAC',
  509. 'MNP',
  510. 'MTQ',
  511. 'MRT',
  512. 'MSR',
  513. 'MLT',
  514. 'MUS',
  515. 'MDV',
  516. 'MWI',
  517. 'MEX',
  518. 'MYS',
  519. 'MOZ',
  520. 'NAM',
  521. 'NCL',
  522. 'NER',
  523. 'NFK',
  524. 'NGA',
  525. 'NIC',
  526. 'NLD',
  527. 'NOR',
  528. 'NPL',
  529. 'NRU',
  530. 'NIU',
  531. 'NZL',
  532. 'OMN',
  533. 'PAN',
  534. 'PER',
  535. 'PYF',
  536. 'PNG',
  537. 'PHL',
  538. 'PAK',
  539. 'POL',
  540. 'SPM',
  541. 'PCN',
  542. 'PRI',
  543. 'PSE',
  544. 'PRT',
  545. 'PLW',
  546. 'PRY',
  547. 'QAT',
  548. 'REU',
  549. 'ROU',
  550. 'RUS',
  551. 'RWA',
  552. 'SAU',
  553. 'SLB',
  554. 'SYC',
  555. 'SDN',
  556. 'SWE',
  557. 'SGP',
  558. 'SHN',
  559. 'SVN',
  560. 'SJM',
  561. 'SVK',
  562. 'SLE',
  563. 'SMR',
  564. 'SEN',
  565. 'SOM',
  566. 'SUR',
  567. 'STP',
  568. 'SLV',
  569. 'SYR',
  570. 'SWZ',
  571. 'TCA',
  572. 'TCD',
  573. 'ATF',
  574. 'TGO',
  575. 'THA',
  576. 'TJK',
  577. 'TKL',
  578. 'TKM',
  579. 'TUN',
  580. 'TON',
  581. 'TLS',
  582. 'TUR',
  583. 'TTO',
  584. 'TUV',
  585. 'TWN',
  586. 'TZA',
  587. 'UKR',
  588. 'UGA',
  589. 'UMI',
  590. 'USA',
  591. 'URY',
  592. 'UZB',
  593. 'VAT',
  594. 'VCT',
  595. 'VEN',
  596. 'VGB',
  597. 'VIR',
  598. 'VNM',
  599. 'VUT',
  600. 'WLF',
  601. 'WSM',
  602. 'YEM',
  603. 'MYT',
  604. 'SRB',
  605. 'ZAF',
  606. 'ZMB',
  607. 'MNE',
  608. 'ZWE',
  609. 'A1',
  610. 'A2',
  611. 'O1',
  612. 'ALA',
  613. 'GGY',
  614. 'IMN',
  615. 'JEY',
  616. 'BLM',
  617. 'MAF',
  618. 'BES',
  619. 'SSD',
  620. 'O1',
  621. );
  622. /**
  623. * Contry names.
  624. *
  625. * @var array
  626. */
  627. public $GEOIP_COUNTRY_NAMES = array(
  628. '',
  629. 'Asia/Pacific Region',
  630. 'Europe',
  631. 'Andorra',
  632. 'United Arab Emirates',
  633. 'Afghanistan',
  634. 'Antigua and Barbuda',
  635. 'Anguilla',
  636. 'Albania',
  637. 'Armenia',
  638. 'Curacao',
  639. 'Angola',
  640. 'Antarctica',
  641. 'Argentina',
  642. 'American Samoa',
  643. 'Austria',
  644. 'Australia',
  645. 'Aruba',
  646. 'Azerbaijan',
  647. 'Bosnia and Herzegovina',
  648. 'Barbados',
  649. 'Bangladesh',
  650. 'Belgium',
  651. 'Burkina Faso',
  652. 'Bulgaria',
  653. 'Bahrain',
  654. 'Burundi',
  655. 'Benin',
  656. 'Bermuda',
  657. 'Brunei Darussalam',
  658. 'Bolivia',
  659. 'Brazil',
  660. 'Bahamas',
  661. 'Bhutan',
  662. 'Bouvet Island',
  663. 'Botswana',
  664. 'Belarus',
  665. 'Belize',
  666. 'Canada',
  667. 'Cocos (Keeling) Islands',
  668. 'Congo, The Democratic Republic of the',
  669. 'Central African Republic',
  670. 'Congo',
  671. 'Switzerland',
  672. "Cote D'Ivoire",
  673. 'Cook Islands',
  674. 'Chile',
  675. 'Cameroon',
  676. 'China',
  677. 'Colombia',
  678. 'Costa Rica',
  679. 'Cuba',
  680. 'Cape Verde',
  681. 'Christmas Island',
  682. 'Cyprus',
  683. 'Czech Republic',
  684. 'Germany',
  685. 'Djibouti',
  686. 'Denmark',
  687. 'Dominica',
  688. 'Dominican Republic',
  689. 'Algeria',
  690. 'Ecuador',
  691. 'Estonia',
  692. 'Egypt',
  693. 'Western Sahara',
  694. 'Eritrea',
  695. 'Spain',
  696. 'Ethiopia',
  697. 'Finland',
  698. 'Fiji',
  699. 'Falkland Islands (Malvinas)',
  700. 'Micronesia, Federated States of',
  701. 'Faroe Islands',
  702. 'France',
  703. 'Sint Maarten (Dutch part)',
  704. 'Gabon',
  705. 'United Kingdom',
  706. 'Grenada',
  707. 'Georgia',
  708. 'French Guiana',
  709. 'Ghana',
  710. 'Gibraltar',
  711. 'Greenland',
  712. 'Gambia',
  713. 'Guinea',
  714. 'Guadeloupe',
  715. 'Equatorial Guinea',
  716. 'Greece',
  717. 'South Georgia and the South Sandwich Islands',
  718. 'Guatemala',
  719. 'Guam',
  720. 'Guinea-Bissau',
  721. 'Guyana',
  722. 'Hong Kong',
  723. 'Heard Island and McDonald Islands',
  724. 'Honduras',
  725. 'Croatia',
  726. 'Haiti',
  727. 'Hungary',
  728. 'Indonesia',
  729. 'Ireland',
  730. 'Israel',
  731. 'India',
  732. 'British Indian Ocean Territory',
  733. 'Iraq',
  734. 'Iran, Islamic Republic of',
  735. 'Iceland',
  736. 'Italy',
  737. 'Jamaica',
  738. 'Jordan',
  739. 'Japan',
  740. 'Kenya',
  741. 'Kyrgyzstan',
  742. 'Cambodia',
  743. 'Kiribati',
  744. 'Comoros',
  745. 'Saint Kitts and Nevis',
  746. "Korea, Democratic People's Republic of",
  747. 'Korea, Republic of',
  748. 'Kuwait',
  749. 'Cayman Islands',
  750. 'Kazakhstan',
  751. "Lao People's Democratic Republic",
  752. 'Lebanon',
  753. 'Saint Lucia',
  754. 'Liechtenstein',
  755. 'Sri Lanka',
  756. 'Liberia',
  757. 'Lesotho',
  758. 'Lithuania',
  759. 'Luxembourg',
  760. 'Latvia',
  761. 'Libya',
  762. 'Morocco',
  763. 'Monaco',
  764. 'Moldova, Republic of',
  765. 'Madagascar',
  766. 'Marshall Islands',
  767. 'Macedonia',
  768. 'Mali',
  769. 'Myanmar',
  770. 'Mongolia',
  771. 'Macau',
  772. 'Northern Mariana Islands',
  773. 'Martinique',
  774. 'Mauritania',
  775. 'Montserrat',
  776. 'Malta',
  777. 'Mauritius',
  778. 'Maldives',
  779. 'Malawi',
  780. 'Mexico',
  781. 'Malaysia',
  782. 'Mozambique',
  783. 'Namibia',
  784. 'New Caledonia',
  785. 'Niger',
  786. 'Norfolk Island',
  787. 'Nigeria',
  788. 'Nicaragua',
  789. 'Netherlands',
  790. 'Norway',
  791. 'Nepal',
  792. 'Nauru',
  793. 'Niue',
  794. 'New Zealand',
  795. 'Oman',
  796. 'Panama',
  797. 'Peru',
  798. 'French Polynesia',
  799. 'Papua New Guinea',
  800. 'Philippines',
  801. 'Pakistan',
  802. 'Poland',
  803. 'Saint Pierre and Miquelon',
  804. 'Pitcairn Islands',
  805. 'Puerto Rico',
  806. 'Palestinian Territory',
  807. 'Portugal',
  808. 'Palau',
  809. 'Paraguay',
  810. 'Qatar',
  811. 'Reunion',
  812. 'Romania',
  813. 'Russian Federation',
  814. 'Rwanda',
  815. 'Saudi Arabia',
  816. 'Solomon Islands',
  817. 'Seychelles',
  818. 'Sudan',
  819. 'Sweden',
  820. 'Singapore',
  821. 'Saint Helena',
  822. 'Slovenia',
  823. 'Svalbard and Jan Mayen',
  824. 'Slovakia',
  825. 'Sierra Leone',
  826. 'San Marino',
  827. 'Senegal',
  828. 'Somalia',
  829. 'Suriname',
  830. 'Sao Tome and Principe',
  831. 'El Salvador',
  832. 'Syrian Arab Republic',
  833. 'Swaziland',
  834. 'Turks and Caicos Islands',
  835. 'Chad',
  836. 'French Southern Territories',
  837. 'Togo',
  838. 'Thailand',
  839. 'Tajikistan',
  840. 'Tokelau',
  841. 'Turkmenistan',
  842. 'Tunisia',
  843. 'Tonga',
  844. 'Timor-Leste',
  845. 'Turkey',
  846. 'Trinidad and Tobago',
  847. 'Tuvalu',
  848. 'Taiwan',
  849. 'Tanzania, United Republic of',
  850. 'Ukraine',
  851. 'Uganda',
  852. 'United States Minor Outlying Islands',
  853. 'United States',
  854. 'Uruguay',
  855. 'Uzbekistan',
  856. 'Holy See (Vatican City State)',
  857. 'Saint Vincent and the Grenadines',
  858. 'Venezuela',
  859. 'Virgin Islands, British',
  860. 'Virgin Islands, U.S.',
  861. 'Vietnam',
  862. 'Vanuatu',
  863. 'Wallis and Futuna',
  864. 'Samoa',
  865. 'Yemen',
  866. 'Mayotte',
  867. 'Serbia',
  868. 'South Africa',
  869. 'Zambia',
  870. 'Montenegro',
  871. 'Zimbabwe',
  872. 'Anonymous Proxy',
  873. 'Satellite Provider',
  874. 'Other',
  875. 'Aland Islands',
  876. 'Guernsey',
  877. 'Isle of Man',
  878. 'Jersey',
  879. 'Saint Barthelemy',
  880. 'Saint Martin',
  881. 'Bonaire, Saint Eustatius and Saba',
  882. 'South Sudan',
  883. 'Other',
  884. );
  885. /**
  886. * 2 letters continent codes.
  887. *
  888. * @var array
  889. */
  890. public $GEOIP_CONTINENT_CODES = array(
  891. '--',
  892. 'AS',
  893. 'EU',
  894. 'EU',
  895. 'AS',
  896. 'AS',
  897. 'NA',
  898. 'NA',
  899. 'EU',
  900. 'AS',
  901. 'NA',
  902. 'AF',
  903. 'AN',
  904. 'SA',
  905. 'OC',
  906. 'EU',
  907. 'OC',
  908. 'NA',
  909. 'AS',
  910. 'EU',
  911. 'NA',
  912. 'AS',
  913. 'EU',
  914. 'AF',
  915. 'EU',
  916. 'AS',
  917. 'AF',
  918. 'AF',
  919. 'NA',
  920. 'AS',
  921. 'SA',
  922. 'SA',
  923. 'NA',
  924. 'AS',
  925. 'AN',
  926. 'AF',
  927. 'EU',
  928. 'NA',
  929. 'NA',
  930. 'AS',
  931. 'AF',
  932. 'AF',
  933. 'AF',
  934. 'EU',
  935. 'AF',
  936. 'OC',
  937. 'SA',
  938. 'AF',
  939. 'AS',
  940. 'SA',
  941. 'NA',
  942. 'NA',
  943. 'AF',
  944. 'AS',
  945. 'AS',
  946. 'EU',
  947. 'EU',
  948. 'AF',
  949. 'EU',
  950. 'NA',
  951. 'NA',
  952. 'AF',
  953. 'SA',
  954. 'EU',
  955. 'AF',
  956. 'AF',
  957. 'AF',
  958. 'EU',
  959. 'AF',
  960. 'EU',
  961. 'OC',
  962. 'SA',
  963. 'OC',
  964. 'EU',
  965. 'EU',
  966. 'NA',
  967. 'AF',
  968. 'EU',
  969. 'NA',
  970. 'AS',
  971. 'SA',
  972. 'AF',
  973. 'EU',
  974. 'NA',
  975. 'AF',
  976. 'AF',
  977. 'NA',
  978. 'AF',
  979. 'EU',
  980. 'AN',
  981. 'NA',
  982. 'OC',
  983. 'AF',
  984. 'SA',
  985. 'AS',
  986. 'AN',
  987. 'NA',
  988. 'EU',
  989. 'NA',
  990. 'EU',
  991. 'AS',
  992. 'EU',
  993. 'AS',
  994. 'AS',
  995. 'AS',
  996. 'AS',
  997. 'AS',
  998. 'EU',
  999. 'EU',
  1000. 'NA',
  1001. 'AS',
  1002. 'AS',
  1003. 'AF',
  1004. 'AS',
  1005. 'AS',
  1006. 'OC',
  1007. 'AF',
  1008. 'NA',
  1009. 'AS',
  1010. 'AS',
  1011. 'AS',
  1012. 'NA',
  1013. 'AS',
  1014. 'AS',
  1015. 'AS',
  1016. 'NA',
  1017. 'EU',
  1018. 'AS',
  1019. 'AF',
  1020. 'AF',
  1021. 'EU',
  1022. 'EU',
  1023. 'EU',
  1024. 'AF',
  1025. 'AF',
  1026. 'EU',
  1027. 'EU',
  1028. 'AF',
  1029. 'OC',
  1030. 'EU',
  1031. 'AF',
  1032. 'AS',
  1033. 'AS',
  1034. 'AS',
  1035. 'OC',
  1036. 'NA',
  1037. 'AF',
  1038. 'NA',
  1039. 'EU',
  1040. 'AF',
  1041. 'AS',
  1042. 'AF',
  1043. 'NA',
  1044. 'AS',
  1045. 'AF',
  1046. 'AF',
  1047. 'OC',
  1048. 'AF',
  1049. 'OC',
  1050. 'AF',
  1051. 'NA',
  1052. 'EU',
  1053. 'EU',
  1054. 'AS',
  1055. 'OC',
  1056. 'OC',
  1057. 'OC',
  1058. 'AS',
  1059. 'NA',
  1060. 'SA',
  1061. 'OC',
  1062. 'OC',
  1063. 'AS',
  1064. 'AS',
  1065. 'EU',
  1066. 'NA',
  1067. 'OC',
  1068. 'NA',
  1069. 'AS',
  1070. 'EU',
  1071. 'OC',
  1072. 'SA',
  1073. 'AS',
  1074. 'AF',
  1075. 'EU',
  1076. 'EU',
  1077. 'AF',
  1078. 'AS',
  1079. 'OC',
  1080. 'AF',
  1081. 'AF',
  1082. 'EU',
  1083. 'AS',
  1084. 'AF',
  1085. 'EU',
  1086. 'EU',
  1087. 'EU',
  1088. 'AF',
  1089. 'EU',
  1090. 'AF',
  1091. 'AF',
  1092. 'SA',
  1093. 'AF',
  1094. 'NA',
  1095. 'AS',
  1096. 'AF',
  1097. 'NA',
  1098. 'AF',
  1099. 'AN',
  1100. 'AF',
  1101. 'AS',
  1102. 'AS',
  1103. 'OC',
  1104. 'AS',
  1105. 'AF',
  1106. 'OC',
  1107. 'AS',
  1108. 'EU',
  1109. 'NA',
  1110. 'OC',
  1111. 'AS',
  1112. 'AF',
  1113. 'EU',
  1114. 'AF',
  1115. 'OC',
  1116. 'NA',
  1117. 'SA',
  1118. 'AS',
  1119. 'EU',
  1120. 'NA',
  1121. 'SA',
  1122. 'NA',
  1123. 'NA',
  1124. 'AS',
  1125. 'OC',
  1126. 'OC',
  1127. 'OC',
  1128. 'AS',
  1129. 'AF',
  1130. 'EU',
  1131. 'AF',
  1132. 'AF',
  1133. 'EU',
  1134. 'AF',
  1135. '--',
  1136. '--',
  1137. '--',
  1138. 'EU',
  1139. 'EU',
  1140. 'EU',
  1141. 'EU',
  1142. 'NA',
  1143. 'NA',
  1144. 'NA',
  1145. 'AF',
  1146. '--',
  1147. );
  1148. /** @var WC_Logger Logger instance */
  1149. public static $log = false;
  1150. /**
  1151. * Logging method.
  1152. *
  1153. * @param string $message Log message.
  1154. * @param string $level Optional. Default 'info'.
  1155. * emergency|alert|critical|error|warning|notice|info|debug
  1156. */
  1157. public static function log( $message, $level = 'info' ) {
  1158. if ( empty( self::$log ) ) {
  1159. self::$log = wc_get_logger();
  1160. }
  1161. self::$log->log( $level, $message, array( 'source' => 'geoip' ) );
  1162. }
  1163. /**
  1164. * Open geoip file.
  1165. *
  1166. * @param string $filename
  1167. * @param int $flags
  1168. */
  1169. public function geoip_open( $filename, $flags ) {
  1170. $this->flags = $flags;
  1171. if ( $this->flags & self::GEOIP_SHARED_MEMORY ) {
  1172. $this->shmid = @shmop_open( self::GEOIP_SHM_KEY, 'a', 0, 0 );
  1173. } else {
  1174. if ( $this->filehandle = fopen( $filename, 'rb' ) ) {
  1175. if ( $this->flags & self::GEOIP_MEMORY_CACHE ) {
  1176. $s_array = fstat( $this->filehandle );
  1177. $this->memory_buffer = fread( $this->filehandle, $s_array['size'] );
  1178. }
  1179. } else {
  1180. $this->log( 'GeoIP API: Can not open ' . $filename, 'error' );
  1181. }
  1182. }
  1183. $this->_setup_segments();
  1184. }
  1185. /**
  1186. * Setup segments.
  1187. *
  1188. * @return WC_Geo_IP instance
  1189. */
  1190. private function _setup_segments() {
  1191. $this->databaseType = self::GEOIP_COUNTRY_EDITION;
  1192. $this->record_length = self::STANDARD_RECORD_LENGTH;
  1193. if ( $this->flags & self::GEOIP_SHARED_MEMORY ) {
  1194. $offset = @shmop_size( $this->shmid ) - 3;
  1195. for ( $i = 0; $i < self::STRUCTURE_INFO_MAX_SIZE; $i++ ) {
  1196. $delim = @shmop_read( $this->shmid, $offset, 3 );
  1197. $offset += 3;
  1198. if ( ( chr( 255 ) . chr( 255 ) . chr( 255 ) ) == $delim ) {
  1199. $this->databaseType = ord( @shmop_read( $this->shmid, $offset, 1 ) );
  1200. if ( $this->databaseType >= 106 ) {
  1201. $this->databaseType -= 105;
  1202. }
  1203. $offset++;
  1204. if ( self::GEOIP_REGION_EDITION_REV0 == $this->databaseType ) {
  1205. $this->databaseSegments = self::GEOIP_STATE_BEGIN_REV0;
  1206. } elseif ( self::GEOIP_REGION_EDITION_REV1 == $this->databaseType ) {
  1207. $this->databaseSegments = self::GEOIP_STATE_BEGIN_REV1;
  1208. } elseif ( ( self::GEOIP_CITY_EDITION_REV0 == $this->databaseType )
  1209. || ( self::GEOIP_CITY_EDITION_REV1 == $this->databaseType )
  1210. || ( self::GEOIP_ORG_EDITION == $this->databaseType )
  1211. || ( self::GEOIP_ORG_EDITION_V6 == $this->databaseType )
  1212. || ( self::GEOIP_DOMAIN_EDITION == $this->databaseType )
  1213. || ( self::GEOIP_DOMAIN_EDITION_V6 == $this->databaseType )
  1214. || ( self::GEOIP_ISP_EDITION == $this->databaseType )
  1215. || ( self::GEOIP_ISP_EDITION_V6 == $this->databaseType )
  1216. || ( self::GEOIP_USERTYPE_EDITION == $this->databaseType )
  1217. || ( self::GEOIP_USERTYPE_EDITION_V6 == $this->databaseType )
  1218. || ( self::GEOIP_LOCATIONA_EDITION == $this->databaseType )
  1219. || ( self::GEOIP_ACCURACYRADIUS_EDITION == $this->databaseType )
  1220. || ( self::GEOIP_CITY_EDITION_REV0_V6 == $this->databaseType )
  1221. || ( self::GEOIP_CITY_EDITION_REV1_V6 == $this->databaseType )
  1222. || ( self::GEOIP_NETSPEED_EDITION_REV1 == $this->databaseType )
  1223. || ( self::GEOIP_NETSPEED_EDITION_REV1_V6 == $this->databaseType )
  1224. || ( self::GEOIP_ASNUM_EDITION == $this->databaseType )
  1225. || ( self::GEOIP_ASNUM_EDITION_V6 == $this->databaseType )
  1226. ) {
  1227. $this->databaseSegments = 0;
  1228. $buf = @shmop_read( $this->shmid, $offset, self::SEGMENT_RECORD_LENGTH );
  1229. for ( $j = 0; $j < self::SEGMENT_RECORD_LENGTH; $j++ ) {
  1230. $this->databaseSegments += ( ord( $buf[ $j ] ) << ( $j * 8 ) );
  1231. }
  1232. if ( ( self::GEOIP_ORG_EDITION == $this->databaseType )
  1233. || ( self::GEOIP_ORG_EDITION_V6 == $this->databaseType )
  1234. || ( self::GEOIP_DOMAIN_EDITION == $this->databaseType )
  1235. || ( self::GEOIP_DOMAIN_EDITION_V6 == $this->databaseType )
  1236. || ( self::GEOIP_ISP_EDITION == $this->databaseType )
  1237. || ( self::GEOIP_ISP_EDITION_V6 == $this->databaseType )
  1238. ) {
  1239. $this->record_length = self::ORG_RECORD_LENGTH;
  1240. }
  1241. }
  1242. break;
  1243. } else {
  1244. $offset -= 4;
  1245. }
  1246. }
  1247. if ( ( self::GEOIP_COUNTRY_EDITION == $this->databaseType )
  1248. || ( self::GEOIP_COUNTRY_EDITION_V6 == $this->databaseType )
  1249. || ( self::GEOIP_PROXY_EDITION == $this->databaseType )
  1250. || ( self::GEOIP_NETSPEED_EDITION == $this->databaseType )
  1251. ) {
  1252. $this->databaseSegments = self::GEOIP_COUNTRY_BEGIN;
  1253. }
  1254. } else {
  1255. $filepos = ftell( $this->filehandle );
  1256. fseek( $this->filehandle, -3, SEEK_END );
  1257. for ( $i = 0; $i < self::STRUCTURE_INFO_MAX_SIZE; $i++ ) {
  1258. $delim = fread( $this->filehandle, 3 );
  1259. if ( ( chr( 255 ) . chr( 255 ) . chr( 255 ) ) == $delim ) {
  1260. $this->databaseType = ord( fread( $this->filehandle, 1 ) );
  1261. if ( $this->databaseType >= 106 ) {
  1262. $this->databaseType -= 105;
  1263. }
  1264. if ( self::GEOIP_REGION_EDITION_REV0 == $this->databaseType ) {
  1265. $this->databaseSegments = self::GEOIP_STATE_BEGIN_REV0;
  1266. } elseif ( self::GEOIP_REGION_EDITION_REV1 == $this->databaseType ) {
  1267. $this->databaseSegments = self::GEOIP_STATE_BEGIN_REV1;
  1268. } elseif ( ( self::GEOIP_CITY_EDITION_REV0 == $this->databaseType )
  1269. || ( self::GEOIP_CITY_EDITION_REV1 == $this->databaseType )
  1270. || ( self::GEOIP_CITY_EDITION_REV0_V6 == $this->databaseType )
  1271. || ( self::GEOIP_CITY_EDITION_REV1_V6 == $this->databaseType )
  1272. || ( self::GEOIP_ORG_EDITION == $this->databaseType )
  1273. || ( self::GEOIP_DOMAIN_EDITION == $this->databaseType )
  1274. || ( self::GEOIP_ISP_EDITION == $this->databaseType )
  1275. || ( self::GEOIP_ORG_EDITION_V6 == $this->databaseType )
  1276. || ( self::GEOIP_DOMAIN_EDITION_V6 == $this->databaseType )
  1277. || ( self::GEOIP_ISP_EDITION_V6 == $this->databaseType )
  1278. || ( self::GEOIP_LOCATIONA_EDITION == $this->databaseType )
  1279. || ( self::GEOIP_ACCURACYRADIUS_EDITION == $this->databaseType )
  1280. || ( self::GEOIP_NETSPEED_EDITION_REV1 == $this->databaseType )
  1281. || ( self::GEOIP_NETSPEED_EDITION_REV1_V6 == $this->databaseType )
  1282. || ( self::GEOIP_USERTYPE_EDITION == $this->databaseType )
  1283. || ( self::GEOIP_USERTYPE_EDITION_V6 == $this->databaseType )
  1284. || ( self::GEOIP_ASNUM_EDITION == $this->databaseType )
  1285. || ( self::GEOIP_ASNUM_EDITION_V6 == $this->databaseType )
  1286. ) {
  1287. $this->databaseSegments = 0;
  1288. $buf = fread( $this->filehandle, self::SEGMENT_RECORD_LENGTH );
  1289. for ( $j = 0; $j < self::SEGMENT_RECORD_LENGTH; $j++ ) {
  1290. $this->databaseSegments += ( ord( $buf[ $j ] ) << ( $j * 8 ) );
  1291. }
  1292. if ( ( self::GEOIP_ORG_EDITION == $this->databaseType )
  1293. || ( self::GEOIP_DOMAIN_EDITION == $this->databaseType )
  1294. || ( self::GEOIP_ISP_EDITION == $this->databaseType )
  1295. || ( self::GEOIP_ORG_EDITION_V6 == $this->databaseType )
  1296. || ( self::GEOIP_DOMAIN_EDITION_V6 == $this->databaseType )
  1297. || ( self::GEOIP_ISP_EDITION_V6 == $this->databaseType )
  1298. ) {
  1299. $this->record_length = self::ORG_RECORD_LENGTH;
  1300. }
  1301. }
  1302. break;
  1303. } else {
  1304. fseek( $this->filehandle, -4, SEEK_CUR );
  1305. }
  1306. }
  1307. if ( ( self::GEOIP_COUNTRY_EDITION == $this->databaseType )
  1308. || ( self::GEOIP_COUNTRY_EDITION_V6 == $this->databaseType )
  1309. || ( self::GEOIP_PROXY_EDITION == $this->databaseType )
  1310. || ( self::GEOIP_NETSPEED_EDITION == $this->databaseType )
  1311. ) {
  1312. $this->databaseSegments = self::GEOIP_COUNTRY_BEGIN;
  1313. }
  1314. fseek( $this->filehandle, $filepos, SEEK_SET );
  1315. }
  1316. return $this;
  1317. }
  1318. /**
  1319. * Close geoip file.
  1320. *
  1321. * @return bool
  1322. */
  1323. public function geoip_close() {
  1324. if ( $this->flags & self::GEOIP_SHARED_MEMORY ) {
  1325. return true;
  1326. }
  1327. return fclose( $this->filehandle );
  1328. }
  1329. /**
  1330. * Common get record.
  1331. *
  1332. * @param string $seek_country
  1333. * @return WC_Geo_IP_Record instance
  1334. */
  1335. private function _common_get_record( $seek_country ) {
  1336. // workaround php's broken substr, strpos, etc handling with
  1337. // mbstring.func_overload and mbstring.internal_encoding
  1338. $mbExists = extension_loaded( 'mbstring' );
  1339. if ( $mbExists ) {
  1340. $enc = mb_internal_encoding();
  1341. mb_internal_encoding( 'ISO-8859-1' );
  1342. }
  1343. $record_pointer = $seek_country + ( 2 * $this->record_length - 1 ) * $this->databaseSegments;
  1344. if ( $this->flags & self::GEOIP_MEMORY_CACHE ) {
  1345. $record_buf = substr( $this->memory_buffer, $record_pointer, FULL_RECORD_LENGTH );
  1346. } elseif ( $this->flags & self::GEOIP_SHARED_MEMORY ) {
  1347. $record_buf = @shmop_read( $this->shmid, $record_pointer, FULL_RECORD_LENGTH );
  1348. } else {
  1349. fseek( $this->filehandle, $record_pointer, SEEK_SET );
  1350. $record_buf = fread( $this->filehandle, FULL_RECORD_LENGTH );
  1351. }
  1352. $record = new WC_Geo_IP_Record();
  1353. $record_buf_pos = 0;
  1354. $char = ord( substr( $record_buf, $record_buf_pos, 1 ) );
  1355. $record->country_code = $this->GEOIP_COUNTRY_CODES[ $char ];
  1356. $record->country_code3 = $this->GEOIP_COUNTRY_CODES3[ $char ];
  1357. $record->country_name = $this->GEOIP_COUNTRY_NAMES[ $char ];
  1358. $record->continent_code = $this->GEOIP_CONTINENT_CODES[ $char ];
  1359. $str_length = 0;
  1360. $record_buf_pos++;
  1361. // Get region
  1362. $char = ord( substr( $record_buf, $record_buf_pos + $str_length, 1 ) );
  1363. while ( 0 != $char ) {
  1364. $str_length++;
  1365. $char = ord( substr( $record_buf, $record_buf_pos + $str_length, 1 ) );
  1366. }
  1367. if ( $str_length > 0 ) {
  1368. $record->region = substr( $record_buf, $record_buf_pos, $str_length );
  1369. }
  1370. $record_buf_pos += $str_length + 1;
  1371. $str_length = 0;
  1372. // Get city
  1373. $char = ord( substr( $record_buf, $record_buf_pos + $str_length, 1 ) );
  1374. while ( 0 != $char ) {
  1375. $str_length++;
  1376. $char = ord( substr( $record_buf, $record_buf_pos + $str_length, 1 ) );
  1377. }
  1378. if ( $str_length > 0 ) {
  1379. $record->city = substr( $record_buf, $record_buf_pos, $str_length );
  1380. }
  1381. $record_buf_pos += $str_length + 1;
  1382. $str_length = 0;
  1383. // Get postal code
  1384. $char = ord( substr( $record_buf, $record_buf_pos + $str_length, 1 ) );
  1385. while ( 0 != $char ) {
  1386. $str_length++;
  1387. $char = ord( substr( $record_buf, $record_buf_pos + $str_length, 1 ) );
  1388. }
  1389. if ( $str_length > 0 ) {
  1390. $record->postal_code = substr( $record_buf, $record_buf_pos, $str_length );
  1391. }
  1392. $record_buf_pos += $str_length + 1;
  1393. // Get latitude and longitude
  1394. $latitude = 0;
  1395. $longitude = 0;
  1396. for ( $j = 0; $j < 3; ++$j ) {
  1397. $char = ord( substr( $record_buf, $record_buf_pos++, 1 ) );
  1398. $latitude += ( $char << ( $j * 8 ) );
  1399. }
  1400. $record->latitude = ( $latitude / 10000 ) - 180;
  1401. for ( $j = 0; $j < 3; ++$j ) {
  1402. $char = ord( substr( $record_buf, $record_buf_pos++, 1 ) );
  1403. $longitude += ( $char << ( $j * 8 ) );
  1404. }
  1405. $record->longitude = ( $longitude / 10000 ) - 180;
  1406. if ( self::GEOIP_CITY_EDITION_REV1 == $this->databaseType ) {
  1407. $metroarea_combo = 0;
  1408. if ( 'US' === $record->country_code ) {
  1409. for ( $j = 0; $j < 3; ++$j ) {
  1410. $char = ord( substr( $record_buf, $record_buf_pos++, 1 ) );
  1411. $metroarea_combo += ( $char << ( $j * 8 ) );
  1412. }
  1413. $record->metro_code = $record->dma_code = floor( $metroarea_combo / 1000 );
  1414. $record->area_code = $metroarea_combo % 1000;
  1415. }
  1416. }
  1417. if ( $mbExists ) {
  1418. mb_internal_encoding( $enc );
  1419. }
  1420. return $record;
  1421. }
  1422. /**
  1423. * Get record.
  1424. *
  1425. * @param int $ipnum
  1426. * @return WC_Geo_IP_Record instance
  1427. */
  1428. private function _get_record( $ipnum ) {
  1429. $seek_country = $this->_geoip_seek_country( $ipnum );
  1430. if ( $seek_country == $this->databaseSegments ) {
  1431. return null;
  1432. }
  1433. return $this->_common_get_record( $seek_country );
  1434. }
  1435. /**
  1436. * Seek country IPv6.
  1437. *
  1438. * @param int $ipnum
  1439. * @return string
  1440. */
  1441. public function _geoip_seek_country_v6( $ipnum ) {
  1442. // arrays from unpack start with offset 1
  1443. // yet another php mystery. array_merge work around
  1444. // this broken behaviour
  1445. $v6vec = array_merge( unpack( 'C16', $ipnum ) );
  1446. $offset = 0;
  1447. for ( $depth = 127; $depth >= 0; --$depth ) {
  1448. if ( $this->flags & self::GEOIP_MEMORY_CACHE ) {
  1449. $buf = $this->_safe_substr(
  1450. $this->memory_buffer,
  1451. 2 * $this->record_length * $offset,
  1452. 2 * $this->record_length
  1453. );
  1454. } elseif ( $this->flags & self::GEOIP_SHARED_MEMORY ) {
  1455. $buf = @shmop_read(
  1456. $this->shmid,
  1457. 2 * $this->record_length * $offset,
  1458. 2 * $this->record_length
  1459. );
  1460. } else {
  1461. if ( 0 != fseek( $this->filehandle, 2 * $this->record_length * $offset, SEEK_SET ) ) {
  1462. break;
  1463. }
  1464. $buf = fread( $this->filehandle, 2 * $this->record_length );
  1465. }
  1466. $x = array( 0, 0 );
  1467. for ( $i = 0; $i < 2; ++$i ) {
  1468. for ( $j = 0; $j < $this->record_length; ++$j ) {
  1469. $x[ $i ] += ord( $buf[ $this->record_length * $i + $j ] ) << ( $j * 8 );
  1470. }
  1471. }
  1472. $bnum = 127 - $depth;
  1473. $idx = $bnum >> 3;
  1474. $b_mask = 1 << ( $bnum & 7 ^ 7 );
  1475. if ( ( $v6vec[ $idx ] & $b_mask ) > 0 ) {
  1476. if ( $x[1] >= $this->databaseSegments ) {
  1477. return $x[1];
  1478. }
  1479. $offset = $x[1];
  1480. } else {
  1481. if ( $x[0] >= $this->databaseSegments ) {
  1482. return $x[0];
  1483. }
  1484. $offset = $x[0];
  1485. }
  1486. }
  1487. $this->log( 'GeoIP API: Error traversing database - perhaps it is corrupt?', 'error' );
  1488. return false;
  1489. }
  1490. /**
  1491. * Seek country.
  1492. *
  1493. * @param int $ipnum
  1494. * @return string
  1495. */
  1496. private function _geoip_seek_country( $ipnum ) {
  1497. $offset = 0;
  1498. for ( $depth = 31; $depth >= 0; --$depth ) {
  1499. if ( $this->flags & self::GEOIP_MEMORY_CACHE ) {
  1500. $buf = $this->_safe_substr(
  1501. $this->memory_buffer,
  1502. 2 * $this->record_length * $offset,
  1503. 2 * $this->record_length
  1504. );
  1505. } elseif ( $this->flags & self::GEOIP_SHARED_MEMORY ) {
  1506. $buf = @shmop_read(
  1507. $this->shmid,
  1508. 2 * $this->record_length * $offset,
  1509. 2 * $this->record_length
  1510. );
  1511. } else {
  1512. if ( 0 != fseek( $this->filehandle, 2 * $this->record_length * $offset, SEEK_SET ) ) {
  1513. break;
  1514. }
  1515. $buf = fread( $this->filehandle, 2 * $this->record_length );
  1516. }
  1517. $x = array( 0, 0 );
  1518. for ( $i = 0; $i < 2; ++$i ) {
  1519. for ( $j = 0; $j < $this->record_length; ++$j ) {
  1520. $x[ $i ] += ord( $buf[ $this->record_length * $i + $j ] ) << ( $j * 8 );
  1521. }
  1522. }
  1523. if ( $ipnum & ( 1 << $depth ) ) {
  1524. if ( $x[1] >= $this->databaseSegments ) {
  1525. return $x[1];
  1526. }
  1527. $offset = $x[1];
  1528. } else {
  1529. if ( $x[0] >= $this->databaseSegments ) {
  1530. return $x[0];
  1531. }
  1532. $offset = $x[0];
  1533. }
  1534. }
  1535. $this->log( 'GeoIP API: Error traversing database - perhaps it is corrupt?', 'error' );
  1536. return false;
  1537. }
  1538. /**
  1539. * Record by addr.
  1540. *
  1541. * @param string $addr
  1542. *
  1543. * @return WC_Geo_IP_Record
  1544. */
  1545. public function geoip_record_by_addr( $addr ) {
  1546. if ( null == $addr ) {
  1547. return 0;
  1548. }
  1549. $ipnum = ip2long( $addr );
  1550. return $this->_get_record( $ipnum );
  1551. }
  1552. /**
  1553. * Country ID by addr IPv6.
  1554. *
  1555. * @param string $addr
  1556. * @return int|bool
  1557. */
  1558. public function geoip_country_id_by_addr_v6( $addr ) {
  1559. if ( ! defined( 'AF_INET6' ) ) {
  1560. $this->log( 'GEOIP (geoip_country_id_by_addr_v6): PHP was compiled with --disable-ipv6 option' );
  1561. return false;
  1562. }
  1563. $ipnum = inet_pton( $addr );
  1564. return $this->_geoip_seek_country_v6( $ipnum ) - self::GEOIP_COUNTRY_BEGIN;
  1565. }
  1566. /**
  1567. * Country ID by addr.
  1568. *
  1569. * @param string $addr
  1570. * @return int
  1571. */
  1572. public function geoip_country_id_by_addr( $addr ) {
  1573. $ipnum = ip2long( $addr );
  1574. return $this->_geoip_seek_country( $ipnum ) - self::GEOIP_COUNTRY_BEGIN;
  1575. }
  1576. /**
  1577. * Country code by addr IPv6.
  1578. *
  1579. * @param string $addr
  1580. * @return string
  1581. */
  1582. public function geoip_country_code_by_addr_v6( $addr ) {
  1583. $country_id = $this->geoip_country_id_by_addr_v6( $addr );
  1584. if ( false !== $country_id && isset( $this->GEOIP_COUNTRY_CODES[ $country_id ] ) ) {
  1585. return $this->GEOIP_COUNTRY_CODES[ $country_id ];
  1586. }
  1587. return false;
  1588. }
  1589. /**
  1590. * Country code by addr.
  1591. *
  1592. * @param string $addr
  1593. * @return string
  1594. */
  1595. public function geoip_country_code_by_addr( $addr ) {
  1596. if ( self::GEOIP_CITY_EDITION_REV1 == $this->databaseType ) {
  1597. $record = $this->geoip_record_by_addr( $addr );
  1598. if ( false !== $record ) {
  1599. return $record->country_code;
  1600. }
  1601. } else {
  1602. $country_id = $this->geoip_country_id_by_addr( $addr );
  1603. if ( false !== $country_id && isset( $this->GEOIP_COUNTRY_CODES[ $country_id ] ) ) {
  1604. return $this->GEOIP_COUNTRY_CODES[ $country_id ];
  1605. }
  1606. }
  1607. return false;
  1608. }
  1609. /**
  1610. * Encode string.
  1611. *
  1612. * @param string $string
  1613. * @param int $start
  1614. * @param int $length
  1615. * @return string
  1616. */
  1617. private function _safe_substr( $string, $start, $length ) {
  1618. // workaround php's broken substr, strpos, etc handling with
  1619. // mbstring.func_overload and mbstring.internal_encoding
  1620. $mb_exists = extension_loaded( 'mbstring' );
  1621. if ( $mb_exists ) {
  1622. $enc = mb_internal_encoding();
  1623. mb_internal_encoding( 'ISO-8859-1' );
  1624. }
  1625. $buf = substr( $string, $start, $length );
  1626. if ( $mb_exists ) {
  1627. mb_internal_encoding( $enc );
  1628. }
  1629. return $buf;
  1630. }
  1631. }
  1632. /**
  1633. * Geo IP Record class.
  1634. */
  1635. class WC_Geo_IP_Record {
  1636. /**
  1637. * Country code.
  1638. *
  1639. * @var string
  1640. */
  1641. public $country_code;
  1642. /**
  1643. * 3 letters country code.
  1644. *
  1645. * @var string
  1646. */
  1647. public $country_code3;
  1648. /**
  1649. * Country name.
  1650. *
  1651. * @var string
  1652. */
  1653. public $country_name;
  1654. /**
  1655. * Region.
  1656. *
  1657. * @var string
  1658. */
  1659. public $region;
  1660. /**
  1661. * City.
  1662. *
  1663. * @var string
  1664. */
  1665. public $city;
  1666. /**
  1667. * Postal code.
  1668. *
  1669. * @var string
  1670. */
  1671. public $postal_code;
  1672. /**
  1673. * Latitude
  1674. *
  1675. * @var int
  1676. */
  1677. public $latitude;
  1678. /**
  1679. * Longitude.
  1680. *
  1681. * @var int
  1682. */
  1683. public $longitude;
  1684. /**
  1685. * Area code.
  1686. *
  1687. * @var int
  1688. */
  1689. public $area_code;
  1690. /**
  1691. * DMA Code.
  1692. *
  1693. * Metro and DMA code are the same.
  1694. * Use metro code instead.
  1695. *
  1696. * @var float
  1697. */
  1698. public $dma_code;
  1699. /**
  1700. * Metro code.
  1701. *
  1702. * @var float
  1703. */
  1704. public $metro_code;
  1705. /**
  1706. * Continent code.
  1707. *
  1708. * @var string
  1709. */
  1710. public $continent_code;
  1711. }