Compare commits
7 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 8ca117c04c | |||
|
|
f81c3e0649 | ||
|
|
fa183f628b | ||
|
|
1808fa8757 | ||
| a176102cd9 | |||
|
|
f7306b70c8 | ||
|
|
083494bd4d |
2
.hta_config/conf.php
Normal file
2
.hta_config/conf.php
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
<?php
|
||||||
|
define('SITE_NAME', 'Tools');
|
||||||
2
.hta_config/var.php
Normal file
2
.hta_config/var.php
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
<?php
|
||||||
|
$BASE_URL = 'http://localhost:2050';
|
||||||
BIN
.hta_lib/data/GeoLite2-ASN.mmdb
Normal file
BIN
.hta_lib/data/GeoLite2-ASN.mmdb
Normal file
Binary file not shown.
BIN
.hta_lib/data/GeoLite2-City.mmdb
Normal file
BIN
.hta_lib/data/GeoLite2-City.mmdb
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 60 MiB |
261
.hta_lib/maxmind-db-reader/CHANGELOG.md
Normal file
261
.hta_lib/maxmind-db-reader/CHANGELOG.md
Normal file
@@ -0,0 +1,261 @@
|
|||||||
|
CHANGELOG
|
||||||
|
=========
|
||||||
|
|
||||||
|
1.13.1 (2025-11-21)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
* First PIE release. No other changes.
|
||||||
|
|
||||||
|
1.13.0 (2025-11-20)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
* A redundant `filesize()` call in the reader's constructor was removed.
|
||||||
|
Pull request by Pavel Djundik. GitHub #189.
|
||||||
|
|
||||||
|
1.12.1 (2025-05-05)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
* The C extension now checks that the database metadata lookup was
|
||||||
|
successful.
|
||||||
|
|
||||||
|
1.12.0 (2024-11-14)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
* Improve the error handling when the user tries to open a directory
|
||||||
|
with the pure PHP reader.
|
||||||
|
* Improve the typehints on arrays in the PHPDocs.
|
||||||
|
|
||||||
|
1.11.1 (2023-12-01)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
* Resolve warnings when compiling the C extension.
|
||||||
|
* Fix various type issues detected by PHPStan level. Pull request by
|
||||||
|
LauraTaylorUK. GitHub #160.
|
||||||
|
|
||||||
|
1.11.0 (2021-10-18)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
* Replace runtime define of a constant to facilitate opcache preloading.
|
||||||
|
Reported by vedadkajtaz. GitHub #134.
|
||||||
|
* Resolve minor issue found by the Clang static analyzer in the C
|
||||||
|
extension.
|
||||||
|
|
||||||
|
1.10.1 (2021-04-14)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
* Fix a `TypeError` exception in the pure PHP reader when using large
|
||||||
|
databases on 32-bit PHP builds with the `bcmath` extension. Reported
|
||||||
|
by dodo1708. GitHub #124.
|
||||||
|
|
||||||
|
1.10.0 (2021-02-09)
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
* When using the pure PHP reader, unsigned integers up to PHP_MAX_INT
|
||||||
|
will now be integers in PHP rather than strings. Previously integers
|
||||||
|
greater than 2^24 on 32-bit platforms and 2^56 on 64-bit platforms
|
||||||
|
would be strings due to the use of `gmp` or `bcmath` to decode them.
|
||||||
|
Reported by Alejandro Celaya. GitHub #119.
|
||||||
|
|
||||||
|
1.9.0 (2021-01-07)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* The `maxminddb` extension is now buildable on Windows. Pull request
|
||||||
|
by Jan Ehrhardt. GitHub #115.
|
||||||
|
|
||||||
|
1.8.0 (2020-10-01)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* Fixes for PHP 8.0. Pull Request by Remi Collet. GitHub #108.
|
||||||
|
|
||||||
|
1.7.0 (2020-08-07)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* IMPORTANT: PHP 7.2 or greater is now required.
|
||||||
|
* The extension no longer depends on the pure PHP classes in
|
||||||
|
`maxmind-db/reader`. You can use it independently.
|
||||||
|
* Type hints have been added to both the pure PHP implementation
|
||||||
|
and the extension.
|
||||||
|
* The `metadata` method on the reader now returns a new copy of the
|
||||||
|
metadata object rather than the actual object used by the reader.
|
||||||
|
* Work around PHP `is_readable()` bug. Reported by Ben Roberts. GitHub
|
||||||
|
#92.
|
||||||
|
* This is the first release of the extension as a PECL package.
|
||||||
|
GitHub #34.
|
||||||
|
|
||||||
|
1.6.0 (2019-12-19)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* 1.5.0 and 1.5.1 contained a possible memory corruptions when using
|
||||||
|
`getWithPrefixLen`. This has been fixed. Reported by proton-ab.
|
||||||
|
GitHub #96.
|
||||||
|
* The `composer.json` file now conflicts with all versions of the
|
||||||
|
`maxminddb` C extension less than the Composer version. This is to
|
||||||
|
reduce the chance of having an older, conflicting version of the
|
||||||
|
extension installed. You will need to upgrade the extension before
|
||||||
|
running `composer update`. Pull request by Benoît Burnichon. GitHub
|
||||||
|
#97.
|
||||||
|
|
||||||
|
1.5.1 (2019-12-12)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* Minor performance improvements.
|
||||||
|
* Make tests pass with older versions of libmaxminddb. PR by Remi
|
||||||
|
Collet. GitHub #90.
|
||||||
|
* Test enhancements. PR by Chun-Sheng, Li. GitHub #91.
|
||||||
|
|
||||||
|
1.5.0 (2019-09-30)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* PHP 5.6 or greater is now required.
|
||||||
|
* The C extension now supports PHP 8. Pull request by John Boehr.
|
||||||
|
GitHub #87.
|
||||||
|
* A new method, `getWithPrefixLen`, was added to the `Reader` class.
|
||||||
|
This method returns an array containing the record and the prefix
|
||||||
|
length for that record. GitHub #89.
|
||||||
|
|
||||||
|
1.4.1 (2019-01-04)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* The `maxminddb` extension now returns a string when a `uint32`
|
||||||
|
value is greater than `LONG_MAX`. Previously, the value would
|
||||||
|
overflow. This generally only affects 32-bit machines. Reported
|
||||||
|
by Remi Collet. GitHub #79.
|
||||||
|
* For `uint64` values, the `maxminddb` extension now returns an
|
||||||
|
integer rather than a string when the value is less than or equal
|
||||||
|
to `LONG_MAX`. This more closely matches the behavior of the pure
|
||||||
|
PHP reader.
|
||||||
|
|
||||||
|
1.4.0 (2018-11-20)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* The `maxminddb` extension now has the arginfo when using reflection.
|
||||||
|
PR by Remi Collet. GitHub #75.
|
||||||
|
* The `maxminddb` extension now provides `MINFO()` function that
|
||||||
|
displays the extension version and the libmaxminddb version. PR by
|
||||||
|
Remi Collet. GitHub #74.
|
||||||
|
* The `maxminddb` `configure` script now uses `pkg-config` when
|
||||||
|
available to get libmaxmindb build info. PR by Remi Collet.
|
||||||
|
GitHub #73.
|
||||||
|
* The pure PHP reader now correctly decodes integers on 32-bit platforms.
|
||||||
|
Previously, large integers would overflow. Reported by Remi Collet.
|
||||||
|
GitHub #77.
|
||||||
|
* There are small performance improvements for the pure PHP reader.
|
||||||
|
|
||||||
|
1.3.0 (2018-02-21)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* IMPORTANT: The `maxminddb` extension now obeys `open_basedir`. If
|
||||||
|
`open_basedir` is set, you _must_ store the database within the
|
||||||
|
specified directory. Placing the file outside of this directory
|
||||||
|
will result in an exception. Please test your integration before
|
||||||
|
upgrading the extension. This does not affect the pure PHP
|
||||||
|
implementation, which has always had this restriction. Reported
|
||||||
|
by Benoît Burnichon. GitHub #61.
|
||||||
|
* A custom `autoload.php` file is provided for installations without
|
||||||
|
Composer. GitHub #56.
|
||||||
|
|
||||||
|
1.2.0 (2017-10-27)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* PHP 5.4 or greater is now required.
|
||||||
|
* The `Reader` class for the `maxminddb` extension is no longer final.
|
||||||
|
This was change to match the behavior of the pure PHP class.
|
||||||
|
Reported and fixed by venyii. GitHub #52 & #54.
|
||||||
|
|
||||||
|
1.1.3 (2017-01-19)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* Fix incorrect version in `ext/php_maxminddb.h`. GitHub #48.
|
||||||
|
|
||||||
|
1.1.2 (2016-11-22)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* Searching for database metadata only occurs within the last 128KB
|
||||||
|
(128 * 1024 bytes) of the file, speeding detection of corrupt
|
||||||
|
datafiles. Reported by Eric Teubert. GitHub #42.
|
||||||
|
* Suggest relevant extensions when installing with Composer. GitHub #37.
|
||||||
|
|
||||||
|
1.1.1 (2016-09-15)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* Development files were added to the `.gitattributes` as `export-ignore` so
|
||||||
|
that they are not part of the Composer release. Pull request by Michele
|
||||||
|
Locati. GitHub #39.
|
||||||
|
|
||||||
|
1.1.0 (2016-01-04)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* The MaxMind DB extension now supports PHP 7. Pull request by John Boehr.
|
||||||
|
GitHub #27.
|
||||||
|
|
||||||
|
1.0.3 (2015-03-13)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* All uses of `strlen` were removed. This should prevent issues in situations
|
||||||
|
where the function is overloaded or otherwise broken.
|
||||||
|
|
||||||
|
1.0.2 (2015-01-19)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* Previously the MaxMind DB extension would cause a segfault if the Reader
|
||||||
|
object's destructor was called without first having called the constructor.
|
||||||
|
(Reported by Matthias Saou & Juan Peri. GitHub #20.)
|
||||||
|
|
||||||
|
1.0.1 (2015-01-12)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* In the last several releases, the version number in the extension was
|
||||||
|
incorrect. This release is being done to correct it. No other code changes
|
||||||
|
are included.
|
||||||
|
|
||||||
|
1.0.0 (2014-09-22)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* First production release.
|
||||||
|
* In the pure PHP reader, a string length test after `fread()` was replaced
|
||||||
|
with the difference between the start pointer and the end pointer. This
|
||||||
|
provided a 15% speed increase.
|
||||||
|
|
||||||
|
0.3.3 (2014-09-15)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* Clarified behavior of 128-bit type in documentation.
|
||||||
|
* Updated phpunit and fixed some test breakage from the newer version.
|
||||||
|
|
||||||
|
0.3.2 (2014-09-10)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* Fixed invalid reference to global class RuntimeException from namespaced
|
||||||
|
code. Fixed by Steven Don. GitHub issue #15.
|
||||||
|
* Additional documentation of `Metadata` class as well as misc. documentation
|
||||||
|
cleanup.
|
||||||
|
|
||||||
|
0.3.1 (2014-05-01)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* The API now works when `mbstring.func_overload` is set.
|
||||||
|
* BCMath is no longer required. If the decoder encounters a big integer,
|
||||||
|
it will try to use GMP and then BCMath. If both of those fail, it will
|
||||||
|
throw an exception. No databases released by MaxMind currently use big
|
||||||
|
integers.
|
||||||
|
* The API now officially supports HHVM when using the pure PHP reader.
|
||||||
|
|
||||||
|
0.3.0 (2014-02-19)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* This API is now licensed under the Apache License, Version 2.0.
|
||||||
|
* The code for the C extension was cleaned up, fixing several potential
|
||||||
|
issues.
|
||||||
|
|
||||||
|
0.2.0 (2013-10-21)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* Added optional C extension for using libmaxminddb in place of the pure PHP
|
||||||
|
reader.
|
||||||
|
* Significantly improved error handling in pure PHP reader.
|
||||||
|
* Improved performance for IPv4 lookups in an IPv6 database.
|
||||||
|
|
||||||
|
0.1.0 (2013-07-16)
|
||||||
|
------------------
|
||||||
|
|
||||||
|
* Initial release
|
||||||
202
.hta_lib/maxmind-db-reader/LICENSE
Normal file
202
.hta_lib/maxmind-db-reader/LICENSE
Normal file
@@ -0,0 +1,202 @@
|
|||||||
|
|
||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction,
|
||||||
|
and distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by
|
||||||
|
the copyright owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
|
other entities that control, are controlled by, or are under common
|
||||||
|
control with that entity. For the purposes of this definition,
|
||||||
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
|
direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity
|
||||||
|
exercising permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications,
|
||||||
|
including but not limited to software source code, documentation
|
||||||
|
source, and configuration files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical
|
||||||
|
transformation or translation of a Source form, including but
|
||||||
|
not limited to compiled object code, generated documentation,
|
||||||
|
and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or
|
||||||
|
Object form, made available under the License, as indicated by a
|
||||||
|
copyright notice that is included in or attached to the work
|
||||||
|
(an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object
|
||||||
|
form, that is based on (or derived from) the Work and for which the
|
||||||
|
editorial revisions, annotations, elaborations, or other modifications
|
||||||
|
represent, as a whole, an original work of authorship. For the purposes
|
||||||
|
of this License, Derivative Works shall not include works that remain
|
||||||
|
separable from, or merely link (or bind by name) to the interfaces of,
|
||||||
|
the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including
|
||||||
|
the original version of the Work and any modifications or additions
|
||||||
|
to that Work or Derivative Works thereof, that is intentionally
|
||||||
|
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||||
|
or by an individual or Legal Entity authorized to submit on behalf of
|
||||||
|
the copyright owner. For the purposes of this definition, "submitted"
|
||||||
|
means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems,
|
||||||
|
and issue tracking systems that are managed by, or on behalf of, the
|
||||||
|
Licensor for the purpose of discussing and improving the Work, but
|
||||||
|
excluding communication that is conspicuously marked or otherwise
|
||||||
|
designated in writing by the copyright owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||||
|
on behalf of whom a Contribution has been received by Licensor and
|
||||||
|
subsequently incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the
|
||||||
|
Work and such Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
(except as stated in this section) patent license to make, have made,
|
||||||
|
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||||
|
where such license applies only to those patent claims licensable
|
||||||
|
by such Contributor that are necessarily infringed by their
|
||||||
|
Contribution(s) alone or by combination of their Contribution(s)
|
||||||
|
with the Work to which such Contribution(s) was submitted. If You
|
||||||
|
institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||||
|
or a Contribution incorporated within the Work constitutes direct
|
||||||
|
or contributory patent infringement, then any patent licenses
|
||||||
|
granted to You under this License for that Work shall terminate
|
||||||
|
as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution. You may reproduce and distribute copies of the
|
||||||
|
Work or Derivative Works thereof in any medium, with or without
|
||||||
|
modifications, and in Source or Object form, provided that You
|
||||||
|
meet the following conditions:
|
||||||
|
|
||||||
|
(a) You must give any other recipients of the Work or
|
||||||
|
Derivative Works a copy of this License; and
|
||||||
|
|
||||||
|
(b) You must cause any modified files to carry prominent notices
|
||||||
|
stating that You changed the files; and
|
||||||
|
|
||||||
|
(c) You must retain, in the Source form of any Derivative Works
|
||||||
|
that You distribute, all copyright, patent, trademark, and
|
||||||
|
attribution notices from the Source form of the Work,
|
||||||
|
excluding those notices that do not pertain to any part of
|
||||||
|
the Derivative Works; and
|
||||||
|
|
||||||
|
(d) If the Work includes a "NOTICE" text file as part of its
|
||||||
|
distribution, then any Derivative Works that You distribute must
|
||||||
|
include a readable copy of the attribution notices contained
|
||||||
|
within such NOTICE file, excluding those notices that do not
|
||||||
|
pertain to any part of the Derivative Works, in at least one
|
||||||
|
of the following places: within a NOTICE text file distributed
|
||||||
|
as part of the Derivative Works; within the Source form or
|
||||||
|
documentation, if provided along with the Derivative Works; or,
|
||||||
|
within a display generated by the Derivative Works, if and
|
||||||
|
wherever such third-party notices normally appear. The contents
|
||||||
|
of the NOTICE file are for informational purposes only and
|
||||||
|
do not modify the License. You may add Your own attribution
|
||||||
|
notices within Derivative Works that You distribute, alongside
|
||||||
|
or as an addendum to the NOTICE text from the Work, provided
|
||||||
|
that such additional attribution notices cannot be construed
|
||||||
|
as modifying the License.
|
||||||
|
|
||||||
|
You may add Your own copyright statement to Your modifications and
|
||||||
|
may provide additional or different license terms and conditions
|
||||||
|
for use, reproduction, or distribution of Your modifications, or
|
||||||
|
for any such Derivative Works as a whole, provided Your use,
|
||||||
|
reproduction, and distribution of the Work otherwise complies with
|
||||||
|
the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||||
|
any Contribution intentionally submitted for inclusion in the Work
|
||||||
|
by You to the Licensor shall be under the terms and conditions of
|
||||||
|
this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify
|
||||||
|
the terms of any separate license agreement you may have executed
|
||||||
|
with Licensor regarding such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks. This License does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or product names of the Licensor,
|
||||||
|
except as required for reasonable and customary use in describing the
|
||||||
|
origin of the Work and reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||||
|
agreed to in writing, Licensor provides the Work (and each
|
||||||
|
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
implied, including, without limitation, any warranties or conditions
|
||||||
|
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||||
|
appropriateness of using or redistributing the Work and assume any
|
||||||
|
risks associated with Your exercise of permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability. In no event and under no legal theory,
|
||||||
|
whether in tort (including negligence), contract, or otherwise,
|
||||||
|
unless required by applicable law (such as deliberate and grossly
|
||||||
|
negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special,
|
||||||
|
incidental, or consequential damages of any character arising as a
|
||||||
|
result of this License or out of the use or inability to use the
|
||||||
|
Work (including but not limited to damages for loss of goodwill,
|
||||||
|
work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses), even if such Contributor
|
||||||
|
has been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability. While redistributing
|
||||||
|
the Work or Derivative Works thereof, You may choose to offer,
|
||||||
|
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||||
|
or other liability obligations and/or rights consistent with this
|
||||||
|
License. However, in accepting such obligations, You may act only
|
||||||
|
on Your own behalf and on Your sole responsibility, not on behalf
|
||||||
|
of any other Contributor, and only if You agree to indemnify,
|
||||||
|
defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
APPENDIX: How to apply the Apache License to your work.
|
||||||
|
|
||||||
|
To apply the Apache License to your work, attach the following
|
||||||
|
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||||
|
replaced with your own identifying information. (Don't include
|
||||||
|
the brackets!) The text should be enclosed in the appropriate
|
||||||
|
comment syntax for the file format. We also recommend that a
|
||||||
|
file or class name and description of purpose be included on the
|
||||||
|
same "printed page" as the copyright notice for easier
|
||||||
|
identification within third-party archives.
|
||||||
|
|
||||||
|
Copyright [yyyy] [name of copyright owner]
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
214
.hta_lib/maxmind-db-reader/README.md
Normal file
214
.hta_lib/maxmind-db-reader/README.md
Normal file
@@ -0,0 +1,214 @@
|
|||||||
|
# MaxMind DB Reader PHP API #
|
||||||
|
|
||||||
|
## Description ##
|
||||||
|
|
||||||
|
This is the PHP API for reading MaxMind DB files. MaxMind DB is a binary file
|
||||||
|
format that stores data indexed by IP address subnets (IPv4 or IPv6).
|
||||||
|
|
||||||
|
## Installation ##
|
||||||
|
|
||||||
|
### C Extension (Recommended for Performance) ###
|
||||||
|
|
||||||
|
For significantly faster IP lookups, we recommend installing the C extension via
|
||||||
|
[PIE](https://github.com/php/pie):
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pie install maxmind-db/reader-ext
|
||||||
|
```
|
||||||
|
|
||||||
|
The C extension requires the [libmaxminddb](https://github.com/maxmind/libmaxminddb)
|
||||||
|
C library. See the [installation instructions](https://github.com/maxmind/MaxMind-DB-Reader-php-ext#prerequisites)
|
||||||
|
for your platform.
|
||||||
|
|
||||||
|
### Pure PHP (No Compilation Required) ###
|
||||||
|
|
||||||
|
If you prefer not to compile a C extension or need maximum portability, you can
|
||||||
|
install the pure PHP implementation with [Composer](https://getcomposer.org/).
|
||||||
|
|
||||||
|
### Download Composer ###
|
||||||
|
|
||||||
|
To download Composer, run in the root directory of your project:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
curl -sS https://getcomposer.org/installer | php
|
||||||
|
```
|
||||||
|
|
||||||
|
You should now have the file `composer.phar` in your project directory.
|
||||||
|
|
||||||
|
### Install Dependencies ###
|
||||||
|
|
||||||
|
Run in your project root:
|
||||||
|
|
||||||
|
```
|
||||||
|
php composer.phar require maxmind-db/reader:^1.13.1
|
||||||
|
```
|
||||||
|
|
||||||
|
You should now have the files `composer.json` and `composer.lock` as well as
|
||||||
|
the directory `vendor` in your project directory. If you use a version control
|
||||||
|
system, `composer.json` should be added to it.
|
||||||
|
|
||||||
|
### Require Autoloader ###
|
||||||
|
|
||||||
|
After installing the dependencies, you need to require the Composer autoloader
|
||||||
|
from your code:
|
||||||
|
|
||||||
|
```php
|
||||||
|
require 'vendor/autoload.php';
|
||||||
|
```
|
||||||
|
|
||||||
|
## Installation (Standalone) ##
|
||||||
|
|
||||||
|
If you don't want to use Composer for some reason, a custom
|
||||||
|
`autoload.php` is provided for you in the project root. To use the
|
||||||
|
library, simply include that file,
|
||||||
|
|
||||||
|
```php
|
||||||
|
require('/path/to/MaxMind-DB-Reader-php/autoload.php');
|
||||||
|
```
|
||||||
|
|
||||||
|
and then instantiate the reader class normally:
|
||||||
|
|
||||||
|
```php
|
||||||
|
use MaxMind\Db\Reader;
|
||||||
|
$reader = new Reader('example.mmdb');
|
||||||
|
```
|
||||||
|
|
||||||
|
## Installation (RPM)
|
||||||
|
|
||||||
|
RPMs are available in the [official Fedora repository](https://apps.fedoraproject.org/packages/php-maxminddb).
|
||||||
|
|
||||||
|
To install on Fedora, run:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
dnf install php-maxminddb
|
||||||
|
```
|
||||||
|
|
||||||
|
To install on CentOS or RHEL 7, first [enable the EPEL repository](https://fedoraproject.org/wiki/EPEL)
|
||||||
|
and then run:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
yum install php-maxminddb
|
||||||
|
```
|
||||||
|
|
||||||
|
Please note that these packages are *not* maintained by MaxMind.
|
||||||
|
|
||||||
|
## Usage ##
|
||||||
|
|
||||||
|
## Example ##
|
||||||
|
|
||||||
|
```php
|
||||||
|
<?php
|
||||||
|
require_once 'vendor/autoload.php';
|
||||||
|
|
||||||
|
use MaxMind\Db\Reader;
|
||||||
|
|
||||||
|
$ipAddress = '24.24.24.24';
|
||||||
|
$databaseFile = 'GeoIP2-City.mmdb';
|
||||||
|
|
||||||
|
$reader = new Reader($databaseFile);
|
||||||
|
|
||||||
|
// get returns just the record for the IP address
|
||||||
|
print_r($reader->get($ipAddress));
|
||||||
|
|
||||||
|
// getWithPrefixLen returns an array containing the record and the
|
||||||
|
// associated prefix length for that record.
|
||||||
|
print_r($reader->getWithPrefixLen($ipAddress));
|
||||||
|
|
||||||
|
$reader->close();
|
||||||
|
```
|
||||||
|
|
||||||
|
## Optional PHP C Extension ##
|
||||||
|
|
||||||
|
MaxMind provides an optional C extension that is a drop-in replacement for
|
||||||
|
`MaxMind\Db\Reader`. In order to use this extension, you must install the
|
||||||
|
Reader API as described above and install the extension as described below. If
|
||||||
|
you are using an autoloader, no changes to your code should be necessary.
|
||||||
|
|
||||||
|
### Installing Extension via PIE (Recommended) ###
|
||||||
|
|
||||||
|
We recommend installing the extension via [PIE](https://github.com/php/pie):
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pie install maxmind-db/reader-ext
|
||||||
|
```
|
||||||
|
|
||||||
|
See the [extension repository](https://github.com/maxmind/MaxMind-DB-Reader-php-ext#prerequisites)
|
||||||
|
for prerequisites including libmaxminddb installation instructions.
|
||||||
|
|
||||||
|
### Installing Extension via PECL (Legacy) ###
|
||||||
|
|
||||||
|
First install [libmaxminddb](https://github.com/maxmind/libmaxminddb) as
|
||||||
|
described in its [README.md
|
||||||
|
file](https://github.com/maxmind/libmaxminddb/blob/main/README.md#installing-from-a-tarball).
|
||||||
|
After successfully installing libmaxmindb, you may install the extension
|
||||||
|
from [PECL](https://pecl.php.net/package/maxminddb):
|
||||||
|
|
||||||
|
```
|
||||||
|
pecl install maxminddb
|
||||||
|
```
|
||||||
|
|
||||||
|
### Installing Extension from Source ###
|
||||||
|
|
||||||
|
Alternatively, you may install it from the source. To do so, run the following
|
||||||
|
commands from the top-level directory of this distribution:
|
||||||
|
|
||||||
|
```
|
||||||
|
cd ext
|
||||||
|
phpize
|
||||||
|
./configure
|
||||||
|
make
|
||||||
|
make test
|
||||||
|
sudo make install
|
||||||
|
```
|
||||||
|
|
||||||
|
You then must load your extension. The recommended method is to add the
|
||||||
|
following to your `php.ini` file:
|
||||||
|
|
||||||
|
```
|
||||||
|
extension=maxminddb.so
|
||||||
|
```
|
||||||
|
|
||||||
|
Note: You may need to install the PHP development package on your OS such as
|
||||||
|
php5-dev for Debian-based systems or php-devel for RedHat/Fedora-based ones.
|
||||||
|
|
||||||
|
## 128-bit Integer Support ##
|
||||||
|
|
||||||
|
The MaxMind DB format includes 128-bit unsigned integer as a type. Although
|
||||||
|
no MaxMind-distributed database currently makes use of this type, both the
|
||||||
|
pure PHP reader and the C extension support this type. The pure PHP reader
|
||||||
|
requires gmp or bcmath to read databases with 128-bit unsigned integers.
|
||||||
|
|
||||||
|
The integer is currently returned as a hexadecimal string (prefixed with "0x")
|
||||||
|
by the C extension and a decimal string (no prefix) by the pure PHP reader.
|
||||||
|
Any change to make the reader implementations always return either a
|
||||||
|
hexadecimal or decimal representation of the integer will NOT be considered a
|
||||||
|
breaking change.
|
||||||
|
|
||||||
|
## Support ##
|
||||||
|
|
||||||
|
Please report all issues with this code using the [GitHub issue tracker](https://github.com/maxmind/MaxMind-DB-Reader-php/issues).
|
||||||
|
|
||||||
|
If you are having an issue with a MaxMind service that is not specific to the
|
||||||
|
client API, please see [our support page](https://www.maxmind.com/en/support).
|
||||||
|
|
||||||
|
## Requirements ##
|
||||||
|
|
||||||
|
This library requires PHP 7.2 or greater.
|
||||||
|
|
||||||
|
The GMP or BCMath extension may be required to read some databases
|
||||||
|
using the pure PHP API.
|
||||||
|
|
||||||
|
## Contributing ##
|
||||||
|
|
||||||
|
Patches and pull requests are encouraged. All code should follow the PSR-1 and
|
||||||
|
PSR-2 style guidelines. Please include unit tests whenever possible.
|
||||||
|
|
||||||
|
## Versioning ##
|
||||||
|
|
||||||
|
The MaxMind DB Reader PHP API uses [Semantic Versioning](https://semver.org/).
|
||||||
|
|
||||||
|
## Copyright and License ##
|
||||||
|
|
||||||
|
This software is Copyright (c) 2014-2025 by MaxMind, Inc.
|
||||||
|
|
||||||
|
This is free software, licensed under the Apache License, Version 2.0.
|
||||||
47
.hta_lib/maxmind-db-reader/autoload.php
Normal file
47
.hta_lib/maxmind-db-reader/autoload.php
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* PSR-4 autoloader implementation for the MaxMind\DB namespace.
|
||||||
|
* First we define the 'mmdb_autoload' function, and then we register
|
||||||
|
* it with 'spl_autoload_register' so that PHP knows to use it.
|
||||||
|
*
|
||||||
|
* @param mixed $class
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Automatically include the file that defines <code>class</code>.
|
||||||
|
*
|
||||||
|
* @param string $class
|
||||||
|
* the name of the class to load
|
||||||
|
*/
|
||||||
|
function mmdb_autoload($class): void
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* A project-specific mapping between the namespaces and where
|
||||||
|
* they're located. By convention, we include the trailing
|
||||||
|
* slashes. The one-element array here simply makes things easy
|
||||||
|
* to extend in the future if (for example) the test classes
|
||||||
|
* begin to use one another.
|
||||||
|
*/
|
||||||
|
$namespace_map = ['MaxMind\Db\\' => __DIR__ . '/src/MaxMind/Db/'];
|
||||||
|
|
||||||
|
foreach ($namespace_map as $prefix => $dir) {
|
||||||
|
// First swap out the namespace prefix with a directory...
|
||||||
|
$path = str_replace($prefix, $dir, $class);
|
||||||
|
|
||||||
|
// replace the namespace separator with a directory separator...
|
||||||
|
$path = str_replace('\\', '/', $path);
|
||||||
|
|
||||||
|
// and finally, add the PHP file extension to the result.
|
||||||
|
$path .= '.php';
|
||||||
|
|
||||||
|
// $path should now contain the path to a PHP file defining $class
|
||||||
|
if (file_exists($path)) {
|
||||||
|
include $path;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
spl_autoload_register('mmdb_autoload');
|
||||||
43
.hta_lib/maxmind-db-reader/composer.json
Normal file
43
.hta_lib/maxmind-db-reader/composer.json
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
{
|
||||||
|
"name": "maxmind-db/reader",
|
||||||
|
"description": "MaxMind DB Reader API",
|
||||||
|
"keywords": ["database", "geoip", "geoip2", "geolocation", "maxmind"],
|
||||||
|
"homepage": "https://github.com/maxmind/MaxMind-DB-Reader-php",
|
||||||
|
"type": "library",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"authors": [
|
||||||
|
{
|
||||||
|
"name": "Gregory J. Oschwald",
|
||||||
|
"email": "goschwald@maxmind.com",
|
||||||
|
"homepage": "https://www.maxmind.com/"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"require": {
|
||||||
|
"php": ">=7.2"
|
||||||
|
},
|
||||||
|
"suggest": {
|
||||||
|
"ext-bcmath": "bcmath or gmp is required for decoding larger integers with the pure PHP decoder",
|
||||||
|
"ext-gmp": "bcmath or gmp is required for decoding larger integers with the pure PHP decoder",
|
||||||
|
"ext-maxminddb": "A C-based database decoder that provides significantly faster lookups",
|
||||||
|
"maxmind-db/reader-ext": "C extension for significantly faster IP lookups (install via PIE: pie install maxmind-db/reader-ext)"
|
||||||
|
},
|
||||||
|
"conflict": {
|
||||||
|
"ext-maxminddb": "<1.11.1 || >=2.0.0"
|
||||||
|
},
|
||||||
|
"require-dev": {
|
||||||
|
"friendsofphp/php-cs-fixer": "3.*",
|
||||||
|
"phpunit/phpunit": ">=8.0.0,<10.0.0",
|
||||||
|
"squizlabs/php_codesniffer": "4.*",
|
||||||
|
"phpstan/phpstan": "*"
|
||||||
|
},
|
||||||
|
"autoload": {
|
||||||
|
"psr-4": {
|
||||||
|
"MaxMind\\Db\\": "src/MaxMind/Db"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"autoload-dev": {
|
||||||
|
"psr-4": {
|
||||||
|
"MaxMind\\Db\\Test\\Reader\\": "tests/MaxMind/Db/Test/Reader"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
40
.hta_lib/maxmind-db-reader/ext/config.m4
Normal file
40
.hta_lib/maxmind-db-reader/ext/config.m4
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
PHP_ARG_WITH(maxminddb,
|
||||||
|
[Whether to enable the MaxMind DB Reader extension],
|
||||||
|
[ --with-maxminddb Enable MaxMind DB Reader extension support])
|
||||||
|
|
||||||
|
PHP_ARG_ENABLE(maxminddb-debug, for MaxMind DB debug support,
|
||||||
|
[ --enable-maxminddb-debug Enable MaxMind DB debug support], no, no)
|
||||||
|
|
||||||
|
if test $PHP_MAXMINDDB != "no"; then
|
||||||
|
|
||||||
|
AC_PATH_PROG(PKG_CONFIG, pkg-config, no)
|
||||||
|
|
||||||
|
AC_MSG_CHECKING(for libmaxminddb)
|
||||||
|
if test -x "$PKG_CONFIG" && $PKG_CONFIG --exists libmaxminddb; then
|
||||||
|
dnl retrieve build options from pkg-config
|
||||||
|
if $PKG_CONFIG libmaxminddb --atleast-version 1.0.0; then
|
||||||
|
LIBMAXMINDDB_INC=`$PKG_CONFIG libmaxminddb --cflags`
|
||||||
|
LIBMAXMINDDB_LIB=`$PKG_CONFIG libmaxminddb --libs`
|
||||||
|
LIBMAXMINDDB_VER=`$PKG_CONFIG libmaxminddb --modversion`
|
||||||
|
AC_MSG_RESULT(found version $LIBMAXMINDDB_VER)
|
||||||
|
else
|
||||||
|
AC_MSG_ERROR(system libmaxminddb must be upgraded to version >= 1.0.0)
|
||||||
|
fi
|
||||||
|
PHP_EVAL_LIBLINE($LIBMAXMINDDB_LIB, MAXMINDDB_SHARED_LIBADD)
|
||||||
|
PHP_EVAL_INCLINE($LIBMAXMINDDB_INC)
|
||||||
|
else
|
||||||
|
AC_MSG_RESULT(pkg-config information missing)
|
||||||
|
AC_MSG_WARN(will use libmaxmxinddb from compiler default path)
|
||||||
|
|
||||||
|
PHP_CHECK_LIBRARY(maxminddb, MMDB_open)
|
||||||
|
PHP_ADD_LIBRARY(maxminddb, 1, MAXMINDDB_SHARED_LIBADD)
|
||||||
|
fi
|
||||||
|
|
||||||
|
if test $PHP_MAXMINDDB_DEBUG != "no"; then
|
||||||
|
CFLAGS="$CFLAGS -Wall -Wextra -Wno-unused-parameter -Wno-missing-field-initializers -Werror"
|
||||||
|
fi
|
||||||
|
|
||||||
|
PHP_SUBST(MAXMINDDB_SHARED_LIBADD)
|
||||||
|
|
||||||
|
PHP_NEW_EXTENSION(maxminddb, maxminddb.c, $ext_shared)
|
||||||
|
fi
|
||||||
10
.hta_lib/maxmind-db-reader/ext/config.w32
Normal file
10
.hta_lib/maxmind-db-reader/ext/config.w32
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
ARG_WITH("maxminddb", "Enable MaxMind DB Reader extension support", "no");
|
||||||
|
|
||||||
|
if (PHP_MAXMINDDB == "yes") {
|
||||||
|
if (CHECK_HEADER_ADD_INCLUDE("maxminddb.h", "CFLAGS_MAXMINDDB", PHP_MAXMINDDB + ";" + PHP_PHP_BUILD + "\\include\\maxminddb") &&
|
||||||
|
CHECK_LIB("libmaxminddb.lib", "maxminddb", PHP_MAXMINDDB)) {
|
||||||
|
EXTENSION("maxminddb", "maxminddb.c");
|
||||||
|
} else {
|
||||||
|
WARNING('Could not find maxminddb.h or libmaxminddb.lib; skipping');
|
||||||
|
}
|
||||||
|
}
|
||||||
819
.hta_lib/maxmind-db-reader/ext/maxminddb.c
Normal file
819
.hta_lib/maxmind-db-reader/ext/maxminddb.c
Normal file
@@ -0,0 +1,819 @@
|
|||||||
|
/* MaxMind, Inc., licenses this file to you under the Apache License, Version
|
||||||
|
* 2.0 (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "php_maxminddb.h"
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include "config.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <php.h>
|
||||||
|
#include <zend.h>
|
||||||
|
|
||||||
|
#include "Zend/zend_exceptions.h"
|
||||||
|
#include "Zend/zend_types.h"
|
||||||
|
#include "ext/spl/spl_exceptions.h"
|
||||||
|
#include "ext/standard/info.h"
|
||||||
|
#include <maxminddb.h>
|
||||||
|
|
||||||
|
#ifdef ZTS
|
||||||
|
#include <TSRM.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define __STDC_FORMAT_MACROS
|
||||||
|
#include <inttypes.h>
|
||||||
|
|
||||||
|
#define PHP_MAXMINDDB_NS ZEND_NS_NAME("MaxMind", "Db")
|
||||||
|
#define PHP_MAXMINDDB_READER_NS ZEND_NS_NAME(PHP_MAXMINDDB_NS, "Reader")
|
||||||
|
#define PHP_MAXMINDDB_METADATA_NS \
|
||||||
|
ZEND_NS_NAME(PHP_MAXMINDDB_READER_NS, "Metadata")
|
||||||
|
#define PHP_MAXMINDDB_READER_EX_NS \
|
||||||
|
ZEND_NS_NAME(PHP_MAXMINDDB_READER_NS, "InvalidDatabaseException")
|
||||||
|
|
||||||
|
#define Z_MAXMINDDB_P(zv) php_maxminddb_fetch_object(Z_OBJ_P(zv))
|
||||||
|
typedef size_t strsize_t;
|
||||||
|
typedef zend_object free_obj_t;
|
||||||
|
|
||||||
|
/* For PHP 8 compatibility */
|
||||||
|
#if PHP_VERSION_ID < 80000
|
||||||
|
|
||||||
|
#define PROP_OBJ(zv) (zv)
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
#define PROP_OBJ(zv) Z_OBJ_P(zv)
|
||||||
|
|
||||||
|
#define TSRMLS_C
|
||||||
|
#define TSRMLS_CC
|
||||||
|
#define TSRMLS_DC
|
||||||
|
|
||||||
|
/* End PHP 8 compatibility */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef ZEND_ACC_CTOR
|
||||||
|
#define ZEND_ACC_CTOR 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* IS_MIXED was added in 2020 */
|
||||||
|
#ifndef IS_MIXED
|
||||||
|
#define IS_MIXED IS_UNDEF
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* ZEND_THIS was added in 7.4 */
|
||||||
|
#ifndef ZEND_THIS
|
||||||
|
#define ZEND_THIS (&EX(This))
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef struct _maxminddb_obj {
|
||||||
|
MMDB_s *mmdb;
|
||||||
|
zend_object std;
|
||||||
|
} maxminddb_obj;
|
||||||
|
|
||||||
|
PHP_FUNCTION(maxminddb);
|
||||||
|
|
||||||
|
static int
|
||||||
|
get_record(INTERNAL_FUNCTION_PARAMETERS, zval *record, int *prefix_len);
|
||||||
|
static const MMDB_entry_data_list_s *
|
||||||
|
handle_entry_data_list(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC);
|
||||||
|
static const MMDB_entry_data_list_s *
|
||||||
|
handle_array(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC);
|
||||||
|
static const MMDB_entry_data_list_s *
|
||||||
|
handle_map(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC);
|
||||||
|
static void handle_uint128(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC);
|
||||||
|
static void handle_uint64(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC);
|
||||||
|
static void handle_uint32(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC);
|
||||||
|
|
||||||
|
#define CHECK_ALLOCATED(val) \
|
||||||
|
if (!val) { \
|
||||||
|
zend_error(E_ERROR, "Out of memory"); \
|
||||||
|
return; \
|
||||||
|
}
|
||||||
|
|
||||||
|
static zend_object_handlers maxminddb_obj_handlers;
|
||||||
|
static zend_class_entry *maxminddb_ce, *maxminddb_exception_ce, *metadata_ce;
|
||||||
|
|
||||||
|
static inline maxminddb_obj *
|
||||||
|
php_maxminddb_fetch_object(zend_object *obj TSRMLS_DC) {
|
||||||
|
return (maxminddb_obj *)((char *)(obj)-XtOffsetOf(maxminddb_obj, std));
|
||||||
|
}
|
||||||
|
|
||||||
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_maxminddbreader_construct, 0, 0, 1)
|
||||||
|
ZEND_ARG_TYPE_INFO(0, db_file, IS_STRING, 0)
|
||||||
|
ZEND_END_ARG_INFO()
|
||||||
|
|
||||||
|
PHP_METHOD(MaxMind_Db_Reader, __construct) {
|
||||||
|
char *db_file = NULL;
|
||||||
|
strsize_t name_len;
|
||||||
|
zval *_this_zval = NULL;
|
||||||
|
|
||||||
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC,
|
||||||
|
getThis(),
|
||||||
|
"Os",
|
||||||
|
&_this_zval,
|
||||||
|
maxminddb_ce,
|
||||||
|
&db_file,
|
||||||
|
&name_len) == FAILURE) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (0 != php_check_open_basedir(db_file TSRMLS_CC) ||
|
||||||
|
0 != access(db_file, R_OK)) {
|
||||||
|
zend_throw_exception_ex(
|
||||||
|
spl_ce_InvalidArgumentException,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"The file \"%s\" does not exist or is not readable.",
|
||||||
|
db_file);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
MMDB_s *mmdb = (MMDB_s *)ecalloc(1, sizeof(MMDB_s));
|
||||||
|
int const status = MMDB_open(db_file, MMDB_MODE_MMAP, mmdb);
|
||||||
|
|
||||||
|
if (MMDB_SUCCESS != status) {
|
||||||
|
zend_throw_exception_ex(
|
||||||
|
maxminddb_exception_ce,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"Error opening database file (%s). Is this a valid "
|
||||||
|
"MaxMind DB file?",
|
||||||
|
db_file);
|
||||||
|
efree(mmdb);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
maxminddb_obj *mmdb_obj = Z_MAXMINDDB_P(ZEND_THIS);
|
||||||
|
mmdb_obj->mmdb = mmdb;
|
||||||
|
}
|
||||||
|
|
||||||
|
ZEND_BEGIN_ARG_WITH_RETURN_TYPE_INFO_EX(
|
||||||
|
arginfo_maxminddbreader_get, 0, 1, IS_MIXED, 1)
|
||||||
|
ZEND_ARG_TYPE_INFO(0, ip_address, IS_STRING, 0)
|
||||||
|
ZEND_END_ARG_INFO()
|
||||||
|
|
||||||
|
PHP_METHOD(MaxMind_Db_Reader, get) {
|
||||||
|
int prefix_len = 0;
|
||||||
|
get_record(INTERNAL_FUNCTION_PARAM_PASSTHRU, return_value, &prefix_len);
|
||||||
|
}
|
||||||
|
|
||||||
|
ZEND_BEGIN_ARG_WITH_RETURN_TYPE_INFO_EX(
|
||||||
|
arginfo_maxminddbreader_getWithPrefixLen, 0, 1, IS_ARRAY, 1)
|
||||||
|
ZEND_ARG_TYPE_INFO(0, ip_address, IS_STRING, 0)
|
||||||
|
ZEND_END_ARG_INFO()
|
||||||
|
|
||||||
|
PHP_METHOD(MaxMind_Db_Reader, getWithPrefixLen) {
|
||||||
|
zval record, z_prefix_len;
|
||||||
|
|
||||||
|
int prefix_len = 0;
|
||||||
|
if (get_record(INTERNAL_FUNCTION_PARAM_PASSTHRU, &record, &prefix_len) ==
|
||||||
|
FAILURE) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
array_init(return_value);
|
||||||
|
add_next_index_zval(return_value, &record);
|
||||||
|
|
||||||
|
ZVAL_LONG(&z_prefix_len, prefix_len);
|
||||||
|
add_next_index_zval(return_value, &z_prefix_len);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
get_record(INTERNAL_FUNCTION_PARAMETERS, zval *record, int *prefix_len) {
|
||||||
|
char *ip_address = NULL;
|
||||||
|
strsize_t name_len;
|
||||||
|
zval *this_zval = NULL;
|
||||||
|
|
||||||
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC,
|
||||||
|
getThis(),
|
||||||
|
"Os",
|
||||||
|
&this_zval,
|
||||||
|
maxminddb_ce,
|
||||||
|
&ip_address,
|
||||||
|
&name_len) == FAILURE) {
|
||||||
|
return FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const maxminddb_obj *mmdb_obj = (maxminddb_obj *)Z_MAXMINDDB_P(ZEND_THIS);
|
||||||
|
|
||||||
|
MMDB_s *mmdb = mmdb_obj->mmdb;
|
||||||
|
|
||||||
|
if (NULL == mmdb) {
|
||||||
|
zend_throw_exception_ex(spl_ce_BadMethodCallException,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"Attempt to read from a closed MaxMind DB.");
|
||||||
|
return FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct addrinfo hints = {
|
||||||
|
.ai_family = AF_UNSPEC,
|
||||||
|
.ai_flags = AI_NUMERICHOST,
|
||||||
|
/* We set ai_socktype so that we only get one result back */
|
||||||
|
.ai_socktype = SOCK_STREAM};
|
||||||
|
|
||||||
|
struct addrinfo *addresses = NULL;
|
||||||
|
int gai_status = getaddrinfo(ip_address, NULL, &hints, &addresses);
|
||||||
|
if (gai_status) {
|
||||||
|
zend_throw_exception_ex(spl_ce_InvalidArgumentException,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"The value \"%s\" is not a valid IP address.",
|
||||||
|
ip_address);
|
||||||
|
return FAILURE;
|
||||||
|
}
|
||||||
|
if (!addresses || !addresses->ai_addr) {
|
||||||
|
zend_throw_exception_ex(
|
||||||
|
spl_ce_InvalidArgumentException,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"getaddrinfo was successful but failed to set the addrinfo");
|
||||||
|
return FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
int sa_family = addresses->ai_addr->sa_family;
|
||||||
|
|
||||||
|
int mmdb_error = MMDB_SUCCESS;
|
||||||
|
MMDB_lookup_result_s result =
|
||||||
|
MMDB_lookup_sockaddr(mmdb, addresses->ai_addr, &mmdb_error);
|
||||||
|
|
||||||
|
freeaddrinfo(addresses);
|
||||||
|
|
||||||
|
if (MMDB_SUCCESS != mmdb_error) {
|
||||||
|
zend_class_entry *ex;
|
||||||
|
if (MMDB_IPV6_LOOKUP_IN_IPV4_DATABASE_ERROR == mmdb_error) {
|
||||||
|
ex = spl_ce_InvalidArgumentException;
|
||||||
|
} else {
|
||||||
|
ex = maxminddb_exception_ce;
|
||||||
|
}
|
||||||
|
zend_throw_exception_ex(ex,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"Error looking up %s. %s",
|
||||||
|
ip_address,
|
||||||
|
MMDB_strerror(mmdb_error));
|
||||||
|
return FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
*prefix_len = result.netmask;
|
||||||
|
|
||||||
|
if (sa_family == AF_INET && mmdb->metadata.ip_version == 6) {
|
||||||
|
/* We return the prefix length given the IPv4 address. If there is
|
||||||
|
no IPv4 subtree, we return a prefix length of 0. */
|
||||||
|
*prefix_len = *prefix_len >= 96 ? *prefix_len - 96 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!result.found_entry) {
|
||||||
|
ZVAL_NULL(record);
|
||||||
|
return SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
MMDB_entry_data_list_s *entry_data_list = NULL;
|
||||||
|
int status = MMDB_get_entry_data_list(&result.entry, &entry_data_list);
|
||||||
|
|
||||||
|
if (MMDB_SUCCESS != status) {
|
||||||
|
zend_throw_exception_ex(maxminddb_exception_ce,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"Error while looking up data for %s. %s",
|
||||||
|
ip_address,
|
||||||
|
MMDB_strerror(status));
|
||||||
|
MMDB_free_entry_data_list(entry_data_list);
|
||||||
|
return FAILURE;
|
||||||
|
} else if (NULL == entry_data_list) {
|
||||||
|
zend_throw_exception_ex(
|
||||||
|
maxminddb_exception_ce,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"Error while looking up data for %s. Your database may "
|
||||||
|
"be corrupt or you have found a bug in libmaxminddb.",
|
||||||
|
ip_address);
|
||||||
|
return FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const MMDB_entry_data_list_s *rv =
|
||||||
|
handle_entry_data_list(entry_data_list, record TSRMLS_CC);
|
||||||
|
if (rv == NULL) {
|
||||||
|
/* We should have already thrown the exception in handle_entry_data_list
|
||||||
|
*/
|
||||||
|
return FAILURE;
|
||||||
|
}
|
||||||
|
MMDB_free_entry_data_list(entry_data_list);
|
||||||
|
return SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_maxminddbreader_void, 0, 0, 0)
|
||||||
|
ZEND_END_ARG_INFO()
|
||||||
|
|
||||||
|
PHP_METHOD(MaxMind_Db_Reader, metadata) {
|
||||||
|
zval *this_zval = NULL;
|
||||||
|
|
||||||
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC,
|
||||||
|
getThis(),
|
||||||
|
"O",
|
||||||
|
&this_zval,
|
||||||
|
maxminddb_ce) == FAILURE) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const maxminddb_obj *const mmdb_obj =
|
||||||
|
(maxminddb_obj *)Z_MAXMINDDB_P(this_zval);
|
||||||
|
|
||||||
|
if (NULL == mmdb_obj->mmdb) {
|
||||||
|
zend_throw_exception_ex(spl_ce_BadMethodCallException,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"Attempt to read from a closed MaxMind DB.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
object_init_ex(return_value, metadata_ce);
|
||||||
|
|
||||||
|
MMDB_entry_data_list_s *entry_data_list;
|
||||||
|
int status =
|
||||||
|
MMDB_get_metadata_as_entry_data_list(mmdb_obj->mmdb, &entry_data_list);
|
||||||
|
if (status != MMDB_SUCCESS) {
|
||||||
|
zend_throw_exception_ex(maxminddb_exception_ce,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"Error while decoding metadata. %s",
|
||||||
|
MMDB_strerror(status));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
zval metadata_array;
|
||||||
|
const MMDB_entry_data_list_s *rv =
|
||||||
|
handle_entry_data_list(entry_data_list, &metadata_array TSRMLS_CC);
|
||||||
|
if (rv == NULL) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
MMDB_free_entry_data_list(entry_data_list);
|
||||||
|
zend_call_method_with_1_params(PROP_OBJ(return_value),
|
||||||
|
metadata_ce,
|
||||||
|
&metadata_ce->constructor,
|
||||||
|
ZEND_CONSTRUCTOR_FUNC_NAME,
|
||||||
|
NULL,
|
||||||
|
&metadata_array);
|
||||||
|
zval_ptr_dtor(&metadata_array);
|
||||||
|
}
|
||||||
|
|
||||||
|
PHP_METHOD(MaxMind_Db_Reader, close) {
|
||||||
|
zval *this_zval = NULL;
|
||||||
|
|
||||||
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC,
|
||||||
|
getThis(),
|
||||||
|
"O",
|
||||||
|
&this_zval,
|
||||||
|
maxminddb_ce) == FAILURE) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
maxminddb_obj *mmdb_obj = (maxminddb_obj *)Z_MAXMINDDB_P(this_zval);
|
||||||
|
|
||||||
|
if (NULL == mmdb_obj->mmdb) {
|
||||||
|
zend_throw_exception_ex(spl_ce_BadMethodCallException,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"Attempt to close a closed MaxMind DB.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
MMDB_close(mmdb_obj->mmdb);
|
||||||
|
efree(mmdb_obj->mmdb);
|
||||||
|
mmdb_obj->mmdb = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const MMDB_entry_data_list_s *
|
||||||
|
handle_entry_data_list(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC) {
|
||||||
|
switch (entry_data_list->entry_data.type) {
|
||||||
|
case MMDB_DATA_TYPE_MAP:
|
||||||
|
return handle_map(entry_data_list, z_value TSRMLS_CC);
|
||||||
|
case MMDB_DATA_TYPE_ARRAY:
|
||||||
|
return handle_array(entry_data_list, z_value TSRMLS_CC);
|
||||||
|
case MMDB_DATA_TYPE_UTF8_STRING:
|
||||||
|
ZVAL_STRINGL(z_value,
|
||||||
|
entry_data_list->entry_data.utf8_string,
|
||||||
|
entry_data_list->entry_data.data_size);
|
||||||
|
break;
|
||||||
|
case MMDB_DATA_TYPE_BYTES:
|
||||||
|
ZVAL_STRINGL(z_value,
|
||||||
|
(char const *)entry_data_list->entry_data.bytes,
|
||||||
|
entry_data_list->entry_data.data_size);
|
||||||
|
break;
|
||||||
|
case MMDB_DATA_TYPE_DOUBLE:
|
||||||
|
ZVAL_DOUBLE(z_value, entry_data_list->entry_data.double_value);
|
||||||
|
break;
|
||||||
|
case MMDB_DATA_TYPE_FLOAT:
|
||||||
|
ZVAL_DOUBLE(z_value, entry_data_list->entry_data.float_value);
|
||||||
|
break;
|
||||||
|
case MMDB_DATA_TYPE_UINT16:
|
||||||
|
ZVAL_LONG(z_value, entry_data_list->entry_data.uint16);
|
||||||
|
break;
|
||||||
|
case MMDB_DATA_TYPE_UINT32:
|
||||||
|
handle_uint32(entry_data_list, z_value TSRMLS_CC);
|
||||||
|
break;
|
||||||
|
case MMDB_DATA_TYPE_BOOLEAN:
|
||||||
|
ZVAL_BOOL(z_value, entry_data_list->entry_data.boolean);
|
||||||
|
break;
|
||||||
|
case MMDB_DATA_TYPE_UINT64:
|
||||||
|
handle_uint64(entry_data_list, z_value TSRMLS_CC);
|
||||||
|
break;
|
||||||
|
case MMDB_DATA_TYPE_UINT128:
|
||||||
|
handle_uint128(entry_data_list, z_value TSRMLS_CC);
|
||||||
|
break;
|
||||||
|
case MMDB_DATA_TYPE_INT32:
|
||||||
|
ZVAL_LONG(z_value, entry_data_list->entry_data.int32);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
zend_throw_exception_ex(maxminddb_exception_ce,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"Invalid data type arguments: %d",
|
||||||
|
entry_data_list->entry_data.type);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
return entry_data_list;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const MMDB_entry_data_list_s *
|
||||||
|
handle_map(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC) {
|
||||||
|
array_init(z_value);
|
||||||
|
const uint32_t map_size = entry_data_list->entry_data.data_size;
|
||||||
|
|
||||||
|
uint32_t i;
|
||||||
|
for (i = 0; i < map_size && entry_data_list; i++) {
|
||||||
|
entry_data_list = entry_data_list->next;
|
||||||
|
|
||||||
|
char *key = estrndup(entry_data_list->entry_data.utf8_string,
|
||||||
|
entry_data_list->entry_data.data_size);
|
||||||
|
if (NULL == key) {
|
||||||
|
zend_throw_exception_ex(maxminddb_exception_ce,
|
||||||
|
0 TSRMLS_CC,
|
||||||
|
"Invalid data type arguments");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
entry_data_list = entry_data_list->next;
|
||||||
|
zval new_value;
|
||||||
|
entry_data_list =
|
||||||
|
handle_entry_data_list(entry_data_list, &new_value TSRMLS_CC);
|
||||||
|
if (entry_data_list != NULL) {
|
||||||
|
add_assoc_zval(z_value, key, &new_value);
|
||||||
|
}
|
||||||
|
efree(key);
|
||||||
|
}
|
||||||
|
return entry_data_list;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const MMDB_entry_data_list_s *
|
||||||
|
handle_array(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC) {
|
||||||
|
const uint32_t size = entry_data_list->entry_data.data_size;
|
||||||
|
|
||||||
|
array_init(z_value);
|
||||||
|
|
||||||
|
uint32_t i;
|
||||||
|
for (i = 0; i < size && entry_data_list; i++) {
|
||||||
|
entry_data_list = entry_data_list->next;
|
||||||
|
zval new_value;
|
||||||
|
entry_data_list =
|
||||||
|
handle_entry_data_list(entry_data_list, &new_value TSRMLS_CC);
|
||||||
|
if (entry_data_list != NULL) {
|
||||||
|
add_next_index_zval(z_value, &new_value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return entry_data_list;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_uint128(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC) {
|
||||||
|
uint64_t high = 0;
|
||||||
|
uint64_t low = 0;
|
||||||
|
#if MMDB_UINT128_IS_BYTE_ARRAY
|
||||||
|
int i;
|
||||||
|
for (i = 0; i < 8; i++) {
|
||||||
|
high = (high << 8) | entry_data_list->entry_data.uint128[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = 8; i < 16; i++) {
|
||||||
|
low = (low << 8) | entry_data_list->entry_data.uint128[i];
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
high = entry_data_list->entry_data.uint128 >> 64;
|
||||||
|
low = (uint64_t)entry_data_list->entry_data.uint128;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
char *num_str;
|
||||||
|
spprintf(&num_str, 0, "0x%016" PRIX64 "%016" PRIX64, high, low);
|
||||||
|
CHECK_ALLOCATED(num_str);
|
||||||
|
|
||||||
|
ZVAL_STRING(z_value, num_str);
|
||||||
|
efree(num_str);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_uint32(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC) {
|
||||||
|
uint32_t val = entry_data_list->entry_data.uint32;
|
||||||
|
|
||||||
|
#if LONG_MAX >= UINT32_MAX
|
||||||
|
ZVAL_LONG(z_value, val);
|
||||||
|
return;
|
||||||
|
#else
|
||||||
|
if (val <= LONG_MAX) {
|
||||||
|
ZVAL_LONG(z_value, val);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *int_str;
|
||||||
|
spprintf(&int_str, 0, "%" PRIu32, val);
|
||||||
|
CHECK_ALLOCATED(int_str);
|
||||||
|
|
||||||
|
ZVAL_STRING(z_value, int_str);
|
||||||
|
efree(int_str);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_uint64(const MMDB_entry_data_list_s *entry_data_list,
|
||||||
|
zval *z_value TSRMLS_DC) {
|
||||||
|
uint64_t val = entry_data_list->entry_data.uint64;
|
||||||
|
|
||||||
|
#if LONG_MAX >= UINT64_MAX
|
||||||
|
ZVAL_LONG(z_value, val);
|
||||||
|
return;
|
||||||
|
#else
|
||||||
|
if (val <= LONG_MAX) {
|
||||||
|
ZVAL_LONG(z_value, val);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *int_str;
|
||||||
|
spprintf(&int_str, 0, "%" PRIu64, val);
|
||||||
|
CHECK_ALLOCATED(int_str);
|
||||||
|
|
||||||
|
ZVAL_STRING(z_value, int_str);
|
||||||
|
efree(int_str);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
static void maxminddb_free_storage(free_obj_t *object TSRMLS_DC) {
|
||||||
|
maxminddb_obj *obj =
|
||||||
|
php_maxminddb_fetch_object((zend_object *)object TSRMLS_CC);
|
||||||
|
if (obj->mmdb != NULL) {
|
||||||
|
MMDB_close(obj->mmdb);
|
||||||
|
efree(obj->mmdb);
|
||||||
|
}
|
||||||
|
|
||||||
|
zend_object_std_dtor(&obj->std TSRMLS_CC);
|
||||||
|
}
|
||||||
|
|
||||||
|
static zend_object *maxminddb_create_handler(zend_class_entry *type TSRMLS_DC) {
|
||||||
|
maxminddb_obj *obj = (maxminddb_obj *)ecalloc(1, sizeof(maxminddb_obj));
|
||||||
|
zend_object_std_init(&obj->std, type TSRMLS_CC);
|
||||||
|
object_properties_init(&(obj->std), type);
|
||||||
|
|
||||||
|
obj->std.handlers = &maxminddb_obj_handlers;
|
||||||
|
|
||||||
|
return &obj->std;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* clang-format off */
|
||||||
|
static zend_function_entry maxminddb_methods[] = {
|
||||||
|
PHP_ME(MaxMind_Db_Reader, __construct, arginfo_maxminddbreader_construct,
|
||||||
|
ZEND_ACC_PUBLIC | ZEND_ACC_CTOR)
|
||||||
|
PHP_ME(MaxMind_Db_Reader, close, arginfo_maxminddbreader_void, ZEND_ACC_PUBLIC)
|
||||||
|
PHP_ME(MaxMind_Db_Reader, get, arginfo_maxminddbreader_get, ZEND_ACC_PUBLIC)
|
||||||
|
PHP_ME(MaxMind_Db_Reader, getWithPrefixLen, arginfo_maxminddbreader_getWithPrefixLen, ZEND_ACC_PUBLIC)
|
||||||
|
PHP_ME(MaxMind_Db_Reader, metadata, arginfo_maxminddbreader_void, ZEND_ACC_PUBLIC)
|
||||||
|
{ NULL, NULL, NULL }
|
||||||
|
};
|
||||||
|
/* clang-format on */
|
||||||
|
|
||||||
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_metadata_construct, 0, 0, 1)
|
||||||
|
ZEND_ARG_TYPE_INFO(0, metadata, IS_ARRAY, 0)
|
||||||
|
ZEND_END_ARG_INFO()
|
||||||
|
|
||||||
|
PHP_METHOD(MaxMind_Db_Reader_Metadata, __construct) {
|
||||||
|
zval *object = NULL;
|
||||||
|
zval *metadata_array = NULL;
|
||||||
|
zend_long node_count = 0;
|
||||||
|
zend_long record_size = 0;
|
||||||
|
|
||||||
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC,
|
||||||
|
getThis(),
|
||||||
|
"Oa",
|
||||||
|
&object,
|
||||||
|
metadata_ce,
|
||||||
|
&metadata_array) == FAILURE) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
zval *tmp = NULL;
|
||||||
|
if ((tmp = zend_hash_str_find(HASH_OF(metadata_array),
|
||||||
|
"binary_format_major_version",
|
||||||
|
sizeof("binary_format_major_version") - 1))) {
|
||||||
|
zend_update_property(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"binaryFormatMajorVersion",
|
||||||
|
sizeof("binaryFormatMajorVersion") - 1,
|
||||||
|
tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((tmp = zend_hash_str_find(HASH_OF(metadata_array),
|
||||||
|
"binary_format_minor_version",
|
||||||
|
sizeof("binary_format_minor_version") - 1))) {
|
||||||
|
zend_update_property(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"binaryFormatMinorVersion",
|
||||||
|
sizeof("binaryFormatMinorVersion") - 1,
|
||||||
|
tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((tmp = zend_hash_str_find(HASH_OF(metadata_array),
|
||||||
|
"build_epoch",
|
||||||
|
sizeof("build_epoch") - 1))) {
|
||||||
|
zend_update_property(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"buildEpoch",
|
||||||
|
sizeof("buildEpoch") - 1,
|
||||||
|
tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((tmp = zend_hash_str_find(HASH_OF(metadata_array),
|
||||||
|
"database_type",
|
||||||
|
sizeof("database_type") - 1))) {
|
||||||
|
zend_update_property(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"databaseType",
|
||||||
|
sizeof("databaseType") - 1,
|
||||||
|
tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((tmp = zend_hash_str_find(HASH_OF(metadata_array),
|
||||||
|
"description",
|
||||||
|
sizeof("description") - 1))) {
|
||||||
|
zend_update_property(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"description",
|
||||||
|
sizeof("description") - 1,
|
||||||
|
tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((tmp = zend_hash_str_find(HASH_OF(metadata_array),
|
||||||
|
"ip_version",
|
||||||
|
sizeof("ip_version") - 1))) {
|
||||||
|
zend_update_property(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"ipVersion",
|
||||||
|
sizeof("ipVersion") - 1,
|
||||||
|
tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((tmp = zend_hash_str_find(
|
||||||
|
HASH_OF(metadata_array), "languages", sizeof("languages") - 1))) {
|
||||||
|
zend_update_property(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"languages",
|
||||||
|
sizeof("languages") - 1,
|
||||||
|
tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((tmp = zend_hash_str_find(HASH_OF(metadata_array),
|
||||||
|
"record_size",
|
||||||
|
sizeof("record_size") - 1))) {
|
||||||
|
zend_update_property(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"recordSize",
|
||||||
|
sizeof("recordSize") - 1,
|
||||||
|
tmp);
|
||||||
|
if (Z_TYPE_P(tmp) == IS_LONG) {
|
||||||
|
record_size = Z_LVAL_P(tmp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (record_size != 0) {
|
||||||
|
zend_update_property_long(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"nodeByteSize",
|
||||||
|
sizeof("nodeByteSize") - 1,
|
||||||
|
record_size / 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((tmp = zend_hash_str_find(HASH_OF(metadata_array),
|
||||||
|
"node_count",
|
||||||
|
sizeof("node_count") - 1))) {
|
||||||
|
zend_update_property(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"nodeCount",
|
||||||
|
sizeof("nodeCount") - 1,
|
||||||
|
tmp);
|
||||||
|
if (Z_TYPE_P(tmp) == IS_LONG) {
|
||||||
|
node_count = Z_LVAL_P(tmp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (record_size != 0) {
|
||||||
|
zend_update_property_long(metadata_ce,
|
||||||
|
PROP_OBJ(object),
|
||||||
|
"searchTreeSize",
|
||||||
|
sizeof("searchTreeSize") - 1,
|
||||||
|
record_size * node_count / 4);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// clang-format off
|
||||||
|
static zend_function_entry metadata_methods[] = {
|
||||||
|
PHP_ME(MaxMind_Db_Reader_Metadata, __construct, arginfo_metadata_construct, ZEND_ACC_PUBLIC | ZEND_ACC_CTOR)
|
||||||
|
{NULL, NULL, NULL}
|
||||||
|
};
|
||||||
|
// clang-format on
|
||||||
|
|
||||||
|
PHP_MINIT_FUNCTION(maxminddb) {
|
||||||
|
zend_class_entry ce;
|
||||||
|
|
||||||
|
INIT_CLASS_ENTRY(ce, PHP_MAXMINDDB_READER_EX_NS, NULL);
|
||||||
|
maxminddb_exception_ce =
|
||||||
|
zend_register_internal_class_ex(&ce, zend_ce_exception);
|
||||||
|
|
||||||
|
INIT_CLASS_ENTRY(ce, PHP_MAXMINDDB_READER_NS, maxminddb_methods);
|
||||||
|
maxminddb_ce = zend_register_internal_class(&ce TSRMLS_CC);
|
||||||
|
maxminddb_ce->create_object = maxminddb_create_handler;
|
||||||
|
|
||||||
|
INIT_CLASS_ENTRY(ce, PHP_MAXMINDDB_METADATA_NS, metadata_methods);
|
||||||
|
metadata_ce = zend_register_internal_class(&ce TSRMLS_CC);
|
||||||
|
zend_declare_property_null(metadata_ce,
|
||||||
|
"binaryFormatMajorVersion",
|
||||||
|
sizeof("binaryFormatMajorVersion") - 1,
|
||||||
|
ZEND_ACC_PUBLIC);
|
||||||
|
zend_declare_property_null(metadata_ce,
|
||||||
|
"binaryFormatMinorVersion",
|
||||||
|
sizeof("binaryFormatMinorVersion") - 1,
|
||||||
|
ZEND_ACC_PUBLIC);
|
||||||
|
zend_declare_property_null(
|
||||||
|
metadata_ce, "buildEpoch", sizeof("buildEpoch") - 1, ZEND_ACC_PUBLIC);
|
||||||
|
zend_declare_property_null(metadata_ce,
|
||||||
|
"databaseType",
|
||||||
|
sizeof("databaseType") - 1,
|
||||||
|
ZEND_ACC_PUBLIC);
|
||||||
|
zend_declare_property_null(
|
||||||
|
metadata_ce, "description", sizeof("description") - 1, ZEND_ACC_PUBLIC);
|
||||||
|
zend_declare_property_null(
|
||||||
|
metadata_ce, "ipVersion", sizeof("ipVersion") - 1, ZEND_ACC_PUBLIC);
|
||||||
|
zend_declare_property_null(
|
||||||
|
metadata_ce, "languages", sizeof("languages") - 1, ZEND_ACC_PUBLIC);
|
||||||
|
zend_declare_property_null(metadata_ce,
|
||||||
|
"nodeByteSize",
|
||||||
|
sizeof("nodeByteSize") - 1,
|
||||||
|
ZEND_ACC_PUBLIC);
|
||||||
|
zend_declare_property_null(
|
||||||
|
metadata_ce, "nodeCount", sizeof("nodeCount") - 1, ZEND_ACC_PUBLIC);
|
||||||
|
zend_declare_property_null(
|
||||||
|
metadata_ce, "recordSize", sizeof("recordSize") - 1, ZEND_ACC_PUBLIC);
|
||||||
|
zend_declare_property_null(metadata_ce,
|
||||||
|
"searchTreeSize",
|
||||||
|
sizeof("searchTreeSize") - 1,
|
||||||
|
ZEND_ACC_PUBLIC);
|
||||||
|
|
||||||
|
memcpy(&maxminddb_obj_handlers,
|
||||||
|
zend_get_std_object_handlers(),
|
||||||
|
sizeof(zend_object_handlers));
|
||||||
|
maxminddb_obj_handlers.clone_obj = NULL;
|
||||||
|
maxminddb_obj_handlers.offset = XtOffsetOf(maxminddb_obj, std);
|
||||||
|
maxminddb_obj_handlers.free_obj = maxminddb_free_storage;
|
||||||
|
zend_declare_class_constant_string(maxminddb_ce,
|
||||||
|
"MMDB_LIB_VERSION",
|
||||||
|
sizeof("MMDB_LIB_VERSION") - 1,
|
||||||
|
MMDB_lib_version() TSRMLS_CC);
|
||||||
|
|
||||||
|
return SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
static PHP_MINFO_FUNCTION(maxminddb) {
|
||||||
|
php_info_print_table_start();
|
||||||
|
|
||||||
|
php_info_print_table_row(2, "MaxMind DB Reader", "enabled");
|
||||||
|
php_info_print_table_row(
|
||||||
|
2, "maxminddb extension version", PHP_MAXMINDDB_VERSION);
|
||||||
|
php_info_print_table_row(
|
||||||
|
2, "libmaxminddb library version", MMDB_lib_version());
|
||||||
|
|
||||||
|
php_info_print_table_end();
|
||||||
|
}
|
||||||
|
|
||||||
|
zend_module_entry maxminddb_module_entry = {STANDARD_MODULE_HEADER,
|
||||||
|
PHP_MAXMINDDB_EXTNAME,
|
||||||
|
NULL,
|
||||||
|
PHP_MINIT(maxminddb),
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
PHP_MINFO(maxminddb),
|
||||||
|
PHP_MAXMINDDB_VERSION,
|
||||||
|
STANDARD_MODULE_PROPERTIES};
|
||||||
|
|
||||||
|
#ifdef COMPILE_DL_MAXMINDDB
|
||||||
|
ZEND_GET_MODULE(maxminddb)
|
||||||
|
#endif
|
||||||
24
.hta_lib/maxmind-db-reader/ext/php_maxminddb.h
Normal file
24
.hta_lib/maxmind-db-reader/ext/php_maxminddb.h
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
/* MaxMind, Inc., licenses this file to you under the Apache License, Version
|
||||||
|
* 2.0 (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <zend_interfaces.h>
|
||||||
|
|
||||||
|
#ifndef PHP_MAXMINDDB_H
|
||||||
|
#define PHP_MAXMINDDB_H 1
|
||||||
|
#define PHP_MAXMINDDB_VERSION "1.13.1"
|
||||||
|
#define PHP_MAXMINDDB_EXTNAME "maxminddb"
|
||||||
|
|
||||||
|
extern zend_module_entry maxminddb_module_entry;
|
||||||
|
#define phpext_maxminddb_ptr &maxminddb_module_entry
|
||||||
|
|
||||||
|
#endif
|
||||||
12
.hta_lib/maxmind-db-reader/ext/tests/001-load.phpt
Normal file
12
.hta_lib/maxmind-db-reader/ext/tests/001-load.phpt
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
--TEST--
|
||||||
|
Check for maxminddb presence
|
||||||
|
--SKIPIF--
|
||||||
|
<?php if (!extension_loaded('maxminddb')) {
|
||||||
|
echo 'skip';
|
||||||
|
} ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
echo 'maxminddb extension is available';
|
||||||
|
?>
|
||||||
|
--EXPECT--
|
||||||
|
maxminddb extension is available
|
||||||
13
.hta_lib/maxmind-db-reader/ext/tests/002-final.phpt
Normal file
13
.hta_lib/maxmind-db-reader/ext/tests/002-final.phpt
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
--TEST--
|
||||||
|
Check that Reader class is not final
|
||||||
|
--SKIPIF--
|
||||||
|
<?php if (!extension_loaded('maxminddb')) {
|
||||||
|
echo 'skip';
|
||||||
|
} ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
$reflectionClass = new \ReflectionClass('MaxMind\Db\Reader');
|
||||||
|
var_dump($reflectionClass->isFinal());
|
||||||
|
?>
|
||||||
|
--EXPECT--
|
||||||
|
bool(false)
|
||||||
12
.hta_lib/maxmind-db-reader/ext/tests/003-open-basedir.phpt
Normal file
12
.hta_lib/maxmind-db-reader/ext/tests/003-open-basedir.phpt
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
--TEST--
|
||||||
|
openbase_dir is followed
|
||||||
|
--INI--
|
||||||
|
open_basedir=/--dne--
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
use MaxMind\Db\Reader;
|
||||||
|
|
||||||
|
$reader = new Reader('/usr/local/share/GeoIP/GeoIP2-City.mmdb');
|
||||||
|
?>
|
||||||
|
--EXPECTREGEX--
|
||||||
|
.*open_basedir restriction in effect.*
|
||||||
61
.hta_lib/maxmind-db-reader/package.xml
Normal file
61
.hta_lib/maxmind-db-reader/package.xml
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
<?xml version="1.0"?>
|
||||||
|
<package version="2.0" xmlns="http://pear.php.net/dtd/package-2.0"
|
||||||
|
xmlns:tasks="http://pear.php.net/dtd/tasks-1.0"
|
||||||
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xsi:schemaLocation="http://pear.php.net/dtd/tasks-1.0 http://pear.php.net/dtd/tasks-1.0.xsd http://pear.php.net/dtd/package-2.0 http://pear.php.net/dtd/package-2.0.xsd">
|
||||||
|
|
||||||
|
<name>maxminddb</name>
|
||||||
|
<channel>pecl.php.net</channel>
|
||||||
|
<summary>Reader for the MaxMind DB file format</summary>
|
||||||
|
<description>This is the PHP extension for reading MaxMind DB files. MaxMind DB is a binary file format that stores data indexed by IP address subnets (IPv4 or IPv6).</description>
|
||||||
|
<lead>
|
||||||
|
<name>Greg Oschwald</name>
|
||||||
|
<user>oschwald</user>
|
||||||
|
<email>goschwald@maxmind.com</email>
|
||||||
|
<active>yes</active>
|
||||||
|
</lead>
|
||||||
|
<date>2025-11-21</date>
|
||||||
|
<version>
|
||||||
|
<release>1.13.1</release>
|
||||||
|
<api>1.13.1</api>
|
||||||
|
</version>
|
||||||
|
<stability>
|
||||||
|
<release>stable</release>
|
||||||
|
<api>stable</api>
|
||||||
|
</stability>
|
||||||
|
<license uri="https://github.com/maxmind/MaxMind-DB-Reader-php/blob/main/LICENSE">Apache License 2.0</license>
|
||||||
|
<notes>* First PIE release. No other changes.</notes>
|
||||||
|
<contents>
|
||||||
|
<dir name="/">
|
||||||
|
<file role="doc" name="LICENSE"/>
|
||||||
|
<file role="doc" name="CHANGELOG.md"/>
|
||||||
|
<file role="doc" name="README.md"/>
|
||||||
|
|
||||||
|
<dir name="ext">
|
||||||
|
<file role="src" name="config.m4"/>
|
||||||
|
<file role="src" name="config.w32"/>
|
||||||
|
|
||||||
|
<file role="src" name="maxminddb.c"/>
|
||||||
|
<file role="src" name="php_maxminddb.h"/>
|
||||||
|
|
||||||
|
<dir name="tests">
|
||||||
|
<file role="test" name="001-load.phpt"/>
|
||||||
|
<file role="test" name="002-final.phpt"/>
|
||||||
|
<file role="test" name="003-open-basedir.phpt"/>
|
||||||
|
</dir>
|
||||||
|
</dir>
|
||||||
|
</dir>
|
||||||
|
</contents>
|
||||||
|
<dependencies>
|
||||||
|
<required>
|
||||||
|
<php>
|
||||||
|
<min>7.2.0</min>
|
||||||
|
</php>
|
||||||
|
<pearinstaller>
|
||||||
|
<min>1.10.0</min>
|
||||||
|
</pearinstaller>
|
||||||
|
</required>
|
||||||
|
</dependencies>
|
||||||
|
<providesextension>maxminddb</providesextension>
|
||||||
|
<extsrcrelease />
|
||||||
|
</package>
|
||||||
404
.hta_lib/maxmind-db-reader/src/MaxMind/Db/Reader.php
Normal file
404
.hta_lib/maxmind-db-reader/src/MaxMind/Db/Reader.php
Normal file
@@ -0,0 +1,404 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace MaxMind\Db;
|
||||||
|
|
||||||
|
use MaxMind\Db\Reader\Decoder;
|
||||||
|
use MaxMind\Db\Reader\InvalidDatabaseException;
|
||||||
|
use MaxMind\Db\Reader\Metadata;
|
||||||
|
use MaxMind\Db\Reader\Util;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Instances of this class provide a reader for the MaxMind DB format. IP
|
||||||
|
* addresses can be looked up using the get method.
|
||||||
|
*/
|
||||||
|
class Reader
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
private static $DATA_SECTION_SEPARATOR_SIZE = 16;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var string
|
||||||
|
*/
|
||||||
|
private static $METADATA_START_MARKER = "\xAB\xCD\xEFMaxMind.com";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var int<0, max>
|
||||||
|
*/
|
||||||
|
private static $METADATA_START_MARKER_LENGTH = 14;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
private static $METADATA_MAX_SIZE = 131072; // 128 * 1024 = 128KiB
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var Decoder
|
||||||
|
*/
|
||||||
|
private $decoder;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var resource
|
||||||
|
*/
|
||||||
|
private $fileHandle;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
private $fileSize;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
private $ipV4Start;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var Metadata
|
||||||
|
*/
|
||||||
|
private $metadata;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a Reader for the MaxMind DB format. The file passed to it must
|
||||||
|
* be a valid MaxMind DB file such as a GeoIp2 database file.
|
||||||
|
*
|
||||||
|
* @param string $database the MaxMind DB file to use
|
||||||
|
*
|
||||||
|
* @throws \InvalidArgumentException for invalid database path or unknown arguments
|
||||||
|
* @throws InvalidDatabaseException
|
||||||
|
* if the database is invalid or there is an error reading
|
||||||
|
* from it
|
||||||
|
*/
|
||||||
|
public function __construct(string $database)
|
||||||
|
{
|
||||||
|
if (\func_num_args() !== 1) {
|
||||||
|
throw new \ArgumentCountError(
|
||||||
|
\sprintf('%s() expects exactly 1 parameter, %d given', __METHOD__, \func_num_args())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_dir($database)) {
|
||||||
|
// This matches the error that the C extension throws.
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
"Error opening database file ($database). Is this a valid MaxMind DB file?"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$fileHandle = @fopen($database, 'rb');
|
||||||
|
if ($fileHandle === false) {
|
||||||
|
throw new \InvalidArgumentException(
|
||||||
|
"The file \"$database\" does not exist or is not readable."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
$this->fileHandle = $fileHandle;
|
||||||
|
|
||||||
|
$fstat = fstat($fileHandle);
|
||||||
|
if ($fstat === false) {
|
||||||
|
throw new \UnexpectedValueException(
|
||||||
|
"Error determining the size of \"$database\"."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
$this->fileSize = $fstat['size'];
|
||||||
|
|
||||||
|
$start = $this->findMetadataStart($database);
|
||||||
|
$metadataDecoder = new Decoder($this->fileHandle, $start);
|
||||||
|
[$metadataArray] = $metadataDecoder->decode($start);
|
||||||
|
$this->metadata = new Metadata($metadataArray);
|
||||||
|
$this->decoder = new Decoder(
|
||||||
|
$this->fileHandle,
|
||||||
|
$this->metadata->searchTreeSize + self::$DATA_SECTION_SEPARATOR_SIZE
|
||||||
|
);
|
||||||
|
$this->ipV4Start = $this->ipV4StartNode();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the record for the IP address.
|
||||||
|
*
|
||||||
|
* @param string $ipAddress the IP address to look up
|
||||||
|
*
|
||||||
|
* @throws \BadMethodCallException if this method is called on a closed database
|
||||||
|
* @throws \InvalidArgumentException if something other than a single IP address is passed to the method
|
||||||
|
* @throws InvalidDatabaseException
|
||||||
|
* if the database is invalid or there is an error reading
|
||||||
|
* from it
|
||||||
|
*
|
||||||
|
* @return mixed the record for the IP address
|
||||||
|
*/
|
||||||
|
public function get(string $ipAddress)
|
||||||
|
{
|
||||||
|
if (\func_num_args() !== 1) {
|
||||||
|
throw new \ArgumentCountError(
|
||||||
|
\sprintf('%s() expects exactly 1 parameter, %d given', __METHOD__, \func_num_args())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[$record] = $this->getWithPrefixLen($ipAddress);
|
||||||
|
|
||||||
|
return $record;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the record for the IP address and its associated network prefix length.
|
||||||
|
*
|
||||||
|
* @param string $ipAddress the IP address to look up
|
||||||
|
*
|
||||||
|
* @throws \BadMethodCallException if this method is called on a closed database
|
||||||
|
* @throws \InvalidArgumentException if something other than a single IP address is passed to the method
|
||||||
|
* @throws InvalidDatabaseException
|
||||||
|
* if the database is invalid or there is an error reading
|
||||||
|
* from it
|
||||||
|
*
|
||||||
|
* @return array{0:mixed, 1:int} an array where the first element is the record and the
|
||||||
|
* second the network prefix length for the record
|
||||||
|
*/
|
||||||
|
public function getWithPrefixLen(string $ipAddress): array
|
||||||
|
{
|
||||||
|
if (\func_num_args() !== 1) {
|
||||||
|
throw new \ArgumentCountError(
|
||||||
|
\sprintf('%s() expects exactly 1 parameter, %d given', __METHOD__, \func_num_args())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!\is_resource($this->fileHandle)) {
|
||||||
|
throw new \BadMethodCallException(
|
||||||
|
'Attempt to read from a closed MaxMind DB.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
[$pointer, $prefixLen] = $this->findAddressInTree($ipAddress);
|
||||||
|
if ($pointer === 0) {
|
||||||
|
return [null, $prefixLen];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [$this->resolveDataPointer($pointer), $prefixLen];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{0:int, 1:int}
|
||||||
|
*/
|
||||||
|
private function findAddressInTree(string $ipAddress): array
|
||||||
|
{
|
||||||
|
$packedAddr = @inet_pton($ipAddress);
|
||||||
|
if ($packedAddr === false) {
|
||||||
|
throw new \InvalidArgumentException(
|
||||||
|
"The value \"$ipAddress\" is not a valid IP address."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$rawAddress = unpack('C*', $packedAddr);
|
||||||
|
if ($rawAddress === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack the unsigned char of the packed in_addr representation.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$bitCount = \count($rawAddress) * 8;
|
||||||
|
|
||||||
|
// The first node of the tree is always node 0, at the beginning of the
|
||||||
|
// value
|
||||||
|
$node = 0;
|
||||||
|
|
||||||
|
$metadata = $this->metadata;
|
||||||
|
|
||||||
|
// Check if we are looking up an IPv4 address in an IPv6 tree. If this
|
||||||
|
// is the case, we can skip over the first 96 nodes.
|
||||||
|
if ($metadata->ipVersion === 6) {
|
||||||
|
if ($bitCount === 32) {
|
||||||
|
$node = $this->ipV4Start;
|
||||||
|
}
|
||||||
|
} elseif ($metadata->ipVersion === 4 && $bitCount === 128) {
|
||||||
|
throw new \InvalidArgumentException(
|
||||||
|
"Error looking up $ipAddress. You attempted to look up an"
|
||||||
|
. ' IPv6 address in an IPv4-only database.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$nodeCount = $metadata->nodeCount;
|
||||||
|
|
||||||
|
for ($i = 0; $i < $bitCount && $node < $nodeCount; ++$i) {
|
||||||
|
$tempBit = 0xFF & $rawAddress[($i >> 3) + 1];
|
||||||
|
$bit = 1 & ($tempBit >> 7 - ($i % 8));
|
||||||
|
|
||||||
|
$node = $this->readNode($node, $bit);
|
||||||
|
}
|
||||||
|
if ($node === $nodeCount) {
|
||||||
|
// Record is empty
|
||||||
|
return [0, $i];
|
||||||
|
}
|
||||||
|
if ($node > $nodeCount) {
|
||||||
|
// Record is a data pointer
|
||||||
|
return [$node, $i];
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Invalid or corrupt database. Maximum search depth reached without finding a leaf node'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function ipV4StartNode(): int
|
||||||
|
{
|
||||||
|
// If we have an IPv4 database, the start node is the first node
|
||||||
|
if ($this->metadata->ipVersion === 4) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$node = 0;
|
||||||
|
|
||||||
|
for ($i = 0; $i < 96 && $node < $this->metadata->nodeCount; ++$i) {
|
||||||
|
$node = $this->readNode($node, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $node;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function readNode(int $nodeNumber, int $index): int
|
||||||
|
{
|
||||||
|
$baseOffset = $nodeNumber * $this->metadata->nodeByteSize;
|
||||||
|
|
||||||
|
switch ($this->metadata->recordSize) {
|
||||||
|
case 24:
|
||||||
|
$bytes = Util::read($this->fileHandle, $baseOffset + $index * 3, 3);
|
||||||
|
$rc = unpack('N', "\x00" . $bytes);
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack the unsigned long of the node.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $node] = $rc;
|
||||||
|
|
||||||
|
return $node;
|
||||||
|
|
||||||
|
case 28:
|
||||||
|
$bytes = Util::read($this->fileHandle, $baseOffset + 3 * $index, 4);
|
||||||
|
if ($index === 0) {
|
||||||
|
$middle = (0xF0 & \ord($bytes[3])) >> 4;
|
||||||
|
} else {
|
||||||
|
$middle = 0x0F & \ord($bytes[0]);
|
||||||
|
}
|
||||||
|
$rc = unpack('N', \chr($middle) . substr($bytes, $index, 3));
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack the unsigned long of the node.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $node] = $rc;
|
||||||
|
|
||||||
|
return $node;
|
||||||
|
|
||||||
|
case 32:
|
||||||
|
$bytes = Util::read($this->fileHandle, $baseOffset + $index * 4, 4);
|
||||||
|
$rc = unpack('N', $bytes);
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack the unsigned long of the node.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $node] = $rc;
|
||||||
|
|
||||||
|
return $node;
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Unknown record size: '
|
||||||
|
. $this->metadata->recordSize
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
private function resolveDataPointer(int $pointer)
|
||||||
|
{
|
||||||
|
$resolved = $pointer - $this->metadata->nodeCount
|
||||||
|
+ $this->metadata->searchTreeSize;
|
||||||
|
if ($resolved >= $this->fileSize) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
"The MaxMind DB file's search tree is corrupt"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
[$data] = $this->decoder->decode($resolved);
|
||||||
|
|
||||||
|
return $data;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This is an extremely naive but reasonably readable implementation. There
|
||||||
|
* are much faster algorithms (e.g., Boyer-Moore) for this if speed is ever
|
||||||
|
* an issue, but I suspect it won't be.
|
||||||
|
*/
|
||||||
|
private function findMetadataStart(string $filename): int
|
||||||
|
{
|
||||||
|
$handle = $this->fileHandle;
|
||||||
|
$fileSize = $this->fileSize;
|
||||||
|
$marker = self::$METADATA_START_MARKER;
|
||||||
|
$markerLength = self::$METADATA_START_MARKER_LENGTH;
|
||||||
|
|
||||||
|
$minStart = $fileSize - min(self::$METADATA_MAX_SIZE, $fileSize);
|
||||||
|
|
||||||
|
for ($offset = $fileSize - $markerLength; $offset >= $minStart; --$offset) {
|
||||||
|
if (fseek($handle, $offset) !== 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
$value = fread($handle, $markerLength);
|
||||||
|
if ($value === $marker) {
|
||||||
|
return $offset + $markerLength;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
"Error opening database file ($filename). "
|
||||||
|
. 'Is this a valid MaxMind DB file?'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws \InvalidArgumentException if arguments are passed to the method
|
||||||
|
* @throws \BadMethodCallException if the database has been closed
|
||||||
|
*
|
||||||
|
* @return Metadata object for the database
|
||||||
|
*/
|
||||||
|
public function metadata(): Metadata
|
||||||
|
{
|
||||||
|
if (\func_num_args()) {
|
||||||
|
throw new \ArgumentCountError(
|
||||||
|
\sprintf('%s() expects exactly 0 parameters, %d given', __METHOD__, \func_num_args())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Not technically required, but this makes it consistent with
|
||||||
|
// C extension and it allows us to change our implementation later.
|
||||||
|
if (!\is_resource($this->fileHandle)) {
|
||||||
|
throw new \BadMethodCallException(
|
||||||
|
'Attempt to read from a closed MaxMind DB.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return clone $this->metadata;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Closes the MaxMind DB and returns resources to the system.
|
||||||
|
*
|
||||||
|
* @throws \Exception
|
||||||
|
* if an I/O error occurs
|
||||||
|
*/
|
||||||
|
public function close(): void
|
||||||
|
{
|
||||||
|
if (\func_num_args()) {
|
||||||
|
throw new \ArgumentCountError(
|
||||||
|
\sprintf('%s() expects exactly 0 parameters, %d given', __METHOD__, \func_num_args())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!\is_resource($this->fileHandle)) {
|
||||||
|
throw new \BadMethodCallException(
|
||||||
|
'Attempt to close a closed MaxMind DB.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
fclose($this->fileHandle);
|
||||||
|
}
|
||||||
|
}
|
||||||
452
.hta_lib/maxmind-db-reader/src/MaxMind/Db/Reader/Decoder.php
Normal file
452
.hta_lib/maxmind-db-reader/src/MaxMind/Db/Reader/Decoder.php
Normal file
@@ -0,0 +1,452 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace MaxMind\Db\Reader;
|
||||||
|
|
||||||
|
// @codingStandardsIgnoreLine
|
||||||
|
|
||||||
|
class Decoder
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @var resource
|
||||||
|
*/
|
||||||
|
private $fileStream;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
private $pointerBase;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is only used for unit testing.
|
||||||
|
*
|
||||||
|
* @var bool
|
||||||
|
*/
|
||||||
|
private $pointerTestHack;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var bool
|
||||||
|
*/
|
||||||
|
private $switchByteOrder;
|
||||||
|
|
||||||
|
private const _EXTENDED = 0;
|
||||||
|
private const _POINTER = 1;
|
||||||
|
private const _UTF8_STRING = 2;
|
||||||
|
private const _DOUBLE = 3;
|
||||||
|
private const _BYTES = 4;
|
||||||
|
private const _UINT16 = 5;
|
||||||
|
private const _UINT32 = 6;
|
||||||
|
private const _MAP = 7;
|
||||||
|
private const _INT32 = 8;
|
||||||
|
private const _UINT64 = 9;
|
||||||
|
private const _UINT128 = 10;
|
||||||
|
private const _ARRAY = 11;
|
||||||
|
// 12 is the container type
|
||||||
|
// 13 is the end marker type
|
||||||
|
private const _BOOLEAN = 14;
|
||||||
|
private const _FLOAT = 15;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param resource $fileStream
|
||||||
|
*/
|
||||||
|
public function __construct(
|
||||||
|
$fileStream,
|
||||||
|
int $pointerBase = 0,
|
||||||
|
bool $pointerTestHack = false
|
||||||
|
) {
|
||||||
|
$this->fileStream = $fileStream;
|
||||||
|
$this->pointerBase = $pointerBase;
|
||||||
|
|
||||||
|
$this->pointerTestHack = $pointerTestHack;
|
||||||
|
|
||||||
|
$this->switchByteOrder = $this->isPlatformLittleEndian();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array<mixed>
|
||||||
|
*/
|
||||||
|
public function decode(int $offset): array
|
||||||
|
{
|
||||||
|
$ctrlByte = \ord(Util::read($this->fileStream, $offset, 1));
|
||||||
|
++$offset;
|
||||||
|
|
||||||
|
$type = $ctrlByte >> 5;
|
||||||
|
|
||||||
|
// Pointers are a special case, we don't read the next $size bytes, we
|
||||||
|
// use the size to determine the length of the pointer and then follow
|
||||||
|
// it.
|
||||||
|
if ($type === self::_POINTER) {
|
||||||
|
[$pointer, $offset] = $this->decodePointer($ctrlByte, $offset);
|
||||||
|
|
||||||
|
// for unit testing
|
||||||
|
if ($this->pointerTestHack) {
|
||||||
|
return [$pointer];
|
||||||
|
}
|
||||||
|
|
||||||
|
[$result] = $this->decode($pointer);
|
||||||
|
|
||||||
|
return [$result, $offset];
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($type === self::_EXTENDED) {
|
||||||
|
$nextByte = \ord(Util::read($this->fileStream, $offset, 1));
|
||||||
|
|
||||||
|
$type = $nextByte + 7;
|
||||||
|
|
||||||
|
if ($type < 8) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Something went horribly wrong in the decoder. An extended type '
|
||||||
|
. 'resolved to a type number < 8 ('
|
||||||
|
. $type
|
||||||
|
. ')'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
++$offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
[$size, $offset] = $this->sizeFromCtrlByte($ctrlByte, $offset);
|
||||||
|
|
||||||
|
return $this->decodeByType($type, $offset, $size);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param int<0, max> $size
|
||||||
|
*
|
||||||
|
* @return array{0:mixed, 1:int}
|
||||||
|
*/
|
||||||
|
private function decodeByType(int $type, int $offset, int $size): array
|
||||||
|
{
|
||||||
|
switch ($type) {
|
||||||
|
case self::_MAP:
|
||||||
|
return $this->decodeMap($size, $offset);
|
||||||
|
|
||||||
|
case self::_ARRAY:
|
||||||
|
return $this->decodeArray($size, $offset);
|
||||||
|
|
||||||
|
case self::_BOOLEAN:
|
||||||
|
return [$this->decodeBoolean($size), $offset];
|
||||||
|
}
|
||||||
|
|
||||||
|
$newOffset = $offset + $size;
|
||||||
|
$bytes = Util::read($this->fileStream, $offset, $size);
|
||||||
|
|
||||||
|
switch ($type) {
|
||||||
|
case self::_BYTES:
|
||||||
|
case self::_UTF8_STRING:
|
||||||
|
return [$bytes, $newOffset];
|
||||||
|
|
||||||
|
case self::_DOUBLE:
|
||||||
|
$this->verifySize(8, $size);
|
||||||
|
|
||||||
|
return [$this->decodeDouble($bytes), $newOffset];
|
||||||
|
|
||||||
|
case self::_FLOAT:
|
||||||
|
$this->verifySize(4, $size);
|
||||||
|
|
||||||
|
return [$this->decodeFloat($bytes), $newOffset];
|
||||||
|
|
||||||
|
case self::_INT32:
|
||||||
|
return [$this->decodeInt32($bytes, $size), $newOffset];
|
||||||
|
|
||||||
|
case self::_UINT16:
|
||||||
|
case self::_UINT32:
|
||||||
|
case self::_UINT64:
|
||||||
|
case self::_UINT128:
|
||||||
|
return [$this->decodeUint($bytes, $size), $newOffset];
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Unknown or unexpected type: ' . $type
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function verifySize(int $expected, int $actual): void
|
||||||
|
{
|
||||||
|
if ($expected !== $actual) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
"The MaxMind DB file's data section contains bad data (unknown data type or corrupt data)"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{0:array<mixed>, 1:int}
|
||||||
|
*/
|
||||||
|
private function decodeArray(int $size, int $offset): array
|
||||||
|
{
|
||||||
|
$array = [];
|
||||||
|
|
||||||
|
for ($i = 0; $i < $size; ++$i) {
|
||||||
|
[$value, $offset] = $this->decode($offset);
|
||||||
|
$array[] = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [$array, $offset];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function decodeBoolean(int $size): bool
|
||||||
|
{
|
||||||
|
return $size !== 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function decodeDouble(string $bytes): float
|
||||||
|
{
|
||||||
|
// This assumes IEEE 754 doubles, but most (all?) modern platforms
|
||||||
|
// use them.
|
||||||
|
$rc = unpack('E', $bytes);
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack a double value from the given bytes.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $double] = $rc;
|
||||||
|
|
||||||
|
return $double;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function decodeFloat(string $bytes): float
|
||||||
|
{
|
||||||
|
// This assumes IEEE 754 floats, but most (all?) modern platforms
|
||||||
|
// use them.
|
||||||
|
$rc = unpack('G', $bytes);
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack a float value from the given bytes.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $float] = $rc;
|
||||||
|
|
||||||
|
return $float;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function decodeInt32(string $bytes, int $size): int
|
||||||
|
{
|
||||||
|
switch ($size) {
|
||||||
|
case 0:
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
case 1:
|
||||||
|
case 2:
|
||||||
|
case 3:
|
||||||
|
$bytes = str_pad($bytes, 4, "\x00", \STR_PAD_LEFT);
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 4:
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
"The MaxMind DB file's data section contains bad data (unknown data type or corrupt data)"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$rc = unpack('l', $this->maybeSwitchByteOrder($bytes));
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack a 32bit integer value from the given bytes.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $int] = $rc;
|
||||||
|
|
||||||
|
return $int;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{0:array<string, mixed>, 1:int}
|
||||||
|
*/
|
||||||
|
private function decodeMap(int $size, int $offset): array
|
||||||
|
{
|
||||||
|
$map = [];
|
||||||
|
|
||||||
|
for ($i = 0; $i < $size; ++$i) {
|
||||||
|
[$key, $offset] = $this->decode($offset);
|
||||||
|
[$value, $offset] = $this->decode($offset);
|
||||||
|
$map[$key] = $value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [$map, $offset];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{0:int, 1:int}
|
||||||
|
*/
|
||||||
|
private function decodePointer(int $ctrlByte, int $offset): array
|
||||||
|
{
|
||||||
|
$pointerSize = (($ctrlByte >> 3) & 0x3) + 1;
|
||||||
|
|
||||||
|
$buffer = Util::read($this->fileStream, $offset, $pointerSize);
|
||||||
|
$offset += $pointerSize;
|
||||||
|
|
||||||
|
switch ($pointerSize) {
|
||||||
|
case 1:
|
||||||
|
$packed = \chr($ctrlByte & 0x7) . $buffer;
|
||||||
|
$rc = unpack('n', $packed);
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack an unsigned short value from the given bytes (pointerSize is 1).'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $pointer] = $rc;
|
||||||
|
$pointer += $this->pointerBase;
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
$packed = "\x00" . \chr($ctrlByte & 0x7) . $buffer;
|
||||||
|
$rc = unpack('N', $packed);
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack an unsigned long value from the given bytes (pointerSize is 2).'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $pointer] = $rc;
|
||||||
|
$pointer += $this->pointerBase + 2048;
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 3:
|
||||||
|
$packed = \chr($ctrlByte & 0x7) . $buffer;
|
||||||
|
|
||||||
|
// It is safe to use 'N' here, even on 32 bit machines as the
|
||||||
|
// first bit is 0.
|
||||||
|
$rc = unpack('N', $packed);
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack an unsigned long value from the given bytes (pointerSize is 3).'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $pointer] = $rc;
|
||||||
|
$pointer += $this->pointerBase + 526336;
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 4:
|
||||||
|
// We cannot use unpack here as we might overflow on 32 bit
|
||||||
|
// machines
|
||||||
|
$pointerOffset = $this->decodeUint($buffer, $pointerSize);
|
||||||
|
|
||||||
|
$pointerBase = $this->pointerBase;
|
||||||
|
|
||||||
|
if (\PHP_INT_MAX - $pointerBase >= $pointerOffset) {
|
||||||
|
$pointer = $pointerOffset + $pointerBase;
|
||||||
|
} else {
|
||||||
|
throw new \RuntimeException(
|
||||||
|
'The database offset is too large to be represented on your platform.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Unexpected pointer size ' . $pointerSize
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return [$pointer, $offset];
|
||||||
|
}
|
||||||
|
|
||||||
|
// @phpstan-ignore-next-line
|
||||||
|
private function decodeUint(string $bytes, int $byteLength)
|
||||||
|
{
|
||||||
|
if ($byteLength === 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// PHP integers are signed. PHP_INT_SIZE - 1 is the number of
|
||||||
|
// complete bytes that can be converted to an integer. However,
|
||||||
|
// we can convert another byte if the leading bit is zero.
|
||||||
|
$useRealInts = $byteLength <= \PHP_INT_SIZE - 1
|
||||||
|
|| ($byteLength === \PHP_INT_SIZE && (\ord($bytes[0]) & 0x80) === 0);
|
||||||
|
|
||||||
|
if ($useRealInts) {
|
||||||
|
$integer = 0;
|
||||||
|
for ($i = 0; $i < $byteLength; ++$i) {
|
||||||
|
$part = \ord($bytes[$i]);
|
||||||
|
$integer = ($integer << 8) + $part;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $integer;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We only use gmp or bcmath if the final value is too big
|
||||||
|
$integerAsString = '0';
|
||||||
|
for ($i = 0; $i < $byteLength; ++$i) {
|
||||||
|
$part = \ord($bytes[$i]);
|
||||||
|
|
||||||
|
if (\extension_loaded('gmp')) {
|
||||||
|
$integerAsString = gmp_strval(gmp_add(gmp_mul($integerAsString, '256'), $part));
|
||||||
|
} elseif (\extension_loaded('bcmath')) {
|
||||||
|
$integerAsString = bcadd(bcmul($integerAsString, '256'), (string) $part);
|
||||||
|
} else {
|
||||||
|
throw new \RuntimeException(
|
||||||
|
'The gmp or bcmath extension must be installed to read this database.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $integerAsString;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return array{0:int, 1:int}
|
||||||
|
*/
|
||||||
|
private function sizeFromCtrlByte(int $ctrlByte, int $offset): array
|
||||||
|
{
|
||||||
|
$size = $ctrlByte & 0x1F;
|
||||||
|
|
||||||
|
if ($size < 29) {
|
||||||
|
return [$size, $offset];
|
||||||
|
}
|
||||||
|
|
||||||
|
$bytesToRead = $size - 28;
|
||||||
|
$bytes = Util::read($this->fileStream, $offset, $bytesToRead);
|
||||||
|
|
||||||
|
if ($size === 29) {
|
||||||
|
$size = 29 + \ord($bytes);
|
||||||
|
} elseif ($size === 30) {
|
||||||
|
$rc = unpack('n', $bytes);
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack an unsigned short value from the given bytes.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $adjust] = $rc;
|
||||||
|
$size = 285 + $adjust;
|
||||||
|
} else {
|
||||||
|
$rc = unpack('N', "\x00" . $bytes);
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack an unsigned long value from the given bytes.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
[, $adjust] = $rc;
|
||||||
|
$size = $adjust + 65821;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [$size, $offset + $bytesToRead];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function maybeSwitchByteOrder(string $bytes): string
|
||||||
|
{
|
||||||
|
return $this->switchByteOrder ? strrev($bytes) : $bytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function isPlatformLittleEndian(): bool
|
||||||
|
{
|
||||||
|
$testint = 0x00FF;
|
||||||
|
$packed = pack('S', $testint);
|
||||||
|
$rc = unpack('v', $packed);
|
||||||
|
if ($rc === false) {
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'Could not unpack an unsigned short value from the given bytes.'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $testint === current($rc);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace MaxMind\Db\Reader;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class should be thrown when unexpected data is found in the database.
|
||||||
|
*/
|
||||||
|
// phpcs:disable
|
||||||
|
class InvalidDatabaseException extends \Exception {}
|
||||||
123
.hta_lib/maxmind-db-reader/src/MaxMind/Db/Reader/Metadata.php
Normal file
123
.hta_lib/maxmind-db-reader/src/MaxMind/Db/Reader/Metadata.php
Normal file
@@ -0,0 +1,123 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace MaxMind\Db\Reader;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class provides the metadata for the MaxMind DB file.
|
||||||
|
*/
|
||||||
|
class Metadata
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* This is an unsigned 16-bit integer indicating the major version number
|
||||||
|
* for the database's binary format.
|
||||||
|
*
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
public $binaryFormatMajorVersion;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is an unsigned 16-bit integer indicating the minor version number
|
||||||
|
* for the database's binary format.
|
||||||
|
*
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
public $binaryFormatMinorVersion;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is an unsigned 64-bit integer that contains the database build
|
||||||
|
* timestamp as a Unix epoch value.
|
||||||
|
*
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
public $buildEpoch;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is a string that indicates the structure of each data record
|
||||||
|
* associated with an IP address. The actual definition of these
|
||||||
|
* structures is left up to the database creator.
|
||||||
|
*
|
||||||
|
* @var string
|
||||||
|
*/
|
||||||
|
public $databaseType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This key will always point to a map (associative array). The keys of
|
||||||
|
* that map will be language codes, and the values will be a description
|
||||||
|
* in that language as a UTF-8 string. May be undefined for some
|
||||||
|
* databases.
|
||||||
|
*
|
||||||
|
* @var array<string, string>
|
||||||
|
*/
|
||||||
|
public $description;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is an unsigned 16-bit integer which is always 4 or 6. It indicates
|
||||||
|
* whether the database contains IPv4 or IPv6 address data.
|
||||||
|
*
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
public $ipVersion;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An array of strings, each of which is a language code. A given record
|
||||||
|
* may contain data items that have been localized to some or all of
|
||||||
|
* these languages. This may be undefined.
|
||||||
|
*
|
||||||
|
* @var array<string>
|
||||||
|
*/
|
||||||
|
public $languages;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
public $nodeByteSize;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is an unsigned 32-bit integer indicating the number of nodes in
|
||||||
|
* the search tree.
|
||||||
|
*
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
public $nodeCount;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is an unsigned 16-bit integer. It indicates the number of bits in a
|
||||||
|
* record in the search tree. Note that each node consists of two records.
|
||||||
|
*
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
public $recordSize;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var int
|
||||||
|
*/
|
||||||
|
public $searchTreeSize;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array<string, mixed> $metadata
|
||||||
|
*/
|
||||||
|
public function __construct(array $metadata)
|
||||||
|
{
|
||||||
|
if (\func_num_args() !== 1) {
|
||||||
|
throw new \ArgumentCountError(
|
||||||
|
\sprintf('%s() expects exactly 1 parameter, %d given', __METHOD__, \func_num_args())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->binaryFormatMajorVersion
|
||||||
|
= $metadata['binary_format_major_version'];
|
||||||
|
$this->binaryFormatMinorVersion
|
||||||
|
= $metadata['binary_format_minor_version'];
|
||||||
|
$this->buildEpoch = $metadata['build_epoch'];
|
||||||
|
$this->databaseType = $metadata['database_type'];
|
||||||
|
$this->languages = $metadata['languages'];
|
||||||
|
$this->description = $metadata['description'];
|
||||||
|
$this->ipVersion = $metadata['ip_version'];
|
||||||
|
$this->nodeCount = $metadata['node_count'];
|
||||||
|
$this->recordSize = $metadata['record_size'];
|
||||||
|
$this->nodeByteSize = $this->recordSize / 4;
|
||||||
|
$this->searchTreeSize = $this->nodeCount * $this->nodeByteSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
33
.hta_lib/maxmind-db-reader/src/MaxMind/Db/Reader/Util.php
Normal file
33
.hta_lib/maxmind-db-reader/src/MaxMind/Db/Reader/Util.php
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace MaxMind\Db\Reader;
|
||||||
|
|
||||||
|
class Util
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @param resource $stream
|
||||||
|
* @param int<0, max> $numberOfBytes
|
||||||
|
*/
|
||||||
|
public static function read($stream, int $offset, int $numberOfBytes): string
|
||||||
|
{
|
||||||
|
if ($numberOfBytes === 0) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
if (fseek($stream, $offset) === 0) {
|
||||||
|
$value = fread($stream, $numberOfBytes);
|
||||||
|
|
||||||
|
// We check that the number of bytes read is equal to the number
|
||||||
|
// asked for. We use ftell as getting the length of $value is
|
||||||
|
// much slower.
|
||||||
|
if ($value !== false && ftell($stream) - $offset === $numberOfBytes) {
|
||||||
|
return $value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new InvalidDatabaseException(
|
||||||
|
'The MaxMind DB file contains bad data'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,9 +1,10 @@
|
|||||||
<?php
|
<?php
|
||||||
require_once('.hta_config/db_config.php');
|
require_once(__DIR__.'/../.hta_config/conf.php');
|
||||||
require_once('.htac_header.php');
|
require_once(__DIR__.'/../.hta_slug/_header.php');
|
||||||
|
require_once(__DIR__.'/../.hta_slug/_nav.php');
|
||||||
?>
|
?>
|
||||||
|
|
||||||
<section class="diZContainer diZmxAuto">
|
<section class="diZContainer diZmxAuto diZmb20">
|
||||||
<h1 class="diZBorderBottom">SiliconPin - Tools</h1>
|
<h1 class="diZBorderBottom">SiliconPin - Tools</h1>
|
||||||
<div class="diZFlexRow diZmy8 diZFlexRow">
|
<div class="diZFlexRow diZmy8 diZFlexRow">
|
||||||
<a href="/tools" class="diZButtonDefault diZmr2"><span>All</span></a>
|
<a href="/tools" class="diZButtonDefault diZmr2"><span>All</span></a>
|
||||||
@@ -49,4 +50,61 @@ require_once('.htac_header.php');
|
|||||||
}
|
}
|
||||||
?>
|
?>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
<style>
|
||||||
|
.diZContainer {
|
||||||
|
max-width: 1200px;
|
||||||
|
margin: auto;
|
||||||
|
padding: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZBorderBottom {
|
||||||
|
border-bottom: 2px solid #87ab63;
|
||||||
|
padding-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZFlexRow {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZButtonDefault {
|
||||||
|
background: #87ab63;
|
||||||
|
color: #121212;
|
||||||
|
padding: 10px 15px;
|
||||||
|
border-radius: 5px;
|
||||||
|
text-decoration: none;
|
||||||
|
font-weight: bold;
|
||||||
|
transition: 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZButtonDefault:hover {
|
||||||
|
background: #6f8f52;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZGridCols1-3 {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||||
|
gap: 15px;
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZblogStyle {
|
||||||
|
background: #1e1e1e;
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 15px;
|
||||||
|
box-shadow: 0px 0px 10px rgba(135, 171, 99, 0.3);
|
||||||
|
transition: transform 0.3s ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZblogStyle:hover {
|
||||||
|
transform: translateY(-5px);
|
||||||
|
box-shadow: 0px 0px 15px rgba(135, 171, 99, 0.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZToolsContentHeight {
|
||||||
|
min-height: 60px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
0
.hta_slug/_footer.php
Normal file
0
.hta_slug/_footer.php
Normal file
0
.hta_slug/_header.php
Normal file
0
.hta_slug/_header.php
Normal file
@@ -1,9 +1,10 @@
|
|||||||
<?php
|
<?php
|
||||||
require_once('.hta_config/db_config.php');
|
require_once(__DIR__.'/../.hta_config/conf.php');
|
||||||
require_once('.htac_header.php');
|
require_once(__DIR__.'/../.hta_slug/_header.php');
|
||||||
|
require_once(__DIR__.'/../.hta_slug/_nav.php');
|
||||||
?>
|
?>
|
||||||
|
|
||||||
<section class="diZContainer diZmxAuto">
|
<section class="diZContainer diZmxAuto diZmb20">
|
||||||
<h1 class="diZBorderBottom">SiliconPin - Tools</h1>
|
<h1 class="diZBorderBottom">SiliconPin - Tools</h1>
|
||||||
<div class="diZFlexRow diZmy8 diZFlexRow diZOverflowAuto diZWhiteSpaceNowrap">
|
<div class="diZFlexRow diZmy8 diZFlexRow diZOverflowAuto diZWhiteSpaceNowrap">
|
||||||
<a href="/tools" class="diZButtonDefault diZmr2"><span>All</span></a>
|
<a href="/tools" class="diZButtonDefault diZmr2"><span>All</span></a>
|
||||||
@@ -41,4 +42,63 @@ require_once('.htac_header.php');
|
|||||||
}
|
}
|
||||||
?>
|
?>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
<style>
|
||||||
|
.diZContainer {
|
||||||
|
max-width: 1200px;
|
||||||
|
margin: auto;
|
||||||
|
padding: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZBorderBottom {
|
||||||
|
border-bottom: 2px solid #87ab63;
|
||||||
|
padding-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZFlexRow {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZButtonDefault {
|
||||||
|
background: #87ab63;
|
||||||
|
color: #121212;
|
||||||
|
padding: 10px 15px;
|
||||||
|
border-radius: 5px;
|
||||||
|
text-decoration: none;
|
||||||
|
font-weight: bold;
|
||||||
|
transition: 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZButtonDefault:hover {
|
||||||
|
background: #6f8f52;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZGridCols1-3 {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||||
|
gap: 15px;
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZblogStyle {
|
||||||
|
background: #1e1e1e;
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 15px;
|
||||||
|
box-shadow: 0px 0px 10px rgba(135, 171, 99, 0.3);
|
||||||
|
transition: transform 0.3s ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZblogStyle:hover {
|
||||||
|
transform: translateY(-5px);
|
||||||
|
box-shadow: 0px 0px 15px rgba(135, 171, 99, 0.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
.diZToolsContentHeight {
|
||||||
|
min-height: 60px;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
</style>
|
||||||
0
.hta_slug/_nav.php
Normal file
0
.hta_slug/_nav.php
Normal file
@@ -1,5 +1,6 @@
|
|||||||
<?php
|
<?php
|
||||||
require_once('.hta_config/siliconpin_sp.php');
|
require_once('../.hta_config/conf.php');
|
||||||
|
require_once('../.hta_slug/_header.php');
|
||||||
|
|
||||||
if($_SERVER['REQUEST_METHOD']=='POST' && strlen($_POST['name']>3)){
|
if($_SERVER['REQUEST_METHOD']=='POST' && strlen($_POST['name']>3)){
|
||||||
$slugText = $_POST['name'];
|
$slugText = $_POST['name'];
|
||||||
113
.hta_slug/color-code-from-image.php
Normal file
113
.hta_slug/color-code-from-image.php
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
<section class="diZContainer diZmxAuto diZmy8">
|
||||||
|
<h1 class="diZBorderBottom">Color Hex/RGB Code From Image</h1>
|
||||||
|
<p>The Color Code Extractor from Image is a simple and efficient web-based tool that allows users to upload an image and click on any part of it to get the RGB and HEX color codes of the pixel they clicked on. This tool is particularly useful for designers, developers, and anyone who needs to identify and use specific colors from images.</p>
|
||||||
|
<div class="diZMaxW600 diZmxAuto toolsSection">
|
||||||
|
<label class="diZFlexColumn " for="imageInput">Choose Image</label><br>
|
||||||
|
<input class="diZPadding10px" type="file" id="imageInput" accept="image/*" /><br><br>
|
||||||
|
<canvas class="diZDisplayNone diZw100" id="canvas"></canvas><br>
|
||||||
|
<div class="diZFlexBetween">
|
||||||
|
<div id="result">Click on the image to get the color code.</div>
|
||||||
|
<button class="diZDisplayNone" id="copyButton" ><span>Copy</span></button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h2>Usage:</h2>
|
||||||
|
<ul>
|
||||||
|
<li><strong>Upload an Image:</strong> Click on the file input field and select an image file from your device.</li>
|
||||||
|
<li><strong>Display the Image:</strong> Once the image is uploaded, it will be displayed on a hidden canvas element.</li>
|
||||||
|
<li><strong>Extract Color Code:</strong> Click on any part of the displayed image. The tool will capture the color of the clicked pixel and display both the RGB and HEX color codes.</li>
|
||||||
|
<li><strong>Copy Color Code:</strong> Click the "Copy Color Code" button to copy the displayed color code to your clipboard for easy use in your projects.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<h2>Features:</h2>
|
||||||
|
<ul>
|
||||||
|
<li><strong>Easy Image Upload:</strong> Supports uploading images directly from your device.</li>
|
||||||
|
<li><strong>Accurate Color Extraction:</strong> Retrieves the exact RGB and HEX color codes from any pixel of the image.</li>
|
||||||
|
<li><strong>Clipboard Copy Functionality:</strong> Allows you to easily copy the extracted color codes to your clipboard.</li>
|
||||||
|
<li><strong>Fallback for Compatibility:</strong> Provides a fallback method for copying color codes on browsers that do not support the Clipboard API.</li>
|
||||||
|
<li><strong>User-Friendly Interface:</strong> Simple and intuitive design for ease of use.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<h2>Use Case:</h2>
|
||||||
|
<ul>
|
||||||
|
<li><strong>Web Designers and Developers:</strong> Quickly extract color codes from images for use in web design and development projects.</li>
|
||||||
|
<li><strong>Graphic Designers:</strong> Identify and use specific colors from images in design software.</li>
|
||||||
|
<li><strong>Digital Artists:</strong> Match colors from reference images to create cohesive artworks.</li>
|
||||||
|
<li><strong>Marketing Professionals:</strong> Ensure brand consistency by extracting and using exact colors from marketing materials and logos.</li>
|
||||||
|
<li><strong>DIY Enthusiasts:</strong> Use color codes to match paint colors, fabrics, or other materials in home decoration projects.</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<script>
|
||||||
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
|
const imageInput = document.getElementById('imageInput');
|
||||||
|
const copyButton = document.getElementById('copyButton');
|
||||||
|
|
||||||
|
imageInput.addEventListener('change', function(event) {
|
||||||
|
const file = event.target.files[0];
|
||||||
|
if (file) {
|
||||||
|
const reader = new FileReader();
|
||||||
|
reader.onload = function(e) {
|
||||||
|
const img = new Image();
|
||||||
|
img.onload = function() {
|
||||||
|
const canvas = document.getElementById('canvas');
|
||||||
|
const ctx = canvas.getContext('2d');
|
||||||
|
canvas.width = img.width;
|
||||||
|
canvas.height = img.height;
|
||||||
|
ctx.drawImage(img, 0, 0);
|
||||||
|
canvas.style.display = 'block';
|
||||||
|
|
||||||
|
canvas.addEventListener('click', function(event) {
|
||||||
|
const rect = canvas.getBoundingClientRect();
|
||||||
|
const x = event.clientX - rect.left;
|
||||||
|
const y = event.clientY - rect.top;
|
||||||
|
console.log(`Clicked at (x, y): (${x}, ${y})`);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const imageData = ctx.getImageData(x, y, 1, 1).data;
|
||||||
|
const rgb = `rgb(${imageData[0]}, ${imageData[1]}, ${imageData[2]})`;
|
||||||
|
const hex = `#${((1 << 24) + (imageData[0] << 16) + (imageData[1] << 8) + imageData[2]).toString(16).slice(1).toUpperCase()}`;
|
||||||
|
const colorCode = `RGB: ${rgb}, HEX: ${hex}`;
|
||||||
|
console.log('Color code:', colorCode);
|
||||||
|
document.getElementById('result').innerText = colorCode;
|
||||||
|
copyButton.style.display = 'block';
|
||||||
|
copyButton.setAttribute('data-color-code', colorCode);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Error getting image data:', err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
img.src = e.target.result;
|
||||||
|
};
|
||||||
|
reader.readAsDataURL(file);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
copyButton.addEventListener('click', function() {
|
||||||
|
const colorCode = copyButton.getAttribute('data-color-code');
|
||||||
|
if (navigator.clipboard && navigator.clipboard.writeText) {
|
||||||
|
navigator.clipboard.writeText(colorCode).then(() => {
|
||||||
|
console.log('Color code copied to clipboard');
|
||||||
|
}).catch(err => {
|
||||||
|
console.error('Could not copy text: ', err);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
const textarea = document.createElement('textarea');
|
||||||
|
textarea.value = colorCode;
|
||||||
|
document.body.appendChild(textarea);
|
||||||
|
textarea.select();
|
||||||
|
try {
|
||||||
|
document.execCommand('copy');
|
||||||
|
console.log('Color code copied to clipboard');
|
||||||
|
copyButton.innerHTML = '<span>Copied!</span>';
|
||||||
|
setTimeout(() => {
|
||||||
|
copyButton.innerHTML = '<span>Copy</span>';
|
||||||
|
}, 2000);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Could not copy text: ', err);
|
||||||
|
}
|
||||||
|
document.body.removeChild(textarea);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
85
.hta_slug/dns-tools-get-a-record.php
Normal file
85
.hta_slug/dns-tools-get-a-record.php
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* DNS A Record Lookup API
|
||||||
|
* Endpoint: /dns-tools-get-a-record
|
||||||
|
* Method: POST
|
||||||
|
* Content-Type: application/json
|
||||||
|
*/
|
||||||
|
|
||||||
|
// -------------------------------
|
||||||
|
// Response headers
|
||||||
|
// -------------------------------
|
||||||
|
header('Content-Type: application/json; charset=utf-8');
|
||||||
|
|
||||||
|
// -------------------------------
|
||||||
|
// Allow only POST
|
||||||
|
// -------------------------------
|
||||||
|
if ($_SERVER['REQUEST_METHOD'] !== 'POST') {
|
||||||
|
http_response_code(405);
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'message' => 'Only POST method allowed'
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// -------------------------------
|
||||||
|
// Read JSON body
|
||||||
|
// -------------------------------
|
||||||
|
$rawInput = file_get_contents('php://input');
|
||||||
|
$data = json_decode($rawInput, true);
|
||||||
|
|
||||||
|
$domain = $data['domain'] ?? '';
|
||||||
|
|
||||||
|
// -------------------------------
|
||||||
|
// Domain validation
|
||||||
|
// -------------------------------
|
||||||
|
function validateDomain(string $domain): bool
|
||||||
|
{
|
||||||
|
return (bool) preg_match(
|
||||||
|
'/^(?!-)(?:[a-zA-Z0-9-]{1,63}\.)+[a-zA-Z]{2,}$/',
|
||||||
|
$domain
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$domain || !validateDomain($domain)) {
|
||||||
|
http_response_code(400);
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'message' => 'Invalid domain'
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// -------------------------------
|
||||||
|
// DNS lookup (NO shell_exec)
|
||||||
|
// -------------------------------
|
||||||
|
$records = dns_get_record($domain, DNS_A);
|
||||||
|
$ips = [];
|
||||||
|
|
||||||
|
if ($records !== false) {
|
||||||
|
foreach ($records as $record) {
|
||||||
|
if (!empty($record['ip'])) {
|
||||||
|
$ips[] = $record['ip'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// -------------------------------
|
||||||
|
// Response
|
||||||
|
// -------------------------------
|
||||||
|
if (empty($ips)) {
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'domain' => $domain,
|
||||||
|
'message' => 'No A records found',
|
||||||
|
'ips' => []
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
echo json_encode([
|
||||||
|
'success' => true,
|
||||||
|
'domain' => $domain,
|
||||||
|
'ips' => $ips
|
||||||
|
]);
|
||||||
71
.hta_slug/dns-tools-get-mx-record.php
Normal file
71
.hta_slug/dns-tools-get-mx-record.php
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* DNS MX Record Lookup API
|
||||||
|
* Endpoint: /dns-tools-get-mx-record
|
||||||
|
* Method: POST
|
||||||
|
* Content-Type: application/json
|
||||||
|
*/
|
||||||
|
|
||||||
|
header('Content-Type: application/json; charset=utf-8');
|
||||||
|
|
||||||
|
// Allow only POST
|
||||||
|
if ($_SERVER['REQUEST_METHOD'] !== 'POST') {
|
||||||
|
http_response_code(405);
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'message' => 'Only POST method allowed'
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read JSON input
|
||||||
|
$input = json_decode(file_get_contents('php://input'), true);
|
||||||
|
$domain = $input['domain'] ?? '';
|
||||||
|
|
||||||
|
// Domain validation
|
||||||
|
function validateDomain(string $domain): bool
|
||||||
|
{
|
||||||
|
return (bool) preg_match(
|
||||||
|
'/^(?!-)(?:[a-zA-Z0-9-]{1,63}\.)+[a-zA-Z]{2,}$/',
|
||||||
|
$domain
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$domain || !validateDomain($domain)) {
|
||||||
|
http_response_code(400);
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'message' => 'Invalid domain'
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch MX records
|
||||||
|
$records = dns_get_record($domain, DNS_MX);
|
||||||
|
$mxRecords = [];
|
||||||
|
|
||||||
|
if ($records !== false) {
|
||||||
|
foreach ($records as $record) {
|
||||||
|
$mxRecords[] = [
|
||||||
|
'host' => $record['target'] ?? '',
|
||||||
|
'priority' => $record['pri'] ?? null
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Response
|
||||||
|
if (empty($mxRecords)) {
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'domain' => $domain,
|
||||||
|
'message' => 'No MX records found',
|
||||||
|
'records' => []
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
echo json_encode([
|
||||||
|
'success' => true,
|
||||||
|
'domain' => $domain,
|
||||||
|
'records' => $mxRecords
|
||||||
|
]);
|
||||||
70
.hta_slug/dns-tools-get-ns-record.php
Normal file
70
.hta_slug/dns-tools-get-ns-record.php
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* DNS NS Record Lookup API
|
||||||
|
* Endpoint: /dns-tools-get-ns-record
|
||||||
|
* Method: POST
|
||||||
|
* Content-Type: application/json
|
||||||
|
*/
|
||||||
|
|
||||||
|
header('Content-Type: application/json; charset=utf-8');
|
||||||
|
|
||||||
|
// Allow only POST requests
|
||||||
|
if ($_SERVER['REQUEST_METHOD'] !== 'POST') {
|
||||||
|
http_response_code(405);
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'message' => 'Only POST method allowed'
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read JSON body
|
||||||
|
$input = json_decode(file_get_contents('php://input'), true);
|
||||||
|
$domain = $input['domain'] ?? '';
|
||||||
|
|
||||||
|
// Domain validation
|
||||||
|
function validateDomain(string $domain): bool
|
||||||
|
{
|
||||||
|
return (bool) preg_match(
|
||||||
|
'/^(?!-)(?:[a-zA-Z0-9-]{1,63}\.)+[a-zA-Z]{2,}$/',
|
||||||
|
$domain
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$domain || !validateDomain($domain)) {
|
||||||
|
http_response_code(400);
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'message' => 'Invalid domain'
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch NS records
|
||||||
|
$records = dns_get_record($domain, DNS_NS);
|
||||||
|
$nsRecords = [];
|
||||||
|
|
||||||
|
if ($records !== false) {
|
||||||
|
foreach ($records as $record) {
|
||||||
|
if (!empty($record['target'])) {
|
||||||
|
$nsRecords[] = $record['target'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Response
|
||||||
|
if (empty($nsRecords)) {
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'domain' => $domain,
|
||||||
|
'message' => 'No NS records found',
|
||||||
|
'records' => []
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
echo json_encode([
|
||||||
|
'success' => true,
|
||||||
|
'domain' => $domain,
|
||||||
|
'records' => $nsRecords
|
||||||
|
]);
|
||||||
6
.hta_slug/geoip_domain.php
Normal file
6
.hta_slug/geoip_domain.php
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
<?php
|
||||||
|
$isp = geoip_isp_by_name('www.example.com');
|
||||||
|
if ($isp) {
|
||||||
|
echo 'This host IP is from ISP: ' . $isp;
|
||||||
|
}
|
||||||
|
?>
|
||||||
@@ -1,6 +1,4 @@
|
|||||||
<?php
|
|
||||||
require_once('.htac_header.php');
|
|
||||||
?>
|
|
||||||
<section class="diZContainer diZmxAuto diZmy8">
|
<section class="diZContainer diZmxAuto diZmy8">
|
||||||
<h1 class="diZBorderBottom">JSON Formatter</h1>
|
<h1 class="diZBorderBottom">JSON Formatter</h1>
|
||||||
<div class="diZFlexRowCol">
|
<div class="diZFlexRowCol">
|
||||||
83
.hta_slug/location-ip-to-location.php
Normal file
83
.hta_slug/location-ip-to-location.php
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
<?php
|
||||||
|
// ❌ No header() here
|
||||||
|
|
||||||
|
require_once __DIR__ . '/../.hta_lib/maxmind-db-reader/autoload.php';
|
||||||
|
|
||||||
|
use MaxMind\Db\Reader;
|
||||||
|
|
||||||
|
function respond($status, $data = null, $error = null)
|
||||||
|
{
|
||||||
|
echo json_encode([
|
||||||
|
'success' => $status,
|
||||||
|
'data' => $data,
|
||||||
|
'error' => $error
|
||||||
|
], JSON_PRETTY_PRINT);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isset($_GET['ip']) || trim($_GET['ip']) === '') {
|
||||||
|
respond(false, null, 'IP parameter is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
$ip = trim($_GET['ip']);
|
||||||
|
|
||||||
|
if (!filter_var($ip, FILTER_VALIDATE_IP)) {
|
||||||
|
respond(false, null, 'Invalid IP address');
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$cityReader = new Reader(__DIR__ . '/../.hta_lib/data/GeoLite2-City.mmdb');
|
||||||
|
$asnReader = new Reader(__DIR__ . '/../.hta_lib/data/GeoLite2-ASN.mmdb');
|
||||||
|
|
||||||
|
$cityData = $cityReader->get($ip);
|
||||||
|
$asnData = $asnReader->get($ip);
|
||||||
|
|
||||||
|
$cityReader->close();
|
||||||
|
$asnReader->close();
|
||||||
|
|
||||||
|
$response = [
|
||||||
|
'ip' => $ip,
|
||||||
|
|
||||||
|
'continent' => [
|
||||||
|
'name' => $cityData['continent']['names']['en'] ?? null,
|
||||||
|
'code' => $cityData['continent']['code'] ?? null
|
||||||
|
],
|
||||||
|
|
||||||
|
'country' => [
|
||||||
|
'name' => $cityData['country']['names']['en'] ?? null,
|
||||||
|
'code' => $cityData['country']['iso_code'] ?? null
|
||||||
|
],
|
||||||
|
|
||||||
|
'state' => [
|
||||||
|
'name' => $cityData['subdivisions'][0]['names']['en'] ?? null,
|
||||||
|
'code' => $cityData['subdivisions'][0]['iso_code'] ?? null
|
||||||
|
],
|
||||||
|
|
||||||
|
'city' => $cityData['city']['names']['en'] ?? null,
|
||||||
|
|
||||||
|
'timezone' => $cityData['location']['time_zone'] ?? null,
|
||||||
|
|
||||||
|
'location' => [
|
||||||
|
'latitude' => $cityData['location']['latitude'] ?? null,
|
||||||
|
'longitude' => $cityData['location']['longitude'] ?? null
|
||||||
|
],
|
||||||
|
|
||||||
|
'postal_code' => $cityData['postal']['code'] ?? null,
|
||||||
|
|
||||||
|
'asn' => [
|
||||||
|
'number' => $asnData['autonomous_system_number'] ?? null,
|
||||||
|
'org' => $asnData['autonomous_system_organization'] ?? null
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|
||||||
|
$anycastASN = [15169, 13335, 36692];
|
||||||
|
$response['is_anycast'] = in_array($response['asn']['number'], $anycastASN);
|
||||||
|
$response['note'] = $response['is_anycast']
|
||||||
|
? 'Anycast IP detected. Location may vary.'
|
||||||
|
: 'IP-based location is approximate.';
|
||||||
|
|
||||||
|
respond(true, $response);
|
||||||
|
|
||||||
|
} catch (Exception $e) {
|
||||||
|
respond(false, null, $e->getMessage());
|
||||||
|
}
|
||||||
72
.hta_slug/mx-lookup.php
Normal file
72
.hta_slug/mx-lookup.php
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* Domain MX Information API
|
||||||
|
* Endpoint: /dns-tools-get-mx-info
|
||||||
|
* Method: POST
|
||||||
|
* Content-Type: application/json
|
||||||
|
*/
|
||||||
|
|
||||||
|
header('Content-Type: application/json; charset=utf-8');
|
||||||
|
|
||||||
|
// Allow only POST requests
|
||||||
|
if ($_SERVER['REQUEST_METHOD'] !== 'POST') {
|
||||||
|
http_response_code(405);
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'message' => 'Only POST method allowed'
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read JSON body
|
||||||
|
$input = json_decode(file_get_contents('php://input'), true);
|
||||||
|
$domain = $input['domain'] ?? '';
|
||||||
|
|
||||||
|
// Domain validation
|
||||||
|
function validateDomain(string $domain): bool
|
||||||
|
{
|
||||||
|
return (bool) filter_var(
|
||||||
|
$domain,
|
||||||
|
FILTER_VALIDATE_DOMAIN,
|
||||||
|
FILTER_FLAG_HOSTNAME
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$domain || !validateDomain($domain)) {
|
||||||
|
http_response_code(400);
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'message' => 'Invalid domain'
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fetch MX records (safe, no shell_exec)
|
||||||
|
$records = dns_get_record($domain, DNS_MX);
|
||||||
|
$mxRecords = [];
|
||||||
|
|
||||||
|
if ($records !== false) {
|
||||||
|
foreach ($records as $record) {
|
||||||
|
$mxRecords[] = [
|
||||||
|
'mail_server' => $record['target'] ?? '',
|
||||||
|
'priority' => $record['pri'] ?? null
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Response
|
||||||
|
if (empty($mxRecords)) {
|
||||||
|
echo json_encode([
|
||||||
|
'success' => false,
|
||||||
|
'domain' => $domain,
|
||||||
|
'message' => 'No MX records found',
|
||||||
|
'records' => []
|
||||||
|
]);
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
echo json_encode([
|
||||||
|
'success' => true,
|
||||||
|
'domain' => $domain,
|
||||||
|
'records' => $mxRecords
|
||||||
|
]);
|
||||||
@@ -9,7 +9,7 @@
|
|||||||
<button class="diZmxAuto" type="submit" id="generatButton" onclick="displayQR();"><span>Generate QR Code</span></button>
|
<button class="diZmxAuto" type="submit" id="generatButton" onclick="displayQR();"><span>Generate QR Code</span></button>
|
||||||
</form>
|
</form>
|
||||||
<div class="diZFlexColumn diZJustifyCenter diZItemsCenter">
|
<div class="diZFlexColumn diZJustifyCenter diZItemsCenter">
|
||||||
<div class="diZDisplayNone" id="qrcode"></div>
|
<div class="diZDisplayNone" id="qrcode" style="width: fit-content"></div>
|
||||||
<div class="diZFlexBetween">
|
<div class="diZFlexBetween">
|
||||||
<button class="" id="download-button"><span>Download</span></button>
|
<button class="" id="download-button"><span>Download</span></button>
|
||||||
<button onclick="getNewCode();" id="neQRButton"><span>New QR Code</span></button>
|
<button onclick="getNewCode();" id="neQRButton"><span>New QR Code</span></button>
|
||||||
128
.hta_slug/toml-to-json.php
Normal file
128
.hta_slug/toml-to-json.php
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
<div class="tomljson-container">
|
||||||
|
<div class="tomljson-box">
|
||||||
|
<h2 class="tomljson-title">TOML to JSON Converter</h2>
|
||||||
|
<textarea cols="45" rows="8" class="tomljson-input" placeholder="Paste TOML here..."></textarea>
|
||||||
|
<div class="tomljson-button-group">
|
||||||
|
<button class="tomljson-convert-btn">Convert</button>
|
||||||
|
<button class="tomljson-copy-btn">Copy JSON</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h3 class="tomljson-output-title">JSON Output:</h3>
|
||||||
|
<pre class="tomljson-output"></pre>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="module">
|
||||||
|
import * as TOML from 'https://cdn.jsdelivr.net/npm/toml@3.0.0/+esm';
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", () => {
|
||||||
|
document.querySelector(".tomljson-convert-btn").addEventListener("click", () => {
|
||||||
|
try {
|
||||||
|
const tomlData = document.querySelector(".tomljson-input").value;
|
||||||
|
const jsonData = TOML.parse(tomlData);
|
||||||
|
document.querySelector(".tomljson-output").textContent = JSON.stringify(jsonData, null, 2);
|
||||||
|
} catch (error) {
|
||||||
|
document.querySelector(".tomljson-output").textContent = "Error: " + error.message;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
document.querySelector(".tomljson-copy-btn").addEventListener("click", () => {
|
||||||
|
const jsonOutput = document.querySelector(".tomljson-output").textContent;
|
||||||
|
navigator.clipboard.writeText(jsonOutput).then(() => {
|
||||||
|
document.querySelector(".tomljson-copy-btn").textContent = "Copied!";
|
||||||
|
setTimeout(() => document.querySelector(".tomljson-copy-btn").textContent = "Copy JSON", 2000);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
|
||||||
|
.tomljson-container {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
width: 100%;
|
||||||
|
margin: 100px 0
|
||||||
|
}
|
||||||
|
|
||||||
|
.tomljson-box {
|
||||||
|
background: #2c2c2c;
|
||||||
|
padding: 25px;
|
||||||
|
border-radius: 10px;
|
||||||
|
box-shadow: 0 8px 16px rgba(0, 0, 0, 0.3);
|
||||||
|
max-width: 450px;
|
||||||
|
width: 100%;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tomljson-title {
|
||||||
|
font-size: 22px;
|
||||||
|
font-weight: bold;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tomljson-input {
|
||||||
|
/* width: 100%; */
|
||||||
|
background: #3c3c3c;
|
||||||
|
color: #ffffff;
|
||||||
|
border: 1px solid #555;
|
||||||
|
border-radius: 5px;
|
||||||
|
/* padding: 12px; */
|
||||||
|
font-size: 14px;
|
||||||
|
resize: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tomljson-button-group {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 10px;
|
||||||
|
margin-top: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tomljson-convert-btn, .tomljson-copy-btn {
|
||||||
|
background: #007bff;
|
||||||
|
border: none;
|
||||||
|
padding: 10px 18px;
|
||||||
|
border-radius: 5px;
|
||||||
|
color: #ffffff;
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 15px;
|
||||||
|
transition: 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tomljson-copy-btn {
|
||||||
|
background: #28a745;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tomljson-convert-btn:hover {
|
||||||
|
background: #0056b3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tomljson-copy-btn:hover {
|
||||||
|
background: #218838;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tomljson-output-title {
|
||||||
|
font-size: 18px;
|
||||||
|
margin-top: 20px;
|
||||||
|
font-weight: bold;
|
||||||
|
text-align: center;
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tomljson-output {
|
||||||
|
background: #3c3c3c;
|
||||||
|
padding: 12px;
|
||||||
|
border-radius: 5px;
|
||||||
|
text-align: left;
|
||||||
|
font-size: 14px;
|
||||||
|
height: 160px;
|
||||||
|
overflow: auto;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
border: 1px solid #555;
|
||||||
|
color: #ffffff;
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
</style>
|
||||||
17
.hta_slug/what-is-my-ip.php
Normal file
17
.hta_slug/what-is-my-ip.php
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
<?php
|
||||||
|
$ipv4 = $_SERVER['REMOTE_ADDR'];
|
||||||
|
|
||||||
|
// Get the IPv6 address of the client if available
|
||||||
|
$ipv6 = $_SERVER['REMOTE_ADDR'];
|
||||||
|
if (strpos($ipv6, ":") !== false) {
|
||||||
|
// IPv6 address detected
|
||||||
|
$ipv6 = explode("%", $ipv6)[0]; // Remove interface suffix, if any
|
||||||
|
} else {
|
||||||
|
// No IPv6 address detected
|
||||||
|
$ipv6 = "IPv6 not available";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Display IPv4 and IPv6 addresses
|
||||||
|
echo "IPv4 Address: " . $ipv4 . "<br>";
|
||||||
|
echo "IPv6 Address: " . $ipv6 . "<br>";
|
||||||
|
?>
|
||||||
74
.hta_slug/who-is.php
Normal file
74
.hta_slug/who-is.php
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
<section class="diZContainer diZmxAuto">
|
||||||
|
<h2 class="diZBorderBottom">Ultimate Domain & IP Lookup Tool</h2>
|
||||||
|
<p class="diZmb20">This tool allows you to perform a comprehensive lookup of domain names or IP addresses, providing detailed WHOIS information.</p>
|
||||||
|
<form method="post" class="diZToolsSection diZmt4 diZmb4 diZBorderRadius diZPadding5px ">
|
||||||
|
<div class="diZFlexRowCol diZJustifyCenter diZItemsCenter ">
|
||||||
|
<input class="diZmr2 diZw70" placeholder="Domain" name="domain" type="text" />
|
||||||
|
<p class="diZmr2">OR</p>
|
||||||
|
<input class="diZmr2 diZw70" placeholder="IP Address" name="ip" type="text" />
|
||||||
|
<button type="submit"><span>Check</span></button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<?php
|
||||||
|
if(isset($_POST['domain']) && $_POST['domain']){
|
||||||
|
function validateDomain($domain) {
|
||||||
|
$regex = "/^(?!\-)(?:[a-zA-Z0-9\-]{0,61}[a-zA-Z0-9]\.)+[a-zA-Z]{2,}$/";
|
||||||
|
if (preg_match($regex, $domain)) {
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$domain = isset($_POST['domain']) ? $_POST['domain'] : '';
|
||||||
|
if ($domain && validateDomain($domain)) {
|
||||||
|
$command = 'whois '.$_POST['domain'];
|
||||||
|
$escaped_command = escapeshellcmd($command);
|
||||||
|
$output = shell_exec($escaped_command);
|
||||||
|
echo '<div class="diZContainer diZmxAuto diZPadding5px"><pre class="diZTextJustify" style="width: fit-content;"> ',$output, '</pre> <br><br></div>';
|
||||||
|
} else {
|
||||||
|
echo "Invalid domain.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(isset($_POST['ip']) && $_POST['ip']){
|
||||||
|
function validatePublicIp($ip) {
|
||||||
|
if (filter_var($ip, FILTER_VALIDATE_IP)) {
|
||||||
|
if (filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
$ip = isset($_POST['ip']) ? $_POST['ip'] : '';
|
||||||
|
if ($ip && validatePublicIp($ip)) {
|
||||||
|
$command = 'whois '.$_POST['ip'];
|
||||||
|
$escaped_command = escapeshellcmd($command);
|
||||||
|
$output = shell_exec($escaped_command);
|
||||||
|
echo '<div class="diZContainer diZmxAuto diZPadding5px"><pre> ',$output, '</pre> <br><br></div>';
|
||||||
|
} else {
|
||||||
|
echo "Invalid IP address.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
?>
|
||||||
|
<div>
|
||||||
|
<h3>Usage:</h3>
|
||||||
|
<ul>
|
||||||
|
<li>Enter a domain name or an IP address into the respective input field.</li>
|
||||||
|
<li>Click the "Check" button to retrieve WHOIS information.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<h3>Features:</h3>
|
||||||
|
<ul>
|
||||||
|
<li>Supports lookup for both domain names and IP addresses.</li>
|
||||||
|
<li>Displays WHOIS data including registration details and administrative contacts.</li>
|
||||||
|
<li>Secure input validation to ensure accurate results.</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<h3>Example Use Cases:</h3>
|
||||||
|
<ul>
|
||||||
|
<li>Investigate ownership and registration details of a domain.</li>
|
||||||
|
<li>Check the WHOIS information of an IP address for network troubleshooting.</li>
|
||||||
|
<li>Verify domain availability and registration status.</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
166
.hta_slug/xml-to-json.php
Normal file
166
.hta_slug/xml-to-json.php
Normal file
@@ -0,0 +1,166 @@
|
|||||||
|
<div class="xmljson-container">
|
||||||
|
<div class="xmljson-box">
|
||||||
|
<h2 class="xmljson-title">XML to JSON Converter</h2>
|
||||||
|
<textarea cols="45" rows="8" class="xmljson-input" placeholder="Paste XML here..."></textarea>
|
||||||
|
<div class="xmljson-button-group">
|
||||||
|
<button class="xmljson-convert-btn">Convert</button>
|
||||||
|
<button class="xmljson-copy-btn">Copy JSON</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<h3 class="xmljson-output-title">JSON Output:</h3>
|
||||||
|
<pre class="xmljson-output"></pre>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
function xmlToJson(xml) {
|
||||||
|
let obj = {};
|
||||||
|
if (xml.nodeType === 1) { // Element
|
||||||
|
if (xml.attributes.length > 0) {
|
||||||
|
obj["@attributes"] = {};
|
||||||
|
for (let j = 0; j < xml.attributes.length; j++) {
|
||||||
|
let attribute = xml.attributes.item(j);
|
||||||
|
obj["@attributes"][attribute.nodeName] = attribute.nodeValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (xml.nodeType === 3) { // Text
|
||||||
|
return xml.nodeValue.trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (xml.hasChildNodes()) {
|
||||||
|
for (let i = 0; i < xml.childNodes.length; i++) {
|
||||||
|
let item = xml.childNodes.item(i);
|
||||||
|
let nodeName = item.nodeName;
|
||||||
|
let jsonNode = xmlToJson(item);
|
||||||
|
|
||||||
|
if (jsonNode !== "") {
|
||||||
|
if (typeof obj[nodeName] === "undefined") {
|
||||||
|
obj[nodeName] = jsonNode;
|
||||||
|
} else {
|
||||||
|
if (!Array.isArray(obj[nodeName])) {
|
||||||
|
obj[nodeName] = [obj[nodeName]];
|
||||||
|
}
|
||||||
|
obj[nodeName].push(jsonNode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", () => {
|
||||||
|
document.querySelector(".xmljson-convert-btn").addEventListener("click", () => {
|
||||||
|
try {
|
||||||
|
const xmlText = document.querySelector(".xmljson-input").value;
|
||||||
|
const parser = new DOMParser();
|
||||||
|
const xmlDoc = parser.parseFromString(xmlText, "text/xml");
|
||||||
|
|
||||||
|
if (xmlDoc.getElementsByTagName("parsererror").length) {
|
||||||
|
throw new Error("Invalid XML format!");
|
||||||
|
}
|
||||||
|
|
||||||
|
let jsonData = xmlToJson(xmlDoc.documentElement);
|
||||||
|
document.querySelector(".xmljson-output").textContent = JSON.stringify(jsonData, null, 2);
|
||||||
|
} catch (error) {
|
||||||
|
document.querySelector(".xmljson-output").textContent = "Error: " + error.message;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
document.querySelector(".xmljson-copy-btn").addEventListener("click", () => {
|
||||||
|
const jsonOutput = document.querySelector(".xmljson-output").textContent;
|
||||||
|
navigator.clipboard.writeText(jsonOutput).then(() => {
|
||||||
|
document.querySelector(".xmljson-copy-btn").textContent = "Copied!";
|
||||||
|
setTimeout(() => document.querySelector(".xmljson-copy-btn").textContent = "Copy JSON", 2000);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.xmljson-container {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
width: 100%;
|
||||||
|
margin: 100px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.xmljson-box {
|
||||||
|
background: #2c2c2c;
|
||||||
|
padding: 25px;
|
||||||
|
border-radius: 10px;
|
||||||
|
box-shadow: 0 8px 16px rgba(0, 0, 0, 0.3);
|
||||||
|
max-width: 450px;
|
||||||
|
width: 100%;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.xmljson-title {
|
||||||
|
font-size: 22px;
|
||||||
|
font-weight: bold;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.xmljson-input {
|
||||||
|
background: #3c3c3c;
|
||||||
|
color: #ffffff;
|
||||||
|
border: 1px solid #555;
|
||||||
|
border-radius: 5px;
|
||||||
|
font-size: 14px;
|
||||||
|
resize: none;
|
||||||
|
padding: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.xmljson-button-group {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 10px;
|
||||||
|
margin-top: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.xmljson-convert-btn, .xmljson-copy-btn {
|
||||||
|
background: #007bff;
|
||||||
|
border: none;
|
||||||
|
padding: 10px 18px;
|
||||||
|
border-radius: 5px;
|
||||||
|
color: #ffffff;
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 15px;
|
||||||
|
transition: 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.xmljson-copy-btn {
|
||||||
|
background: #28a745;
|
||||||
|
}
|
||||||
|
|
||||||
|
.xmljson-convert-btn:hover {
|
||||||
|
background: #0056b3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.xmljson-copy-btn:hover {
|
||||||
|
background: #218838;
|
||||||
|
}
|
||||||
|
|
||||||
|
.xmljson-output-title {
|
||||||
|
font-size: 18px;
|
||||||
|
margin-top: 20px;
|
||||||
|
font-weight: bold;
|
||||||
|
text-align: center;
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.xmljson-output {
|
||||||
|
background: #3c3c3c;
|
||||||
|
padding: 12px;
|
||||||
|
border-radius: 5px;
|
||||||
|
text-align: left;
|
||||||
|
font-size: 14px;
|
||||||
|
height: 160px;
|
||||||
|
overflow: auto;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
border: 1px solid #555;
|
||||||
|
color: #ffffff;
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
8
.htaccess
Normal file
8
.htaccess
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
RewriteEngine On
|
||||||
|
#RewriteCond %{HTTPS} !=on
|
||||||
|
#RewriteRule ^(.*)$ https://%{HTTP_HOST}%{REQUEST_URI} [L,R=301,NE]
|
||||||
|
RewriteCond %{HTTP_HOST} ^www\.(.+)$ [NC]
|
||||||
|
RewriteRule ^(.*)$ https://%1/$1 [R=301,L]
|
||||||
|
RewriteCond %{REQUEST_FILENAME} !-f
|
||||||
|
RewriteCond %{REQUEST_FILENAME} !-d
|
||||||
|
RewriteRule ^(.*) index.php
|
||||||
BIN
MaxMind-DB-Reader-php-main.zip
Normal file
BIN
MaxMind-DB-Reader-php-main.zip
Normal file
Binary file not shown.
@@ -1,34 +0,0 @@
|
|||||||
<section class="diZContainer diZmxAuto">
|
|
||||||
<h2 class="diZBorderBottom">Ultimate Domain & IP Lookup Tool</h2>
|
|
||||||
</section>
|
|
||||||
<!-- <p class="diZTextJustify">Discover detailed information about any domain and IP address with ease using Who-Is. Our tool provides instant access to essential data, including ownership, registration details, and more, all in a user-friendly interface designed for efficiency and accuracy.</p> -->
|
|
||||||
|
|
||||||
<form method="post" class="diZToolsSection diZmt4 diZmb4 diZBorderRadius diZPadding5px">
|
|
||||||
<div class="diZFlexRowCol diZJustifyCenter diZItemsCenter">
|
|
||||||
<input class="diZmr2 diZw70" placeholder="Domain" name="domain" type="text" />
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
<?php
|
|
||||||
if(isset($_POST['domain']) && $_POST['domain']){
|
|
||||||
function validateDomain($domain) {
|
|
||||||
$regex = "/^(?!\-)(?:[a-zA-Z0-9\-]{0,61}[a-zA-Z0-9]\.)+[a-zA-Z]{2,}$/";
|
|
||||||
if (preg_match($regex, $domain)) {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$domain = isset($_POST['domain']) ? $_POST['domain'] : '';
|
|
||||||
if ($domain && validateDomain($domain)) {
|
|
||||||
$command = 'dig '.$_POST['domain'].' A +short';
|
|
||||||
$escaped_command = escapeshellcmd($command);
|
|
||||||
$output = shell_exec($escaped_command);
|
|
||||||
echo '<div class="diZContainer diZmxAuto diZPadding5px"><pre class="diZTextJustify" style="width: fit-content;"> ',$output, '</pre> <br><br></div>';
|
|
||||||
} else {
|
|
||||||
echo "Invalid domain.";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
?>
|
|
||||||
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
<section class="diZContainer diZmxAuto">
|
|
||||||
<h2 class="diZBorderBottom">Ultimate Domain & IP Lookup Tool</h2>
|
|
||||||
</section>
|
|
||||||
<!-- <p class="diZTextJustify">Discover detailed information about any domain and IP address with ease using Who-Is. Our tool provides instant access to essential data, including ownership, registration details, and more, all in a user-friendly interface designed for efficiency and accuracy.</p> -->
|
|
||||||
|
|
||||||
<form method="post" class="diZToolsSection diZmt4 diZmb4 diZBorderRadius diZPadding5px">
|
|
||||||
<div class="diZFlexRowCol diZJustifyCenter diZItemsCenter">
|
|
||||||
<input class="diZmr2 diZw70" placeholder="Domain" name="domain" type="text" />
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
<?php
|
|
||||||
if(isset($_POST['domain']) && $_POST['domain']){
|
|
||||||
function validateDomain($domain) {
|
|
||||||
$regex = "/^(?!\-)(?:[a-zA-Z0-9\-]{0,61}[a-zA-Z0-9]\.)+[a-zA-Z]{2,}$/";
|
|
||||||
if (preg_match($regex, $domain)) {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$domain = isset($_POST['domain']) ? $_POST['domain'] : '';
|
|
||||||
if ($domain && validateDomain($domain)) {
|
|
||||||
$command = 'dig '.$_POST['domain'].' MX +short';
|
|
||||||
$escaped_command = escapeshellcmd($command);
|
|
||||||
$output = shell_exec($escaped_command);
|
|
||||||
echo '<div class="diZContainer diZmxAuto diZPadding5px"><pre class="diZTextJustify" style="width: fit-content;"> ',$output, '</pre> <br><br></div>';
|
|
||||||
} else {
|
|
||||||
echo "Invalid domain.";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
?>
|
|
||||||
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
<section class="diZContainer diZmxAuto">
|
|
||||||
<h2 class="diZBorderBottom">Ultimate Domain & IP Lookup Tool</h2>
|
|
||||||
</section>
|
|
||||||
<!-- <p class="diZTextJustify">Discover detailed information about any domain and IP address with ease using Who-Is. Our tool provides instant access to essential data, including ownership, registration details, and more, all in a user-friendly interface designed for efficiency and accuracy.</p> -->
|
|
||||||
|
|
||||||
<form method="post" class="diZToolsSection diZmt4 diZmb4 diZBorderRadius diZPadding5px">
|
|
||||||
<div class="diZFlexRowCol diZJustifyCenter diZItemsCenter">
|
|
||||||
<input class="diZmr2 diZw70" placeholder="Domain" name="domain" type="text" />
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
<?php
|
|
||||||
if(isset($_POST['domain']) && $_POST['domain']){
|
|
||||||
function validateDomain($domain) {
|
|
||||||
$regex = "/^(?!\-)(?:[a-zA-Z0-9\-]{0,61}[a-zA-Z0-9]\.)+[a-zA-Z]{2,}$/";
|
|
||||||
if (preg_match($regex, $domain)) {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$domain = isset($_POST['domain']) ? $_POST['domain'] : '';
|
|
||||||
if ($domain && validateDomain($domain)) {
|
|
||||||
$command = 'dig '.$_POST['domain'].' NS +short';
|
|
||||||
$escaped_command = escapeshellcmd($command);
|
|
||||||
$output = shell_exec($escaped_command);
|
|
||||||
echo '<div class="diZContainer diZmxAuto diZPadding5px"><pre class="diZTextJustify" style="width: fit-content;"> ',$output, '</pre> <br><br></div>';
|
|
||||||
} else {
|
|
||||||
echo "Invalid domain.";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
?>
|
|
||||||
|
|
||||||
24
index.php
Normal file
24
index.php
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
<?php
|
||||||
|
ini_set('display_errors', 1);
|
||||||
|
ini_set('display_startup_errors', 1);
|
||||||
|
error_reporting(E_ALL);
|
||||||
|
|
||||||
|
$ROOT = __DIR__; // /var/www/html
|
||||||
|
|
||||||
|
require_once $ROOT.'/.hta_config/conf.php';
|
||||||
|
require_once $ROOT.'/.hta_config/var.php';
|
||||||
|
require_once $ROOT.'/.hta_slug/_header.php';
|
||||||
|
require_once $ROOT.'/.hta_slug/_nav.php';
|
||||||
|
|
||||||
|
$url = explode('/', trim($_SERVER['REQUEST_URI'], '/'));
|
||||||
|
$slug = $url[0] ?? '';
|
||||||
|
|
||||||
|
if ($slug === '') {
|
||||||
|
require_once $ROOT.'/.hta_slug/_home.php';
|
||||||
|
} elseif (file_exists($ROOT.'/.hta_slug/'.$slug.'.php')) {
|
||||||
|
require_once $ROOT.'/.hta_slug/'.$slug.'.php';
|
||||||
|
} else {
|
||||||
|
require_once $ROOT.'/.hta_slug/_404.php';
|
||||||
|
}
|
||||||
|
|
||||||
|
require_once $ROOT.'/.hta_slug/_footer.php';
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
<?php
|
|
||||||
require '.hta_lib/vendors/maxmind-db-reader/vendor/autoload.php';
|
|
||||||
|
|
||||||
use GeoIp2\Database\Reader;
|
|
||||||
|
|
||||||
if (isset($_GET['ip'])) {
|
|
||||||
try {
|
|
||||||
$reader = new Reader('.hta_lib/data/GeoLite2-City.mmdb');
|
|
||||||
|
|
||||||
$records = $reader->city($_GET['ip']);
|
|
||||||
|
|
||||||
$city = $records->city->name;
|
|
||||||
|
|
||||||
echo json_encode(['city' => $city]);
|
|
||||||
} catch (Exception $e) {
|
|
||||||
echo json_encode(['error' => $e->getMessage()]);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
echo json_encode(['error' => 'Invalid parameters']);
|
|
||||||
}
|
|
||||||
?>
|
|
||||||
@@ -1,36 +0,0 @@
|
|||||||
<section class="diZContainer diZmxAuto">
|
|
||||||
<h2 class="diZBorderBottom">Domain MX Information Viewer</h2>
|
|
||||||
<p class="diZTextJustify diZmb4">The Domain MX Information Viewer is a powerful and easy-to-use tool designed to help IT professionals and website administrators retrieve detailed mail exchange (MX) records for any given domain. Simply enter a domain name, and this tool will perform a DNS lookup to fetch and display the MX records, providing insights into the domain's email routing information.</p>
|
|
||||||
<form method="post" class="diZToolsSection diZmt4 diZmb4 diZBorderRadius diZPadding5px ">
|
|
||||||
<div class="diZFlexRowCol diZJustifyCenter diZItemsCenter">
|
|
||||||
<input class="diZmr2 diZw70" placeholder="Domain Name" id="Domain" name="domain" type="text" required />
|
|
||||||
<button class="diZmr2" type="submit"><span>Check</span></button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
<?php
|
|
||||||
if (isset($_POST['domain']) && $_POST['domain']) {
|
|
||||||
function validateDomain($domain) {
|
|
||||||
return filter_var($domain, FILTER_VALIDATE_DOMAIN, FILTER_FLAG_HOSTNAME);
|
|
||||||
}
|
|
||||||
|
|
||||||
$domain = $_POST['domain'];
|
|
||||||
if (validateDomain($domain)) {
|
|
||||||
$command = escapeshellcmd('dig ' . $domain . ' MX');
|
|
||||||
$output = shell_exec($command);
|
|
||||||
echo '<div class="diZContainer diZmxAuto diZPadding5px"><pre>' . htmlspecialchars($output, ENT_QUOTES, 'UTF-8') . '</pre></div>';
|
|
||||||
} else {
|
|
||||||
echo '<div class="diZContainer diZmxAuto diZPadding5px">Invalid domain.</div>';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
?>
|
|
||||||
<div class="diZmy20">
|
|
||||||
<h3>Key Features:</h3>
|
|
||||||
<ul>
|
|
||||||
<li><strong>Accurate Domain Validation:</strong> Ensures the entered domain name is valid and properly formatted.</li>
|
|
||||||
<li><strong>Secure Processing:</strong> Utilizes command sanitization and output encoding to prevent security vulnerabilities like command injection and cross-site scripting (XSS).</li>
|
|
||||||
<li><strong>Instant Results:</strong> Quickly retrieves and displays MX records for the specified domain.</li>
|
|
||||||
<li><strong>User-Friendly Interface:</strong> Simplified input form for easy use without the need for additional styling.</li>
|
|
||||||
</ul>
|
|
||||||
<p>This tool is ideal for those seeking quick and reliable MX record information to manage and troubleshoot email delivery issues effectively.</p>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
<?php
|
|
||||||
// Get the IPv4 address of the client
|
|
||||||
// echo $_SERVER['REMOTE_ADDR'];
|
|
||||||
$ipv4 = $_SERVER['REMOTE_ADDR'];
|
|
||||||
|
|
||||||
// Get the IPv6 address of the client if available
|
|
||||||
$ipv6 = $_SERVER['REMOTE_ADDR'];
|
|
||||||
if (strpos($ipv6, ":") !== false) {
|
|
||||||
// IPv6 address detected
|
|
||||||
$ipv6 = explode("%", $ipv6)[0]; // Remove interface suffix, if any
|
|
||||||
} else {
|
|
||||||
// No IPv6 address detected
|
|
||||||
$ipv6 = "IPv6 not available";
|
|
||||||
}
|
|
||||||
|
|
||||||
// Display IPv4 and IPv6 addresses
|
|
||||||
echo "IPv4 Address: " . $ipv4 . "<br>";
|
|
||||||
echo "IPv6 Address: " . $ipv6 . "<br>";
|
|
||||||
?>
|
|
||||||
56
who-is.php
56
who-is.php
@@ -1,56 +0,0 @@
|
|||||||
<section class="diZContainer diZmxAuto">
|
|
||||||
<h2 class="diZBorderBottom">Ultimate Domain & IP Lookup Tool</h2>
|
|
||||||
</section>
|
|
||||||
<!-- <p class="diZTextJustify">Discover detailed information about any domain and IP address with ease using Who-Is. Our tool provides instant access to essential data, including ownership, registration details, and more, all in a user-friendly interface designed for efficiency and accuracy.</p> -->
|
|
||||||
|
|
||||||
<form method="post" class="diZToolsSection diZmt4 diZmb4 diZBorderRadius diZPadding5px">
|
|
||||||
<div class="diZFlexRowCol diZJustifyCenter diZItemsCenter">
|
|
||||||
<input class="diZmr2 diZw70" placeholder="Domain" name="domain" type="text" />
|
|
||||||
<p class="diZmr2">OR</p>
|
|
||||||
<input class="diZmr2 diZw70" placeholder="IP Address" name="ip" type="text" />
|
|
||||||
<input class="diZmr2" type="submit" value="Check" />
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
<?php
|
|
||||||
if(isset($_POST['domain']) && $_POST['domain']){
|
|
||||||
function validateDomain($domain) {
|
|
||||||
$regex = "/^(?!\-)(?:[a-zA-Z0-9\-]{0,61}[a-zA-Z0-9]\.)+[a-zA-Z]{2,}$/";
|
|
||||||
if (preg_match($regex, $domain)) {
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$domain = isset($_POST['domain']) ? $_POST['domain'] : '';
|
|
||||||
if ($domain && validateDomain($domain)) {
|
|
||||||
$command = 'whois '.$_POST['domain'];
|
|
||||||
$escaped_command = escapeshellcmd($command);
|
|
||||||
$output = shell_exec($escaped_command);
|
|
||||||
echo '<div class="diZContainer diZmxAuto diZPadding5px"><pre class="diZTextJustify" style="width: fit-content;"> ',$output, '</pre> <br><br></div>';
|
|
||||||
} else {
|
|
||||||
echo "Invalid domain.";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if(isset($_POST['ip']) && $_POST['ip']){
|
|
||||||
function validatePublicIp($ip) {
|
|
||||||
if (filter_var($ip, FILTER_VALIDATE_IP)) {
|
|
||||||
if (filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
$ip = isset($_POST['ip']) ? $_POST['ip'] : '';
|
|
||||||
if ($ip && validatePublicIp($ip)) {
|
|
||||||
$command = 'whois '.$_POST['ip'];
|
|
||||||
$escaped_command = escapeshellcmd($command);
|
|
||||||
$output = shell_exec($escaped_command);
|
|
||||||
echo '<div class="diZContainer diZmxAuto diZPadding5px"><pre> ',$output, '</pre> <br><br></div>';
|
|
||||||
} else {
|
|
||||||
echo "Invalid IP address.";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
?>
|
|
||||||
|
|
||||||
Reference in New Issue
Block a user