diff --git a/composer.json b/composer.json index fb3fd28b..be4379b9 100644 --- a/composer.json +++ b/composer.json @@ -49,12 +49,10 @@ "phpoffice/phpspreadsheet": "^1.14", "riverslei/payment": "^5.1", "lizhichao/word": "^2.1", - "joypack/tencent-map": "^1.0", "obs/esdk-obs-php": "^3.21", "ucloud/ufile-php-sdk": "^1.0", "swoole/ide-helper": "^4.8", "alibabacloud/dysmsapi-20170525": "2.0.9", - "fastknife/ajcaptcha": "^1.1", "vlucas/phpdotenv": "^5.3", "overtrue/pinyin": "4.1.0", "jpush/jpush": "^3.6", diff --git a/extend/fastknife/ajcaptcha/.gitignore b/extend/fastknife/ajcaptcha/.gitignore new file mode 100644 index 00000000..e69de29b diff --git a/extend/fastknife/ajcaptcha/LICENSE b/extend/fastknife/ajcaptcha/LICENSE new file mode 100644 index 00000000..94a9ed02 --- /dev/null +++ b/extend/fastknife/ajcaptcha/LICENSE @@ -0,0 +1,674 @@ + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright (C) + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + Copyright (C) + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +. diff --git a/extend/fastknife/ajcaptcha/changelog.md b/extend/fastknife/ajcaptcha/changelog.md new file mode 100644 index 00000000..b705626c --- /dev/null +++ b/extend/fastknife/ajcaptcha/changelog.md @@ -0,0 +1,26 @@ +# aj-captcha迭代文档 +## v1.1.2 2021-12-15 +* server 增加`verificationByEncryptCode` 方法,兼容前端二次验证`captchaVerification`值 + +## v1.1.1 2021-12-13 +* 去除多余代码、注释 + +## v1.1.0 2021-12-9 +* 重构工厂类与领域层 +* 永久缓存图片像素值,增加响应性能 + +## v1.0.7 2021-11-24 +* 增强缓存冗错 +* 自定义缓存配置 + +## v1.0.3 2021-10-11 +* 新增二次验证 + + +## v1.0.0 2021-7-16 +* 初步实现滑动验证码与文字点选验证码 + + + + + diff --git a/extend/fastknife/ajcaptcha/composer.json b/extend/fastknife/ajcaptcha/composer.json new file mode 100644 index 00000000..5b66e8a0 --- /dev/null +++ b/extend/fastknife/ajcaptcha/composer.json @@ -0,0 +1,26 @@ +{ + "name": "fastknife/ajcaptcha", + "description": "This is a behavior verification code PHP back-end implementation package", + "type": "library", + "authors": [ + { + "name": "bruce", + "email": "2777314125@qq.com" + } + ], + "require": { + "php": ">=7.1", + "ext-gd": "*", + "ext-openssl": "*", + "intervention/image": "^2.5", + "ext-iconv": "*", + "ext-json": "*" + }, + "minimum-stability": "stable", + "license": "GPL-3.0-only", + "autoload": { + "psr-4": { + "Fastknife\\": "src" + } + } +} diff --git a/extend/fastknife/ajcaptcha/demo.md b/extend/fastknife/ajcaptcha/demo.md new file mode 100644 index 00000000..b40a0393 --- /dev/null +++ b/extend/fastknife/ajcaptcha/demo.md @@ -0,0 +1,239 @@ +### 范例 + +详情请查看test目录的PHP源码 + +#### 配置说明 + +```php +return [ + 'font_file' => '', //自定义字体包路径, 不填使用默认值 + //文字验证码 + 'click_world' => [ + 'backgrounds' => [] + ], + //滑动验证码 + 'block_puzzle' => [ + 'backgrounds' => [], //背景图片路径, 不填使用默认值 + 'templates' => [], //模板图 + + 'offset' => 10, //容错偏移量 + + 'is_cache_pixel' => true, //是否开启缓存图片像素值,开启后能提升服务端响应性能(但要注意更换图片时,需要清除缓存) + ], + //水印 + 'watermark' => [ + 'fontsize' => 12, + 'color' => '#ffffff', + 'text' => '我的水印' + ], + 'cache' => [ + 'constructor' => \Fastknife\Utils\CacheUtils::class,//若您使用了框架,并且想使用类似于redis这样的缓存驱动,则应换成框架的中的缓存驱动 + 'method' => [ + // 遵守PSR-16规范不需要设置此项(tp6, laravel,hyperf)。如tp5就不支持(tp5缓存方法是rm,所以要配置为"delete" => "rm"), + 'get' => 'get', //获取 + 'set' => 'set', //设置 + 'delete' => 'delete',//删除 + 'has' => 'has' //key是否存在 + ], + 'options' => [ + //如果您依然使用\Fastknife\Utils\CacheUtils做为您的缓存驱动,那么您可以自定义缓存配置。 + 'expire' => 300,//缓存有效期 (默认为0 表示永久缓存) + 'prefix' => '', //缓存前缀 + 'path' => '', //缓存目录 + 'serialize' => [], //缓存序列化和反序列化方法 + ] + ] +]; +``` +##### 缓存配置 + +> config.cache.constructor类型为string|array|function 使用以访问回调的方式获得缓存实例; + ++ laravel 配置: + + ``` + 'constructor' => [Illuminate\Support\Facades\Cache::class, 'store'] +``` + ++ tp6(tp5.1) 配置 + +```php + 'constructor' => [think\Facade\Cache::class, 'instance'] +``` + +> 无论配置写成`[think\Facade\Cache::class, 'instance']` 还是写成 `[think\Facade\Cache::class, 'store']` 目的都是为了获取缓存实例,具体情况视框架而定 + + + ++ 灵活自定义: +1. 如果您的需要使用类似以下命令打包配置文件(ThinkPHP,Laravel 命令) + - php think optimize:config + - php artisan optimize + 则需要写成下面这样: +```php + $instance = \think\facade\Cache::store();//获取缓存想实例 + //省略分部代码 + 'constructor' => serialize($instance); +``` + +因为在执行optimize打包命令时,会尝试将对象进行序列化。 + +2. 如果您不需要使用打包压缩命令,或者使用了像hyperf这样的框架,除了上述的写法,还可以写成这样: + +```php + 'constructor' => function () { + $container = \Hyperf\Utils\ApplicationContext::getContainer(); + //在构造函数中传入自已的配置 + return $container->get(\Psr\SimpleCache\CacheInterface::class); + }, +``` + +除此之处,您传入的缓存实例应遵守psr-16规范 + +##### 背景图与滑动图 +> 配置中的backgrounds与templates项均支持,Array, String两种格式 ++ 使用Array(索引数组)格式时,表明它每一项都是一张图片。可以是本地图片路径,也可以是网络图片路径。 +示例: +```php + 'backgrounds' => [ + '/public/images/xxx.jpg', + 'http://www.image.xx.jpg' + ] +``` ++ 使用String格式时,表明它是一个图片所在位置的目录 +示例: +```php + 'backgrounds' => ROOT_PATH . '/resources/defaultImages/jigsaw/original/' +``` +当配置中的backgrounds与templates项为空时,会将`/resources/defaultImages/`目录内所有图片做为默认图片。 + +下面是社区人员一起维护的图片库,这些图片可以与本程序无缝衔接。 +https://gitee.com/anji-plus/AJ-Captcha-Images + +##### 字体配置 +字体配置在水印与文字点击验证功能中使用,其配置格式化String, 指向字体库。为空时会以`/resources/fonts`下的字体文件为默认值。 +示例: +```php +'font_file' => ROOT_PATH . '/resources/fonts/WenQuanZhengHei.ttf' +``` + +#### 获取滑动验证码 + +```php +public function get(){ + $config = require '../src/config.php'; + $service = new BlockPuzzleCaptchaService($config); + $data = $service->get(); + echo json_encode([ + 'error' => false, + 'repCode' => '0000', + 'repData' => $data, + 'repMsg' => null, + 'success' => true, + ]); +} +``` + +#### 滑动验证 + +```php + public function check() + { + $config = require '../src/config.php'; + $service = new BlockPuzzleCaptchaService($config); + $data = $_REQUEST; + $msg = null; + $error = false; + $repCode = '0000'; + try { + $service->check($data['token'], $data['pointJson']); + } catch (\Exception $e) { + $msg = $e->getMessage(); + $error = true; + $repCode = '6111'; + } + echo json_encode([ + 'error' => $error, + 'repCode' => $repCode, + 'repData' => null, + 'repMsg' => $msg, + 'success' => ! $error, + ]); + } +``` + +#### 获取文字验证码 + +```php + public function get() + { + $config = require '../src/config.php'; + $service = new ClickWordCaptchaService($config); + $data = $service->get(); + echo json_encode([ + 'error' => false, + 'repCode' => '0000', + 'repData' => $data, + 'repMsg' => null, + 'success' => true, + ]); + } +``` + +#### 文字验证 + +```php + public function check() + { + $config = require '../src/config.php'; + $service = new ClickWordCaptchaService($config); + $data = $_REQUEST; + $msg = null; + $error = false; + $repCode = '0000'; + try { + $service->check($data['token'], $data['pointJson']); + } catch (\Exception $e) { + $msg = $e->getMessage(); + $error = true; + $repCode = '6111'; + } + echo json_encode([ + 'error' => $error, + 'repCode' => $repCode, + 'repData' => null, + 'repMsg' => $msg, + 'success' => ! $error, + ]); + } +``` + +#### 前端请求头修改示例 + +```javascript +import axios from 'axios'; +import qs from 'qs'; + +axios.defaults.baseURL = 'https://captcha.anji-plus.com/captcha-api'; + +const service = axios.create({ + timeout: 40000, + headers: { + 'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8' + }, +}) +service.interceptors.request.use( + config => { + if (config.hasOwnProperty('data')) { + config.data = qs.stringify(config.data) + } + return config + }, + error => { + Promise.reject(error) + } +) +``` + +本包后续更新 ThinkPHP、Hyperf 等框架的demo,请持续关注 +https://gitee.com/fastknife/aj-captcha \ No newline at end of file diff --git a/extend/fastknife/ajcaptcha/readme.md b/extend/fastknife/ajcaptcha/readme.md new file mode 100644 index 00000000..aa799323 --- /dev/null +++ b/extend/fastknife/ajcaptcha/readme.md @@ -0,0 +1,69 @@ +AJ-Captcha · php +---- + +#### 介绍 + +这个类库使用 PHP实现了行为验证码。基于gd扩展生成滑动验证码和文字验证码。允许 phper定制验证码规则,并且不再使用 curl来请求第三方验证。 + +Java实现: https://gitee.com/anji-plus/captcha + +PHP实现: https://gitee.com/fastknife/aj-captcha + +##### 官方预览效果 + +![block](https://gitee.com/anji-plus/captcha/raw/master/images/%E6%BB%91%E5%8A%A8%E6%8B%BC%E5%9B%BE.gif)   ![click](https://gitee.com/anji-plus/captcha/raw/master/images/%E7%82%B9%E9%80%89%E6%96%87%E5%AD%97.gif) + +#### 注意事项 +* 你需要打开 gd、 openssl扩展 +* PHP版本至少需要7.1 +* 此软件包自带缓存,如有需要请自行更换 +* anji-plus/captcha前端默认请求头是 application/json 需替换为 application/x-www-form-urlencode + +* 滑动验证图响应时间慢? + > 性能慢的主要原因是受php GD库的imagecolorat函数与imagesetpixel函数性能的影响 + 1. 您可以尝试将修改内存`ini_set('memory_limit', '256M')` + 2. 将本包升级到1.1.x版本,开启像素缓存`block_puzzle.is_cache_pixel = true`, 若还是性能还是慢,则将干扰图片关闭`block_puzzle.is_interfere => false` +#### 如何使用 +test 目录下示例了三种使用方式,phper可以参考使用。[查看demo](./demo.md) +> 本软件包需要配合composer一起使用 +1. 非框架使用的场景,直接使用git下载这个软件包。然后执行composer命令`composer install`安装本软件包依赖,接着手动引入对应的 service层文件即可(同test目录里的原生引用方式)。 + +2. 基于框架使用的场景,输入安装命令`composer require fastknife/ajcaptcha`(稳定版) 或者`composer require fastknife/ajcaptcha dev-master`(最新版) ,建议使用composer阿里源(`https://mirrors.aliyun.com/composer`) + * 支持各种前沿框架(ThinkPHP, YII, Laravel, Hyperf,IMI,Swoft,EasySwoole) + * 本软件包内,未使用单例、注册树(容器)模式,不含任何全局变量,基于swoole开发的同学不用担心内存泄露。 + +#### 项目结构 +> 本软件包基于整洁架构理念,设计了下文的目录结构。Domain(领域层)作为内层同心圆承担所有业务逻辑功能,Service(服务层)并向最外层Controller(需自行实现)提供粗颗粒度服务。 + 区别于DDD(领域驱动设计),本软件包的领域层不含Entity(实体),以Logic(逻辑层)实现单元逻辑,为了方便管理作者将逻辑层的数据处理与图形处理分隔,以达到整洁效果。 +``` +AJ-Captcha for php +│ +├─resources 资源 +│ │ +│ ├─defaultImages 图片资源 +│ │ +│ └─fonts 字体 +│ +├─src 源码 +│ │ +│ ├─Domian 领域层 +│ │ +│ ├─Exception 异常 +│ │ +│ ├─Service 服务层 +│ │ +│ ├─ Utils工具类 +│ │ +│ └─ config.php 配置参考 +└─test 测试实例 + │ + ├─thinkphp thinkphp框架测试示例 + │ + ├─laravel laravel框架测试示例 + │ + └─*.php 原生测试文件 配置文件 +``` + +#### [更新日志](./changelog.md) + +若此软件对您有所帮助,您可以点右上角 💘Star💘支持 diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/1.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/1.png new file mode 100644 index 00000000..022aabf9 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/1.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/2.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/2.png new file mode 100644 index 00000000..914908e8 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/2.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/3.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/3.png new file mode 100644 index 00000000..f0f3ce58 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/3.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/4.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/4.png new file mode 100644 index 00000000..c5697f3c Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/4.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/5.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/5.png new file mode 100644 index 00000000..e29e7a3c Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/5.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/6.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/6.png new file mode 100644 index 00000000..2425f412 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/6.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/bg8.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/bg8.png new file mode 100644 index 00000000..5ea54d48 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/original/bg8.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/1.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/1.png new file mode 100644 index 00000000..19050266 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/1.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/2.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/2.png new file mode 100644 index 00000000..b1482d48 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/2.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/3.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/3.png new file mode 100644 index 00000000..cdbb0b18 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/3.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/4.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/4.png new file mode 100644 index 00000000..bc69c962 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/4.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/5.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/5.png new file mode 100644 index 00000000..0080a546 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/5.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/6.png b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/6.png new file mode 100644 index 00000000..b07c3b40 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/jigsaw/slidingBlock/6.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/1.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/1.png new file mode 100644 index 00000000..50dfe28e Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/1.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/2.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/2.png new file mode 100644 index 00000000..15b38ad2 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/2.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/3.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/3.png new file mode 100644 index 00000000..e2e487bd Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/3.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/4.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/4.png new file mode 100644 index 00000000..c34baa40 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/4.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/5.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/5.png new file mode 100644 index 00000000..0b3d11a2 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/5.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/6.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/6.png new file mode 100644 index 00000000..67797a11 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/6.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg10.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg10.png new file mode 100644 index 00000000..c99fbcb0 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg10.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg11.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg11.png new file mode 100644 index 00000000..6a951d32 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg11.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg12.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg12.png new file mode 100644 index 00000000..a38ada50 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg12.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg13.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg13.png new file mode 100644 index 00000000..07af86a8 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg13.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg14.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg14.png new file mode 100644 index 00000000..95593759 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg14.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg15.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg15.png new file mode 100644 index 00000000..cb1ebb63 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg15.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg16.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg16.png new file mode 100644 index 00000000..106b4562 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg16.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg17.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg17.png new file mode 100644 index 00000000..bcdbe765 Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg17.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg18.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg18.png new file mode 100644 index 00000000..ae94e09c Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg18.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg19.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg19.png new file mode 100644 index 00000000..bef9318b Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg19.png differ diff --git a/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg20.png b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg20.png new file mode 100644 index 00000000..36cfbdec Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/defaultImages/pic-click/bg20.png differ diff --git a/extend/fastknife/ajcaptcha/resources/fonts/WenQuanZhengHei.ttf b/extend/fastknife/ajcaptcha/resources/fonts/WenQuanZhengHei.ttf new file mode 100644 index 00000000..f84e9feb Binary files /dev/null and b/extend/fastknife/ajcaptcha/resources/fonts/WenQuanZhengHei.ttf differ diff --git a/extend/fastknife/ajcaptcha/resources/fonts/license.txt b/extend/fastknife/ajcaptcha/resources/fonts/license.txt new file mode 100644 index 00000000..719f68f0 --- /dev/null +++ b/extend/fastknife/ajcaptcha/resources/fonts/license.txt @@ -0,0 +1,55 @@ +文泉驿是一个开源汉字字体项目 + +由旅美学者房骞骞(FangQ) + +于2004年10月创建 + +集中力量解决GNU/Linux + +高质量中文字体匮乏的状况 + +目前,文泉驿已经开发并发布了 + +第一个完整覆盖GB18030汉字 + +(包含27000多个汉字) + +的多规格点阵汉字字型文件 + +第一个覆盖GBK字符集的 + +开源矢量字型文件(文泉驿正黑) + +并提供了目前包含字符数目最多的 + +开源字体——GNU Unifont——中 + +绝大多数中日韩文相关的符号 + +这些字型文件已经逐渐成为 + +主流Linux/Unix发行版 + +中文桌面的首选中文字体 + +目前Ubuntu、Fedora、Slackware + +Magic Linux、CDLinux + +使用文泉驿作为默认中文字体 + +Debian、Gentoo、Mandriva + +ArchLinux、Frugalware + +则提供了官方源支持 + +而FreeBSD则在其ports中有提供 + +所以,今天我们所要分享的就是 + +文泉驿正黑体 + +可在Linux/UNIX,Windows + +Mac OS和嵌入式操作系统中使用 \ No newline at end of file diff --git a/extend/fastknife/ajcaptcha/src/Domain/Factory.php b/extend/fastknife/ajcaptcha/src/Domain/Factory.php new file mode 100644 index 00000000..a95d484b --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Factory.php @@ -0,0 +1,161 @@ +config = $config; + } + + /** + * @return BlockImage + */ + public function makeBlockImage(): BlockImage + { + $data = new BlockData(); + $image = new BlockImage(); + $this->setCommon($image, $data); + $this->setBlock($image, $data); + return $image; + } + + /** + * @return WordImage + */ + public function makeWordImage(): WordImage + { + $data = new WordData(); + $image = new WordImage(); + $this->setCommon($image, $data); + $this->setWord($image, $data); + return $image; + } + + + /** + * 设置公共配置 + * @param BaseImage $image + * @param BaseData $data + */ + protected function setCommon(BaseImage $image, BaseData $data) + { + //固定驱动,少量图片处理场景gd性能远远大于imagick + ImageManagerStatic::configure(['driver' => 'gd']); + + //获得字体数据 + $fontFile = $data->getFontFile($this->config['font_file']); + $image + ->setFontFile($fontFile) + ->setWatermark($this->config['watermark']); + } + + /** + * 设置滑动验证码的配置 + * @param BlockImage $image + * @param BlockData $data + */ + protected function setBlock(BlockImage $image, BlockData $data) + { + //设置背景 + $backgroundVo = $data->getBackgroundVo($this->config['block_puzzle']['backgrounds']); + $image->setBackgroundVo($backgroundVo); + + $templateVo = $data->getTemplateVo($backgroundVo, $this->config['block_puzzle']['templates']); + + $image->setTemplateVo($templateVo); + + $pixelMaps = [$backgroundVo, $templateVo]; + if ( + isset($this->config['block_puzzle']['is_interfere']) && + $this->config['block_puzzle']['is_interfere'] == true + ) { + $interfereVo = $data->getInterfereVo($backgroundVo, $templateVo, $this->config['block_puzzle']['templates']); + $image->setInterfereVo($interfereVo); + $pixelMaps[] = $interfereVo; + } + + if ( + isset($this->config['block_puzzle']['is_cache_pixel']) && + $this->config['block_puzzle']['is_cache_pixel'] === true + ) { + $cache = $this->getCacheInstance(); + foreach ($pixelMaps as $vo) { + /**@var ImageVo $vo * */ + $key = 'image_pixel_map_' . $vo->src; + $result = $cache->get($key); + if (!empty($result) && is_array($result)) { + $vo->setPickMaps($result); + } else { + $vo->preparePickMaps(); + $vo->setFinishCallback(function (ImageVo $imageVo) use ($cache, $key) { + $cache->set($key, $imageVo->getPickMaps(), 0); + }); + } + } + } + + + } + + /** + * 设置文字验证码的配置 + * @param WordImage $image + * @param WordData $data + */ + protected function setWord(WordImage $image, WordData $data) + { + //设置背景 + $backgroundVo = $data->getBackgroundVo($this->config['click_world']['backgrounds']); + $image->setBackgroundVo($backgroundVo); + + //随机文字坐标 + $pointList = $data->getPointList( + $image->getBackgroundVo()->image->getWidth(), + $image->getBackgroundVo()->image->getHeight(), + 3 + ); + $worldList = $data->getWordList(count($pointList)); + $image + ->setWordList($worldList) + ->setWordList($worldList) + ->setPoint($pointList); + } + + /** + * 创建缓存实体 + */ + public function getCacheInstance(): Cache + { + if (empty($this->cacheInstance)) { + $this->cacheInstance = new Cache($this->config['cache']); + } + return $this->cacheInstance; + } + + public function makeWordData(): WordData + { + return new WordData(); + } + + public function makeBlockData(): BlockData + { + return (new BlockData())->setFaultOffset($this->config['block_puzzle']['offset']); + } +} diff --git a/extend/fastknife/ajcaptcha/src/Domain/Logic/BaseData.php b/extend/fastknife/ajcaptcha/src/Domain/Logic/BaseData.php new file mode 100644 index 00000000..0cbee61e --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Logic/BaseData.php @@ -0,0 +1,73 @@ +defaultFontPath; + } + + /** + * 获得随机图片 + * @param $images + * @return string + */ + protected function getRandImage($images): string + { + $index = RandomUtils::getRandomInt(0, count($images) - 1); + return $images[$index]; + } + + /** + * 获取默认图片 + * @param $dir + * @param $images + * @return array|false + */ + protected function getDefaultImage($dir, $images) + { + if (!empty($images)) { + if (is_array($images)) { + return $images; + } + if (is_string($images)) { + $dir = $images; + } + } + return glob($dir . '*.png'); + } + + /** + * 获取一张背景图地址 + * @param null $backgrounds 背景图库 + * @return BackgroundVo + */ + public function getBackgroundVo($backgrounds = null): BackgroundVo + { + $dir = dirname(__DIR__, 3). $this->defaultBackgroundPath; + $backgrounds = $this->getDefaultImage($dir, $backgrounds); + $src = $this->getRandImage($backgrounds); + return new BackgroundVo($src); + } +} diff --git a/extend/fastknife/ajcaptcha/src/Domain/Logic/BaseImage.php b/extend/fastknife/ajcaptcha/src/Domain/Logic/BaseImage.php new file mode 100644 index 00000000..d2058158 --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Logic/BaseImage.php @@ -0,0 +1,106 @@ +point; + } + + /** + * @param $point + * @return WordImage + */ + public function setPoint($point):self + { + $this->point = $point; + return $this; + } + + + protected function makeWatermark(Image $image) + { + if (! empty($this->watermark)) { + $info = imagettfbbox($this->watermark['fontsize'], 0, $this->fontFile, $this->watermark['text']); + $minX = min($info[0], $info[2], $info[4], $info[6]); + $minY = min($info[1], $info[3], $info[5], $info[7]); + $maxY = max($info[1], $info[3], $info[5], $info[7]); + $x = $minX; + $y = abs($minY); + /* 计算文字初始坐标和尺寸 */ + $h = $maxY - $minY; + $x += $image->getWidth() - $this->watermark['fontsize']/2; //留出半个单位字体像素的余白,不至于水印紧贴着右边 + $y += $image->getHeight() - $h; + $image->text($this->watermark['text'], $x, $y, function (Font $font) { + $font->file($this->fontFile); + $font->size($this->watermark['fontsize']); + $font->color($this->watermark['color']); + $font->align('right'); + $font->valign('bottom'); + }); + } + } + + + /** + * @param mixed $watermark + * @return self + */ + public function setWatermark($watermark): self + { + $this->watermark = $watermark; + return $this; + } + + + /** + * @param BackgroundVo $backgroundVo + * @return $this + */ + public function setBackgroundVo(BackgroundVo $backgroundVo):self + { + $this->backgroundVo = $backgroundVo; + return $this; + } + + /** + * @return BackgroundVo + */ + public function getBackgroundVo(): BackgroundVo + { + return $this->backgroundVo; + } + + /** + * @param $file + * @return static + */ + public function setFontFile($file): self + { + $this->fontFile = $file; + return $this; + } + + public abstract function run(); +} diff --git a/extend/fastknife/ajcaptcha/src/Domain/Logic/BlockData.php b/extend/fastknife/ajcaptcha/src/Domain/Logic/BlockData.php new file mode 100644 index 00000000..89011c89 --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Logic/BlockData.php @@ -0,0 +1,130 @@ +faultOffset; + } + + /** + * @param mixed $faultOffset + */ + public function setFaultOffset($faultOffset): self + { + $this->faultOffset = $faultOffset; + return $this; + } + + + /** + * 获取剪切模板Vo + * @param BackgroundVo $backgroundVo + * @param array $templates + * @return TemplateVo + */ + public function getTemplateVo(BackgroundVo $backgroundVo, array $templates = []): TemplateVo + { + $background = $backgroundVo->image; + //初始偏移量,让模板图在背景的右1/2位置 + $bgWidth = intval($background->getWidth() / 2); + //随机获取一张图片 + $src = $this->getRandImage($this->getTemplateImages($templates)); + + $templateVo = new TemplateVo($src); + + //随机获取偏移量 + $offset = RandomUtils::getRandomInt(0, $bgWidth - $templateVo->image->getWidth() - 1); + + $templateVo->setOffset(new OffsetVo($offset + $bgWidth, 0)); + return $templateVo; + } + + + + public function getInterfereVo(BackgroundVo $backgroundVo, TemplateVo $templateVo, $templates = []): TemplateVo + { + //背景 + $background = $backgroundVo->image; + //模板库去重 + $templates = $this->exclude($this->getTemplateImages($templates), $templateVo->src); + + //随机获取一张模板图 + $src = $this->getRandImage($templates); + + $interfereVo = new TemplateVo($src); + + $maxOffsetX = intval($templateVo->image->getWidth()/2); + do { + //随机获取偏移量 + $offsetX = RandomUtils::getRandomInt(0, $background->getWidth() - $templateVo->image->getWidth() - 1); + + //不与原模板重复 + if ( + abs($templateVo->offset->x - $offsetX) > $maxOffsetX + ) { + $offsetVO = new OffsetVo($offsetX, 0); + $interfereVo->setOffset($offsetVO); + return $interfereVo; + } + } while (true); + } + + + protected function getTemplateImages(array $templates = []) + { + $dir = dirname(__DIR__, 3) . '/resources/defaultImages/jigsaw/slidingBlock/'; + return $this->getDefaultImage($dir, $templates); + } + + /** + * 排除 + * @param $templates + * @param $exclude + * @return array + */ + protected function exclude($templates, $exclude): array + { + if (false !== ($key = array_search($exclude, $templates))) { + array_splice($templates,$key,1); + } + return $templates; + } + + + + /** + * @param $originPoint + * @param $targetPoint + * @return void + */ + public function check($originPoint, $targetPoint) + { + if ( + abs($originPoint->x - $targetPoint->x) <= $this->faultOffset + && $originPoint->y == $targetPoint->y + ) { + return; + } + throw new BlockException('验证失败!'); + } + +} diff --git a/extend/fastknife/ajcaptcha/src/Domain/Logic/BlockImage.php b/extend/fastknife/ajcaptcha/src/Domain/Logic/BlockImage.php new file mode 100644 index 00000000..d990320e --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Logic/BlockImage.php @@ -0,0 +1,141 @@ +templateVo; + } + + /** + * @param TemplateVo $templateVo + * @return self + */ + public function setTemplateVo(TemplateVo $templateVo): self + { + $this->templateVo = $templateVo; + return $this; + } + + /** + * @return TemplateVo + */ + public function getInterfereVo(): TemplateVo + { + return $this->interfereVo; + } + + /** + * @param TemplateVo $interfereVo + * @return static + */ + public function setInterfereVo(TemplateVo $interfereVo): self + { + + $this->interfereVo = $interfereVo; + return $this; + } + + public function run() + { + $flag = false; + $this->cutByTemplate($this->templateVo, $this->backgroundVo, function ($param) use (&$flag) { + if (!$flag) { + //记录第一个点 + $this->setPoint(new PointVo($param[0], 5));//前端已将y值写死 + $flag = true; + } + }); + if (!empty($this->interfereVo)) { + $this->cutByTemplate($this->interfereVo, $this->backgroundVo); + } + $this->makeWatermark($this->backgroundVo->image); + } + + + public function cutByTemplate(TemplateVo $templateVo, BackgroundVo $backgroundVo, $callable = null) + { + $template = $templateVo->image; + $width = $template->getWidth(); + $height = $template->getHeight(); + for ($x = 0; $x < $width; $x++) { + for ($y = 0; $y < $height; $y++) { + //背景图对应的坐标 + $bgX = $x + $templateVo->offset->x; + $bgY = $y + $templateVo->offset->y; + //是否不透明 + $isOpacity = $templateVo->isOpacity($x, $y); + if ($isOpacity) { //如果不透明 + if ($callable instanceof \Closure) { + $callable([$bgX, $bgY]); + } + $backgroundVo->vagueImage($bgX, $bgY);//模糊背景图选区 + + $this->copyPickColor($backgroundVo, $bgX, $bgY, $templateVo, $x, $y); + } + if ($templateVo->isBoundary($isOpacity, $x, $y)) { + $backgroundVo->setPixel(self::WHITE, $bgX, $bgY); + $templateVo->setPixel(self::WHITE, $x, $y); + } + } + } + } + + /** + * 把$source的颜色复制到$target上 + * @param ImageVo $source + * @param ImageVo $target + */ + + protected function copyPickColor(ImageVo $source, $sourceX, $sourceY, ImageVo $target, $targetX, $targetY) + { + $bgRgba = $source->getPickColor($sourceX, $sourceY); + $target->setPixel($bgRgba, $targetX, $targetY);//复制背景图片给模板 + } + + /** + * 返回前端需要的格式 + * @return false|string[] + */ + public function response($type = 'background') + { + $image = $type == 'background' ? $this->backgroundVo->image : $this->templateVo->image; + $result = $image->encode('data-url')->getEncoded(); + //返回图片base64的第二部分 + return explode(',', $result)[1]; + } + + /** + * 用来调试 + */ + public function echo($type = 'background') + { + $image = $type == 'background' ? $this->backgroundVo->image : $this->templateVo->image; + die($image->response()); + } +} diff --git a/extend/fastknife/ajcaptcha/src/Domain/Logic/Cache.php b/extend/fastknife/ajcaptcha/src/Domain/Logic/Cache.php new file mode 100644 index 00000000..cf35db12 --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Logic/Cache.php @@ -0,0 +1,114 @@ + 'get', + 'set' => 'set', + 'delete' => 'delete', + 'has' => 'has' + ]; + + public function __construct($config) + { + if (isset($config['method'])) { + $this->methodMap = array_merge($this->methodMap, $config['method']); + } + $this->driver = $this->getDriver($config['constructor'], $config['options']??[]); + } + + public function getDriver($callback, $options) + { + if ($callback instanceof \Closure) { + $result = $callback($options); + } else if (is_object($callback)) { + $result = $callback; + } else if (is_array($callback)) { + $result = call_user_func($callback, $options); + } else if ($this->isSerialized($callback)) { + $result = unserialize($callback); + } else if (is_string($callback) && class_exists($callback)) { + $result = new $callback($options); + } else { + $result = new CacheUtils($options); + } + return $result; + } + + /** + * 是否可以被反序列化 + * @param $data + * @return bool + */ + public function isSerialized($data): bool + { + if (!is_string($data)) { + return false; + } + $data = trim($data); + if ('N;' == $data) { + return true; + } + if (!preg_match('/^([adObis]):/', $data, $badions)) { + return false; + } + switch ($badions[1]) { + case 'a' : + case 'O' : + case 's' : + if (preg_match("/^{$badions[1]}:[0-9]+:.*[;}]\$/s", $data)) + return true; + break; + case 'b' : + case 'i' : + case 'd' : + if (preg_match("/^{$badions[1]}:[0-9.E-]+;\$/", $data)) + return true; + break; + } + return false; + } + + public function getDriverMethod($name) + { + return $this->methodMap[$name]; + } + + public function get($key, $default = null) + { + $method = $this->getDriverMethod('get'); + return $this->execute($method, [$key,$default]); + } + + public function set($key, $value, $ttl = null) + { + $method = $this->getDriverMethod('set'); + return $this->execute($method, [$key, $value, $ttl]); + } + + public function delete($key) + { + $method = $this->getDriverMethod('delete'); + return $this->execute($method, [$key]); + } + + public function has($key) + { + $method = $this->getDriverMethod('has'); + return $this->execute($method, [$key]); + } + + protected function execute(string $method, array $params){ + return $this->driver->$method(...$params); + } + +} diff --git a/extend/fastknife/ajcaptcha/src/Domain/Logic/WordData.php b/extend/fastknife/ajcaptcha/src/Domain/Logic/WordData.php new file mode 100644 index 00000000..e4fba23f --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Logic/WordData.php @@ -0,0 +1,95 @@ +getPoint($width, $height, $i, $number); + } + shuffle($pointList); //随机排序 + return $pointList; + } + + + /** + * @param $list + * @return array + */ + public function array2Point($list): array + { + $result = []; + foreach ($list as $item) { + $result[] = new PointVo($item['x'], $item['y']); + } + return $result; + } + + public function getWordList($number): array + { + return RandomUtils::getRandomChar($number); + } + + /** + * 校验 + * @param array $originPointList + * @param array $targetPointList + */ + public function check(array $originPointList, array $targetPointList) + { + foreach ($originPointList as $key => $originPoint) { + $targetPoint = $targetPointList[$key]; + if ($targetPoint->x - self::FONTSIZE > $originPoint->x + || $targetPoint->x > $originPoint->x + self::FONTSIZE + || $targetPoint->y - self::FONTSIZE > $originPoint->y + || $targetPoint->y > $originPoint->y + self::FONTSIZE) { + throw new WordException('验证失败!'); + } + } + } +} diff --git a/extend/fastknife/ajcaptcha/src/Domain/Logic/WordImage.php b/extend/fastknife/ajcaptcha/src/Domain/Logic/WordImage.php new file mode 100644 index 00000000..96d600c5 --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Logic/WordImage.php @@ -0,0 +1,83 @@ +wordList = $wordList; + return $this; + } + + public function getWordList() + { + return $this->wordList; + } + + + + public function run() + { + $this->inputWords(); + $this->makeWatermark($this->backgroundVo->image); + } + + /** + * 写入文字 + */ + protected function inputWords(){ + foreach ($this->wordList as $key => $word) { + $point = $this->point[$key]; + $this->backgroundVo->image->text($word, $point->x, $point->y, function (Font $font) { + $font->file($this->fontFile); + $font->size(BaseData::FONTSIZE); + $font->color(RandomUtils::getRandomColor()); + $font->angle(RandomUtils::getRandomAngle()); + $font->align('center'); + $font->valign('center'); + }); + } + } + + /** + * 返回前端需要的格式 + * @return false|string[] + */ + public function response() + { + $result = $this->getBackgroundVo()->image->encode('data-url')->getEncoded(); + //返回图片base64的第二部分 + return explode(',', $result)[1]; + } + + /** + * 用来调试 + */ + public function echo() + { + die($this->getBackgroundVo()->image->response()); + } + +} diff --git a/extend/fastknife/ajcaptcha/src/Domain/Vo/BackgroundVo.php b/extend/fastknife/ajcaptcha/src/Domain/Vo/BackgroundVo.php new file mode 100644 index 00000000..f1dea21a --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Vo/BackgroundVo.php @@ -0,0 +1,8 @@ +src = $src; + $this->initImage($src); + } + + public function initImage($src) + { + $this->image = ImageManagerStatic::make($src); + } + + /** + * 获取图片中某一个位置的rgba值 + * @param $x + * @param $y + * @return array + */ + public function getPickColor($x, $y): array + { + if (!isset($this->pickMaps[$x][$y])) { + $this->pickMaps[$x][$y] = $this->image->pickColor($x, $y); + } + return $this->pickMaps[$x][$y]; + } + + + /** + * 设置图片指定位置的颜色值 + */ + public function setPixel($color, $x, $y) + { + $this->image->pixel($color, $x, $y); + } + + /** + * @param int $x + * @param int $y + * @return array + */ + public function getBlurValue(int $x, int $y): array + { + $image = $this->image; + $red = []; + $green = []; + $blue = []; + $alpha = []; + foreach ([ + [0, 1], [0, -1], + [1, 0], [-1, 0], + [1, 1], [1, -1], + [-1, 1], [-1, -1], + ] as $distance) //边框取5个点,4个角取3个点,其余取8个点 + { + $pointX = $x + $distance[0]; + $pointY = $y + $distance[1]; + if ($pointX < 0 || $pointX >= $image->getWidth() || $pointY < 0 || $pointY >= $image->height()) { + continue; + } + [$r, $g, $b, $a] = $this->getPickColor($pointX, $pointY); + $red[] = $r; + $green[] = $g; + $blue[] = $b; + $alpha[] = $a; + } + return [MathUtils::avg($red), MathUtils::avg($green), MathUtils::avg($blue), MathUtils::avg($alpha)]; + } + + + /** + * 是否不透明 + * @param $x + * @param $y + * @return bool + */ + public function isOpacity($x, $y): bool + { + return $this->getPickColor($x, $y)[3] > 0.5; + } + + /** + * 是否为边框 + * @param bool $isOpacity + * @param int $x + * @param int $y + * @return bool + */ + public function isBoundary(bool $isOpacity, int $x, int $y): bool + { + $image = $this->image; + if ($x >= $image->width() - 1 || $y >= $image->height() - 1) { + return false; + } + $right = [$x + 1, $y]; + $down = [$x, $y + 1]; + if ( + $isOpacity && !$this->isOpacity(...$right) + || !$isOpacity && $this->isOpacity(...$right) + || $isOpacity && !$this->isOpacity(...$down) + || !$isOpacity && $this->isOpacity(...$down) + ) { + return true; + } + return false; + } + + /** + * 模糊图片 + * @param $targetX + * @param $targetY + */ + public function vagueImage($targetX, $targetY) + { + $blur = $this->getBlurValue($targetX, $targetY); + $this->setPixel($blur, $targetX, $targetY); + } + + + /** + * @return array + */ + public function getPickMaps(): array + { + return $this->pickMaps; + } + + /** + * @param array $pickMaps + */ + public function setPickMaps(array $pickMaps): void + { + $this->pickMaps = $pickMaps; + } + + /** + * 提前初始化像素 + */ + public function preparePickMaps() + { + $width = $this->image->getWidth(); + $height = $this->image->getHeight(); + for ($x = 0; $x < $width; $x++) { + for ($y = 0; $y < $height; $y++) { + $this->getPickColor($x, $y); + } + } + } + + public function setFinishCallback($finishCallback){ + $this->finishCallback = $finishCallback; + } + + public function __destruct() + { + if(!empty($this->finishCallback) && $this->finishCallback instanceof \Closure){ + ($this->finishCallback)($this); + } + } +} \ No newline at end of file diff --git a/extend/fastknife/ajcaptcha/src/Domain/Vo/OffsetVo.php b/extend/fastknife/ajcaptcha/src/Domain/Vo/OffsetVo.php new file mode 100644 index 00000000..7a6e597f --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Vo/OffsetVo.php @@ -0,0 +1,18 @@ +x = $x; + $this->y = $y; + } + +} diff --git a/extend/fastknife/ajcaptcha/src/Domain/Vo/PointVo.php b/extend/fastknife/ajcaptcha/src/Domain/Vo/PointVo.php new file mode 100644 index 00000000..95c63412 --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Vo/PointVo.php @@ -0,0 +1,22 @@ +x = $x; + $this->y = $y; + } +} diff --git a/extend/fastknife/ajcaptcha/src/Domain/Vo/TemplateVo.php b/extend/fastknife/ajcaptcha/src/Domain/Vo/TemplateVo.php new file mode 100644 index 00000000..8ab7a6eb --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Domain/Vo/TemplateVo.php @@ -0,0 +1,32 @@ +offset; + } + + /** + * @param OffsetVo $offset + */ + public function setOffset(OffsetVo $offset): void + { + $this->offset = $offset; + } + + + +} diff --git a/extend/fastknife/ajcaptcha/src/Exception/BlockException.php b/extend/fastknife/ajcaptcha/src/Exception/BlockException.php new file mode 100644 index 00000000..2efea377 --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Exception/BlockException.php @@ -0,0 +1,10 @@ +factory->getCacheInstance(); + $blockImage = $this->factory->makeBlockImage(); + $blockImage->run(); + $data = [ + 'originalImageBase64' => $blockImage->response(), + 'jigsawImageBase64' => $blockImage->response('template'), + 'secretKey' => RandomUtils::getRandomCode(16, 3), + 'token' => RandomUtils::getUUID(), + ]; + //缓存 + $cacheEntity->set($data['token'], [ + 'secretKey' => $data['secretKey'], + 'point' => $blockImage->getPoint() + ], 7200); + return $data; + } + + + + + /** + * 验证 + * @param string $token + * @param string $pointJson + * @param null $callback + */ + public function validate( $token, $pointJson, $callback = null) + { + //获取并设置 $this->originData + $this->setOriginData($token); + + //数据处理类 + $blockData = $this->factory->makeBlockData(); + + //解码出来的前端坐标 + $targetPoint = $this->encodePoint($this->originData['secretKey'], $pointJson);; + $targetPoint = new PointVo($targetPoint['x'], $targetPoint['y']); + + //检查 + $blockData->check($this->originData['point'], $targetPoint); + if($callback instanceof \Closure){ + $callback(); + } + } +} diff --git a/extend/fastknife/ajcaptcha/src/Service/ClickWordCaptchaService.php b/extend/fastknife/ajcaptcha/src/Service/ClickWordCaptchaService.php new file mode 100644 index 00000000..8a0483af --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Service/ClickWordCaptchaService.php @@ -0,0 +1,64 @@ +factory->getCacheInstance(); + $wordImage = $this->factory->makeWordImage(); + //执行创建 + $wordImage->run(); + $data = [ + 'originalImageBase64' => $wordImage->response(), + 'secretKey' => RandomUtils::getRandomCode(16, 3), + 'token' => RandomUtils::getUUID(), + 'wordList' => $wordImage->getWordList() + ]; + //缓存 + $cacheEntity->set($data['token'], [ + 'secretKey' => $data['secretKey'], + 'point' => $wordImage->getPoint() + ],7200); + return $data; + } + + + /** + * 验证 + * @param $token + * @param $pointJson + * @param null $callback + */ + public function validate($token, $pointJson, $callback = null) + { + //获取并设置 $this->originData + $this->setOriginData($token); + + //数据实例 + $wordData = $this->factory->makeWordData(); + //解码出来的前端坐标 + $pointJson = $this->encodePoint($this->originData['secretKey'], $pointJson); + $targetPointList = $wordData->array2Point($pointJson); + + //检查 + $wordData->check($this->originData['point'], $targetPointList); + if ($callback instanceof \Closure) { + $callback(); + } + } + + + +} diff --git a/extend/fastknife/ajcaptcha/src/Service/Service.php b/extend/fastknife/ajcaptcha/src/Service/Service.php new file mode 100644 index 00000000..035e14aa --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Service/Service.php @@ -0,0 +1,113 @@ +factory = new Factory($config); + } + + abstract public function get(); + + + abstract public function validate( $token, $pointJson, $callback = null); + + /** + * 一次验证 + * @param $token + * @param $pointJson + */ + public function check($token, $pointJson) + { + $this->validate($token, $pointJson, function () use ($token, $pointJson) { + $this->setEncryptCache($token, $pointJson); + }); + } + + private function setEncryptCache($token, $pointJson){ + $cacheEntity = $this->factory->getCacheInstance(); + $pointStr = AesUtils::decrypt($pointJson, $this->originData['secretKey']); + $key = AesUtils::encrypt($token. '---'. $pointStr, $this->originData['secretKey']); + $cacheEntity->set($key, + [ + 'token' => $token, + 'point' => $pointJson + ], + 600 + ); + } + + + /** + * 二次验证 + * @param $token + * @param $pointJson + */ + public function verification($token, $pointJson) + { + $this->validate($token, $pointJson, function () use ($token) { + $cacheEntity = $this->factory->getCacheInstance(); + $cacheEntity->delete($token); + }); + } + + /** + * 二次验证,必须要先执行一次验证才能调用二次验证 + * 兼容前端 captchaVerification 传值 + * @param string $encryptCode + */ + public function verificationByEncryptCode(string $encryptCode) + { + $result = $this->factory->getCacheInstance()->get($encryptCode); + if(empty($result)){ + throw new ParamException('参数错误!'); + } + + $this->validate($result['token'], $result['point'], function () use ($result,$encryptCode) { + $cacheEntity = $this->factory->getCacheInstance(); + $cacheEntity->delete($result['token']); + $cacheEntity->delete($encryptCode); + }); + + } + + /** + * 解码坐标点 + * @param $secretKey + * @param $point + * @return array + */ + protected function encodePoint($secretKey, $point): array + { + $pointJson = AesUtils::decrypt($point, $secretKey); + if ($pointJson == false) { + throw new ParamException('aes验签失败!'); + } + return json_decode($pointJson, true); + } + + protected function setOriginData($token){ + $cacheEntity = $this->factory->getCacheInstance(); + $this->originData = $cacheEntity->get($token); + if (empty($this->originData)) { + throw new ParamException('参数校验失败:token'); + } + } +} diff --git a/extend/fastknife/ajcaptcha/src/Utils/AesUtils.php b/extend/fastknife/ajcaptcha/src/Utils/AesUtils.php new file mode 100644 index 00000000..6c6e193d --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Utils/AesUtils.php @@ -0,0 +1,39 @@ + 300, + 'cache_subdir' => true, + 'prefix' => '', + 'path' => '', + 'hash_type' => 'md5', + 'data_compress' => false, + 'tag_prefix' => 'tag:', + 'serialize' => [], + ]; + + /** + * 缓存读取次数 + * @var integer + */ + protected $readTimes = 0; + + /** + * 缓存写入次数 + * @var integer + */ + protected $writeTimes = 0; + + + /** + * 架构函数 + * @param array $options 参数 + */ + public function __construct(array $options = []) + { + if (!empty($options)) { + $this->options = array_merge($this->options, $options); + } + + if (empty($this->options['path'])) { + $root = isset($_SERVER['DOCUMENT_ROOT']) && !empty($_SERVER['DOCUMENT_ROOT']) ?$_SERVER['DOCUMENT_ROOT'] : getcwd(); + $this->options['path'] = $root. '/runtime/cache'; + } + + if (substr($this->options['path'], -1) != DIRECTORY_SEPARATOR) { + $this->options['path'] .= DIRECTORY_SEPARATOR; + } + } + + /** + * 取得变量的存储文件名 + * @access public + * @param string $name 缓存变量名 + * @return string + */ + public function getCacheKey(string $name): string + { + $name = hash($this->options['hash_type'], $name); + + if ($this->options['cache_subdir']) { + // 使用子目录 + $name = substr($name, 0, 2) . DIRECTORY_SEPARATOR . substr($name, 2); + } + + if ($this->options['prefix']) { + $name = $this->options['prefix'] . DIRECTORY_SEPARATOR . $name; + } + + return $this->options['path'] . $name . '.php'; + } + + /** + * 获取缓存数据 + * @param string $name 缓存标识名 + * @return array|null + */ + protected function getRaw(string $name) + { + $filename = $this->getCacheKey($name); + + if (!is_file($filename)) { + return; + } + + $content = @file_get_contents($filename); + + if (false !== $content) { + $expire = (int) substr($content, 8, 12); + if (0 != $expire && time() - $expire > filemtime($filename)) { + //缓存过期删除缓存文件 + $this->unlink($filename); + return; + } + + $content = substr($content, 32); + + return is_string($content) ? ['content' => $content, 'expire' => $expire] : null; + } + } + + /** + * 判断缓存是否存在 + * @access public + * @param string $name 缓存变量名 + * @return bool + */ + public function has($name): bool + { + return $this->getRaw($name) !== null; + } + + /** + * 读取缓存 + * @access public + * @param string $name 缓存变量名 + * @param mixed $default 默认值 + * @return mixed + */ + public function get($name, $default = null) + { + $this->readTimes++; + + $raw = $this->getRaw($name); + + return is_null($raw) ? $default : $this->unserialize($raw['content']); + } + + /** + * 写入缓存 + * @access public + * @param string $name 缓存变量名 + * @param mixed $value 存储数据 + * @param int $expire 有效时间 0为永久 + * @return bool + */ + public function set($name, $value, $expire = null): bool + { + $this->writeTimes++; + + if (is_null($expire)) { + $expire = $this->options['expire']; + } + + $filename = $this->getCacheKey($name); + + $dir = dirname($filename); + + if (!is_dir($dir)) { + try { + mkdir($dir, 0755, true); + } catch (\Exception $e) { + // 创建失败 + } + } + + $data = $this->serialize($value); + + if ($this->options['data_compress'] && function_exists('gzcompress')) { + //数据压缩 + $data = gzcompress($data, 3); + } + + $data = "\n" . $data; + $result = file_put_contents($filename, $data); + + if ($result) { + clearstatcache(); + return true; + } + + return false; + } + + /** + * 自增缓存(针对数值缓存) + * @access public + * @param string $name 缓存变量名 + * @param int $step 步长 + * @return false|int + */ + public function inc(string $name, int $step = 1) + { + if ($raw = $this->getRaw($name)) { + $value = $this->unserialize($raw['content']) + $step; + $expire = $raw['expire']; + } else { + $value = $step; + $expire = 0; + } + + return $this->set($name, $value, $expire) ? $value : false; + } + + /** + * 自减缓存(针对数值缓存) + * @access public + * @param string $name 缓存变量名 + * @param int $step 步长 + * @return false|int + */ + public function dec(string $name, int $step = 1) + { + return $this->inc($name, -$step); + } + + /** + * 删除缓存 + * @access public + * @param string $name 缓存变量名 + * @return bool + */ + public function delete($name): bool + { + $this->writeTimes++; + + return $this->unlink($this->getCacheKey($name)); + } + + /** + * 清除缓存 + * @access public + * @return bool + */ + public function clear(): bool + { + $this->writeTimes++; + + $dirname = $this->options['path'] . $this->options['prefix']; + + $this->rmdir($dirname); + + return true; + } + + /** + * 删除缓存标签 + * @access public + * @param array $keys 缓存标识列表 + * @return void + */ + public function clearTag(array $keys): void + { + foreach ($keys as $key) { + $this->unlink($key); + } + } + + /** + * 判断文件是否存在后,删除 + * @access private + * @param string $path + * @return bool + */ + private function unlink(string $path): bool + { + try { + return is_file($path) && unlink($path); + } catch (\Exception $e) { + return false; + } + } + + /** + * 删除文件夹 + * @param $dirname + * @return bool + */ + private function rmdir($dirname) + { + if (!is_dir($dirname)) { + return false; + } + + $items = new \FilesystemIterator($dirname); + + foreach ($items as $item) { + if ($item->isDir() && !$item->isLink()) { + $this->rmdir($item->getPathname()); + } else { + $this->unlink($item->getPathname()); + } + } + + @rmdir($dirname); + + return true; + } + /** + * 序列化数据 + * @access protected + * @param mixed $data 缓存数据 + * @return string + */ + protected function serialize($data): string + { + if (is_numeric($data)) { + return (string) $data; + } + + $serialize = $this->options['serialize'][0] ?? "serialize"; + + return $serialize($data); + } + + /** + * 反序列化数据 + * @access protected + * @param string $data 缓存数据 + * @return mixed + */ + protected function unserialize(string $data) + { + if (is_numeric($data)) { + return $data; + } + + $unserialize = $this->options['serialize'][1] ?? "unserialize"; + + return $unserialize($data); + } + + +} diff --git a/extend/fastknife/ajcaptcha/src/Utils/MathUtils.php b/extend/fastknife/ajcaptcha/src/Utils/MathUtils.php new file mode 100644 index 00000000..aa794730 --- /dev/null +++ b/extend/fastknife/ajcaptcha/src/Utils/MathUtils.php @@ -0,0 +1,16 @@ + '', //自定义字体包路径, 不填使用默认值 + //文字验证码 + 'click_world' => [ + 'backgrounds' => [] + ], + //滑动验证码 + 'block_puzzle' => [ + /*背景图片路径, 不填使用默认值, 支持string与array两种数据结构。string为默认图片的目录,array索引数组则为具体图片的地址*/ + 'backgrounds' => [], + + /*模板图,格式同上支持string与array*/ + 'templates' => [], + + 'offset' => 10, //容错偏移量 + + 'is_cache_pixel' => true, //是否开启缓存图片像素值,开启后能提升服务端响应性能(但要注意更换图片时,需要清除缓存) + + 'is_interfere' => true, //开启干扰图 + ], + //水印 + 'watermark' => [ + 'fontsize' => 12, + 'color' => '#000000', + 'text' => '我的水印' + ], + 'cache' => [ + //若您使用了框架,并且想使用类似于redis这样的缓存驱动,则应换成框架的中的缓存驱动 + 'constructor' => \Fastknife\Utils\CacheUtils::class, + 'method' => [ + //遵守PSR-16规范不需要设置此项(tp6, laravel,hyperf)。如tp5就不支持(tp5缓存方法是rm,所以要配置为"delete" => "rm") + /** + 'get' => 'get', //获取 + 'set' => 'set', //设置 + 'delete' => 'delete',//删除 + 'has' => 'has' //key是否存在 + */ + ], + 'options' => [ + //如果您依然使用\Fastknife\Utils\CacheUtils做为您的缓存驱动,那么您可以自定义缓存配置。 + 'expire' => 300,//缓存有效期 (默认为0 表示永久缓存) + 'prefix' => '', //缓存前缀 + 'path' => '', //缓存目录 + 'serialize' => [], //缓存序列化和反序列化方法 + ] + ] +]; diff --git a/extend/fastknife/ajcaptcha/test/BlockPuzzleController.php b/extend/fastknife/ajcaptcha/test/BlockPuzzleController.php new file mode 100644 index 00000000..fd35f3be --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/BlockPuzzleController.php @@ -0,0 +1,82 @@ +get(); + echo json_encode([ + 'error' => false, + 'repCode' => '0000', + 'repData' => $data, + 'repMsg' => null, + 'success' => true, + ]); + } + + /** + * 一次验证 + */ + public function check() + { + $config = require '../src/config.php'; + $service = new BlockPuzzleCaptchaService($config); + $data = $_REQUEST; + $msg = null; + $error = false; + $repCode = '0000'; + try { + $service->check($data['token'], $data['pointJson']); + } catch (\Exception $e) { + $msg = $e->getMessage(); + $error = true; + $repCode = '6111'; + } + echo json_encode([ + 'error' => $error, + 'repCode' => $repCode, + 'repData' => null, + 'repMsg' => $msg, + 'success' => ! $error, + ]); + } + + /** + * 二次验证 + */ + public function verification() + { + $config = require '../src/config.php'; + $service = new BlockPuzzleCaptchaService($config); + $data = $_REQUEST; + $msg = null; + $error = false; + $repCode = '0000'; + + try { + if(isset($data['captchaVerification'])){ + $service->verificationByEncryptCode($data['captchaVerification']); + }else if (isset($data['token']) && isset($data['pointJson'])){ + $service->verification($data['token'], $data['pointJson']); + } else { + throw new \Exception('参数错误!'); + } + } catch (\Exception $e) { + $msg = $e->getMessage(); + $error = true; + $repCode = '6111'; + } + echo json_encode([ + 'error' => $error, + 'repCode' => $repCode, + 'repData' => null, + 'repMsg' => $msg, + 'success' => ! $error, + ]); + } + + +} diff --git a/extend/fastknife/ajcaptcha/test/ClickWordController.php b/extend/fastknife/ajcaptcha/test/ClickWordController.php new file mode 100644 index 00000000..dca5048c --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/ClickWordController.php @@ -0,0 +1,90 @@ +get(); + echo json_encode([ + 'error' => false, + 'repCode' => '0000', + 'repData' => $data, + 'repMsg' => null, + 'success' => true, + ]); + } + + /** + * 一次验证 + */ + public function check() + { + $config = require '../src/config.php'; + $service = new ClickWordCaptchaService($config); + $data = $_REQUEST; + $msg = null; + $error = false; + $repCode = '0000'; + try { + $service->check($data['token'], $data['pointJson']); + } catch (\Exception $e) { + $msg = $e->getMessage(); + $error = true; + $repCode = '6111'; + } + echo json_encode([ + 'error' => $error, + 'repCode' => $repCode, + 'repData' => null, + 'repMsg' => $msg, + 'success' => ! $error, + ]); + } + + /** + * 二次验证 + */ + public function verification() + { + $config = require '../src/config.php'; + $service = new ClickWordCaptchaService($config); + $data = $_REQUEST; + $msg = null; + $error = false; + $repCode = '0000'; + try { + if(isset($data['captchaVerification'])){ + $service->verificationByEncryptCode($data['captchaVerification']); + }else if (isset($data['token']) && isset($data['pointJson'])){ + $service->verification($data['token'], $data['pointJson']); + } else { + throw new \Exception('参数错误!'); + } + } catch (\Exception $e) { + $msg = $e->getMessage(); + $error = true; + $repCode = '6111'; + } + echo json_encode([ + 'error' => $error, + 'repCode' => $repCode, + 'repData' => null, + 'repMsg' => $msg, + 'success' => ! $error, + ]); + } +} + + + + + + + + diff --git a/extend/fastknife/ajcaptcha/test/autoload.php b/extend/fastknife/ajcaptcha/test/autoload.php new file mode 100644 index 00000000..f6aacab6 --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/autoload.php @@ -0,0 +1,16 @@ +check(); diff --git a/extend/fastknife/ajcaptcha/test/get.php b/extend/fastknife/ajcaptcha/test/get.php new file mode 100644 index 00000000..52236a7f --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/get.php @@ -0,0 +1,18 @@ +get(); diff --git a/extend/fastknife/ajcaptcha/test/laravel/IndexController.php b/extend/fastknife/ajcaptcha/test/laravel/IndexController.php new file mode 100644 index 00000000..3c6c9bac --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/laravel/IndexController.php @@ -0,0 +1,106 @@ +getCaptchaService(); + $data = $service->get(); + } catch (\Exception $e) { + return $this->error($e->getMessage()); + } + return $this->success($data); + } + + /** + * 一次验证 + * @return array + */ + public function check() + { + try { + $data = $this->validate(); + $service = $this->getCaptchaService(); + $service->check($data['token'], $data['pointJson']); + } catch (\Exception $e) { + return $this->error($e->getMessage()); + } + return $this->success([]); + } + + /** + * 二次验证 + * @return array + */ + public function verification() + { + try { + $data = $this->validate(); + $service = $this->getCaptchaService(); + $service->verification($data['token'], $data['pointJson']); + } catch (\Exception $e) { + return $this->error($e->getMessage()); + } + return $this->success([]); + } + + protected function getCaptchaService() + { + $captchaType = request()->post('captchaType', null); + $config = config('captcha'); + switch ($captchaType) { + case "clickWord": + $service = new ClickWordCaptchaService($config); + break; + case "blockPuzzle": + $service = new BlockPuzzleCaptchaService($config); + break; + default: + throw new ParamException('captchaType参数不正确!'); + } + return $service; + } + + protected function validate() + { + return Request::instance()->validate([ + 'token' => 'bail|required', + 'pointJson' => 'required', + ]); + + } + + protected function success($data) + { + return [ + 'error' => false, + 'repCode' => '0000', + 'repData' => $data, + 'repMsg' => null, + 'success' => true, + ]; + } + + + protected function error($msg) + { + return [ + 'error' => true, + 'repCode' => '6111', + 'repData' => null, + 'repMsg' => $msg, + 'success' => false, + ]; + } + +} diff --git a/extend/fastknife/ajcaptcha/test/laravel/captcha.php b/extend/fastknife/ajcaptcha/test/laravel/captcha.php new file mode 100644 index 00000000..fe70e52c --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/laravel/captcha.php @@ -0,0 +1,33 @@ + '', //自定义字体包路径, 不填使用默认值 + //文字验证码 + 'click_world' => [ + 'backgrounds' => [] + ], + //滑动验证码 + 'block_puzzle' => [ + 'backgrounds' => [], //背景图片路径, 不填使用默认值 + 'templates' => [], //模板图 + 'offset' => 10, //容错偏移量 + ], + //水印 + 'watermark' => [ + 'fontsize' => 12, + 'color' => '#ffffff', + 'text' => '我的水印' + ], + 'cache' => [ + 'constructor' => [\Illuminate\Support\Facades\Cache::class, 'store'], + 'method' => [ + 'get' => 'get', //获取 + 'set' => 'set', //设置 + 'delete' => 'delete',//删除 + 'has' => 'has' //key是否存在 + ] + ] +]; diff --git a/extend/fastknife/ajcaptcha/test/readme.md b/extend/fastknife/ajcaptcha/test/readme.md new file mode 100644 index 00000000..b1d27860 --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/readme.md @@ -0,0 +1,3 @@ +#### 这个是一个demo测试目录。 +* 当前目录下的php文件是不基于任何框架的demo +* thinkphp 目录是基于thinkphp框架的demo diff --git a/extend/fastknife/ajcaptcha/test/testAes.php b/extend/fastknife/ajcaptcha/test/testAes.php new file mode 100644 index 00000000..312d5f53 --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/testAes.php @@ -0,0 +1,22 @@ + 135, 'y' => 58], + ['x' => 82, 'y' => 72], + ['x' => 56, 'y' => 112] +]; +$point = json_encode($point); +//[{"x":135,"y":58},{"x":82,"y":72},{"x":56,"y":112}] +//var_dump(); +//print_r(AesUtils::encrypt($point,$key)); +//php w2GGF3+0q0K0AdTysBEKynRo9hXBbXBPUZU1GPWJKlM4SwtrbmV17CFcTq/T53Kvlk0FWSbFzfCC1NuAA6wsmw== +//js FOl3kz52f4xptJ/Zf7MoZYQmYa7C1pjQaWP8QqhcX8FH43SvpCBPhaSqqMbE8D55ufhgjBVor01UZRH3uE6DNw== + + + + +//js +5s4V1MeDYk1jpvfwJACJA== +//php C01B0oArq8aMhlMmSdRbDA== diff --git a/extend/fastknife/ajcaptcha/test/testCache.php b/extend/fastknife/ajcaptcha/test/testCache.php new file mode 100644 index 00000000..a10b8225 --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/testCache.php @@ -0,0 +1,21 @@ +get('haha')); + +$cacheEntity->set('haha', 1, 60); + +var_dump($cacheEntity->has('haha')); + +var_dump($cacheEntity->get('haha')); + +$cacheEntity->delete('haha'); + +var_dump($cacheEntity->get('haha')); \ No newline at end of file diff --git a/extend/fastknife/ajcaptcha/test/testImage.php b/extend/fastknife/ajcaptcha/test/testImage.php new file mode 100644 index 00000000..8383506b --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/testImage.php @@ -0,0 +1,28 @@ +makeBlockImage(); + $blockImage->run(); + $blockImage->echo(); +} + +function showWord() +{ + global $config; + $factory = new Factory($config); + $blockImage = $factory->makeWordImage(); + $blockImage->run(); + $blockImage->echo(); +} + +showWord(); \ No newline at end of file diff --git a/extend/fastknife/ajcaptcha/test/thinkphp/Index.php b/extend/fastknife/ajcaptcha/test/thinkphp/Index.php new file mode 100644 index 00000000..545e1791 --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/thinkphp/Index.php @@ -0,0 +1,116 @@ +getCaptchaService(); + $data = $service->get(); + } catch (\Exception $e) { + $this->error($e->getMessage()); + } + $this->success($data); + } + + /** + * 一次验证 + */ + public function check() + { + $data = request()->post(); + try { + $this->validate($data); + $service = $this->getCaptchaService(); + $service->check($data['token'], $data['pointJson']); + } catch (\Exception $e) { + $this->error($e->getMessage()); + } + $this->success([]); + } + + /** + * 二次验证 + */ + public function verification() + { + $data = request()->post(); + try { + $this->validate($data); + $service = $this->getCaptchaService(); + $service->verification($data['token'], $data['pointJson']); + } catch (\Exception $e) { + $this->error($e->getMessage()); + } + $this->success([]); + } + + protected function getCaptchaService() + { + $captchaType = request()->post('captchaType', null); + $config = config('captcha'); + switch ($captchaType) { + case "clickWord": + $service = new ClickWordCaptchaService($config); + break; + case "blockPuzzle": + $service = new BlockPuzzleCaptchaService($config); + break; + default: + throw new ParamException('captchaType参数不正确!'); + } + return $service; + } + + protected function validate($data) + { + $rules = [ + 'token' => ['require'], + 'pointJson' => ['require'] + ]; + $validate = Validate::rule($rules)->failException(true); + $validate->check($data); + } + + protected function success($data) + { + $response = [ + 'error' => false, + 'repCode' => '0000', + 'repData' => $data, + 'repMsg' => null, + 'success' => true, + ]; + throw new HttpResponseException(Response::create($response, 'json')); + } + + + protected function error($msg) + { + $response = [ + 'error' => true, + 'repCode' => '6111', + 'repData' => null, + 'repMsg' => $msg, + 'success' => false, + ]; + throw new HttpResponseException(Response::create($response, 'json')); + } + + +} diff --git a/extend/fastknife/ajcaptcha/test/thinkphp/captcha.php b/extend/fastknife/ajcaptcha/test/thinkphp/captcha.php new file mode 100644 index 00000000..4d1dc199 --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/thinkphp/captcha.php @@ -0,0 +1,27 @@ + '', //自定义字体包路径, 不填使用默认值 + //文字验证码 + 'click_world' => [ + 'backgrounds' => [] + ], + //滑动验证码 + 'block_puzzle' => [ + 'backgrounds' => [], //背景图片路径, 不填使用默认值 + 'templates' => [], //模板图 + 'offset' => 10, //容错偏移量 + ], + //水印 + 'watermark' => [ + 'fontsize' => 12, + 'color' => '#ffffff', + 'text' => '我的水印' + ], + 'cache' => [ + 'constructor' => [\think\facade\Cache::class, 'instance'] + ] +]; diff --git a/extend/fastknife/ajcaptcha/test/verification.php b/extend/fastknife/ajcaptcha/test/verification.php new file mode 100644 index 00000000..12659ff6 --- /dev/null +++ b/extend/fastknife/ajcaptcha/test/verification.php @@ -0,0 +1,19 @@ +verification(); diff --git a/extend/joypack/tencent-map/.gitignore b/extend/joypack/tencent-map/.gitignore new file mode 100644 index 00000000..ab09565e --- /dev/null +++ b/extend/joypack/tencent-map/.gitignore @@ -0,0 +1,4 @@ +/.buildpath +/.project +/.settings/ +/logs/ diff --git a/extend/joypack/tencent-map/LICENSE b/extend/joypack/tencent-map/LICENSE new file mode 100644 index 00000000..261eeb9e --- /dev/null +++ b/extend/joypack/tencent-map/LICENSE @@ -0,0 +1,201 @@ + 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. diff --git a/extend/joypack/tencent-map/README.md b/extend/joypack/tencent-map/README.md new file mode 100644 index 00000000..e1440f8f --- /dev/null +++ b/extend/joypack/tencent-map/README.md @@ -0,0 +1,180 @@ +##### 腾讯位置服务 + +###### 地址解析(地址转坐标) +``` +// 命名空间 +use Joypack\Tencent\Map\WebService\AddressOption; +use Joypack\Tencent\Map\WebService\Address; + +// 实例化参数 +$option = new AddressOption(); +// 设置接口 key +$option->setKey(''); +// 如果使用签名方式校验则需要配置 secret +$option->setSecret(''); +// 设置要解析坐标的地址 +$option->setAddress('安徽省合肥市瑶海区方庙街道万达金街'); + +// 将参数在这里传递 +// 非开发模式只记录 error 类型的日志 +$address = new Address($option, <日志存储路径>, <是否开发模式>); + +// 授权IP校验方式通信(无sig参数) +// $res = $address->request(); + +// 通过签名校验的方式通信 +// 无需使用 $option->setSig() +$res = $address->request(true); + +// $res->logger->print($res, true); + +// 判断请求是否异常 +if($res->error) { + $res->logger->print($res->error, true); +} + +// 打印接口返回的原始数据 +// $res->logger->print($res->getOriginal(), true); + +// 判断接口返回状态 +if($res->status) { + // 打印接口返回信息 + $res->logger->print($res->message, true); +} + +// 打印接口返回数据(内部已完成Array解析) +$res->logger->print($res->result, true); +``` + +###### 逆地址解析(坐标位置描述) +``` +// 命名空间 +use Joypack\Tencent\Map\WebService\LocationOption; +use Joypack\Tencent\Map\WebService\Location; + +// 实例化参数 +$option = new LocationOption(); +// 设置接口 key +$option->setKey(''); +// 如果使用签名方式校验则需要配置 secret +$option->setSecret(''); +// 设置要解析地址的经纬度坐标 +$option->setLocation(31.877089, 117.347885); + +// 将参数在这里传递 +// 非开发模式只记录 error 类型的日志 +$location = new Location($option, LOG_PATH, true); + +// 授权IP校验方式通信(无sig参数) +// $res = $address->request(); + +// 通过签名校验的方式通信 +// 无需使用 $option->setSig() +$res = $location->request(true); + +// $res->logger->print($res, true); + +// 判断请求是否异常 +if($res->error) { + $res->logger->print($res->error, true); +} + +// 打印接口返回的原始数据 +// $res->logger->print($res->getOriginal(), true); + +// 判断接口返回状态 +if($res->status) { + // 打印接口返回信息 + $res->logger->print($res->message, true); +} + +// 打印接口返回数据(内部已完成Array解析) +$res->logger->print($res->result, true); +// 打印经纬度 +$res->logger->print($res->result['location']['lng']); +``` + +###### 坐标转换 +``` +// 命名空间 +use Joypack\Tencent\Map\WebService\TranslateOption; +use Joypack\Tencent\Map\WebService\Translate; + +// 实例化参数 +$option = new TranslateOption(); +$option->setKey(''); +$option->setSecret(''); +// 设置要转换的经纬度类型 +$option->setType($option::TYPE_BAIDU); +// 设置经要转换的经纬度 +$option->setLocation(31.877089, 117.347885); + +// +$location = new Translate($option, LOG_PATH, true); + +// 授权IP校验方式通信(无sig参数) +// $res = $address->request(); + +// 通过签名校验的方式通信 +// 无需使用 $option->setSig() +$res = $location->request(true); + +// $res->logger->print($res, true); + +// 判断请求是否异常 +if($res->error) { + $res->logger->print($res->error, true); +} + +// 打印接口返回的原始数据 +// $res->logger->print($res->getOriginal(), true); + +// 判断接口返回状态 +if($res->status) { + // 打印接口返回信息 + $res->logger->print($res->message, true); +} + +// 打印接口返回数据(内部已完成Array解析) +$res->logger->print($res->locations, true); +``` + +###### IP定位 +``` +// 命名空间 +use Joypack\Tencent\Map\WebService\IpOption; +use Joypack\Tencent\Map\WebService\Ip; + +$option = new IpOption(); +$option->setKey(''); +$option->setSecret(''); +$option->setIp('202.106.0.20'); + +$location = new Ip($option, LOG_PATH, true); + +// 授权IP校验方式通信(无sig参数) +// $res = $address->request(); + +// 通过签名校验的方式通信 +// 无需使用 $option->setSig() +$res = $location->request(true); + +// $res->logger->print($res, true); + +// 判断请求是否异常 +if($res->error) { + $res->logger->print($res->error, true); +} + +// 打印接口返回的原始数据 +// $res->logger->print($res->getOriginal(), true); + +// 判断接口返回状态 +if($res->status) { + // 打印接口返回信息 + $res->logger->print($res->message, true); +} + +// 打印接口返回数据(内部已完成Array解析) +$res->logger->print($res->result, true); +``` \ No newline at end of file diff --git a/extend/joypack/tencent-map/composer.json b/extend/joypack/tencent-map/composer.json new file mode 100644 index 00000000..fc76f922 --- /dev/null +++ b/extend/joypack/tencent-map/composer.json @@ -0,0 +1,33 @@ +{ + "name" : "joypack/tencent-map", + "description" : "腾讯位置服务(WebService)", + "type" : "library", + "version" : "1.0.0", + "keywords" : [ + "腾讯位置服务", + "lbs" + ], + "homepage" : "https://github.com/joypack/tencent-map", + "authors" : [{ + "name" : "堪笑", + "email" : "jixiang.f@gmail.com", + "role" : "Developer", + "homepage" : "http://cli.life" + } + ], + "require": { + "php": ">=5.6" + }, + "support" : { + "email" : "jixiang.f@gmail.com", + "issues" : "https://github.com/joypack/tencent-map/issues", + "wiki" : "https://github.com/joypack/tencent-map/wiki" + }, + "license": "Apache-2.0", + "autoload" : { + "psr-4" : { + "Joypack\\Tencent\\Map\\" : "src" + } + }, + "minimum-stability" : "stable" +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/examples/address.php b/extend/joypack/tencent-map/examples/address.php new file mode 100644 index 00000000..7a2dde8e --- /dev/null +++ b/extend/joypack/tencent-map/examples/address.php @@ -0,0 +1,49 @@ +setKey(''); +$option->setSecret(''); + +$option->setAddress('
'); + +$address = new Address($option, LOG_PATH, true); + +// 授权IP校验方式通信(无sig参数) +// $res = $address->request(); + +// 通过签名校验的方式通信 +// 无需使用 $option->setSig() +$res = $address->request(true); + +// $res->logger->print($res, true); + +// 判断请求是否异常 +if($res->error) { + $res->logger->print($res->error, true); +} + +// 打印接口返回的原始数据 +// $res->logger->print($res->getOriginal(), true); + +// 判断接口返回状态 +if($res->status) { + // 打印接口返回信息 + $res->logger->print($res->message, true); +} + +// 打印接口返回数据(内部已完成Array解析) +$res->logger->print($res->result, true); \ No newline at end of file diff --git a/extend/joypack/tencent-map/examples/ip.php b/extend/joypack/tencent-map/examples/ip.php new file mode 100644 index 00000000..c544b02c --- /dev/null +++ b/extend/joypack/tencent-map/examples/ip.php @@ -0,0 +1,50 @@ +setKey(''); +$option->setSecret(''); + +$option->setIp(''); + +$location = new Ip($option, LOG_PATH, true); + +// 授权IP校验方式通信(无sig参数) +// $res = $address->request(); + +// 通过签名校验的方式通信 +// 无需使用 $option->setSig() +$res = $location->request(true); + +// $res->logger->print($res, true); + +// 判断请求是否异常 +if($res->error) { + $res->logger->print($res->error, true); +} + +// 打印接口返回的原始数据 +// $res->logger->print($res->getOriginal(), true); + +// 判断接口返回状态 +if($res->status) { + // 打印接口返回信息 + $res->logger->print($res->message, true); +} + +// 打印接口返回数据(内部已完成Array解析) +$res->logger->print($res->result, true); diff --git a/extend/joypack/tencent-map/examples/location.php b/extend/joypack/tencent-map/examples/location.php new file mode 100644 index 00000000..16a1971c --- /dev/null +++ b/extend/joypack/tencent-map/examples/location.php @@ -0,0 +1,50 @@ +setKey(''); +$option->setSecret(''); + +$option->setLocation('', ''); + +$location = new Location($option, LOG_PATH, true); + +// 授权IP校验方式通信(无sig参数) +// $res = $address->request(); + +// 通过签名校验的方式通信 +// 无需使用 $option->setSig() +$res = $location->request(true); + +// $res->logger->print($res, true); + +// 判断请求是否异常 +if($res->error) { + $res->logger->print($res->error, true); +} + +// 打印接口返回的原始数据 +// $res->logger->print($res->getOriginal(), true); + +// 判断接口返回状态 +if($res->status) { + // 打印接口返回信息 + $res->logger->print($res->message, true); +} + +// 打印接口返回数据(内部已完成Array解析) +$res->logger->print($res->result, true); +// 打印经纬度 +$res->logger->print($res->result['location']['lng']); diff --git a/extend/joypack/tencent-map/examples/translate.php b/extend/joypack/tencent-map/examples/translate.php new file mode 100644 index 00000000..0d75e156 --- /dev/null +++ b/extend/joypack/tencent-map/examples/translate.php @@ -0,0 +1,50 @@ +setKey(''); +$option->setSecret(''); + +$option->setType($option::TYPE_BAIDU); +$option->setLocation('', ''); + +$location = new Translate($option, LOG_PATH, true); + +// 授权IP校验方式通信(无sig参数) +// $res = $address->request(); + +// 通过签名校验的方式通信 +// 无需使用 $option->setSig() +$res = $location->request(true); + +// $res->logger->print($res, true); + +// 判断请求是否异常 +if($res->error) { + $res->logger->print($res->error, true); +} + +// 打印接口返回的原始数据 +// $res->logger->print($res->getOriginal(), true); + +// 判断接口返回状态 +if($res->status) { + // 打印接口返回信息 + $res->logger->print($res->message, true); +} + +// 打印接口返回数据(内部已完成Array解析) +$res->logger->print($res->locations, true); diff --git a/extend/joypack/tencent-map/src/Bundle.php b/extend/joypack/tencent-map/src/Bundle.php new file mode 100644 index 00000000..6ad11167 --- /dev/null +++ b/extend/joypack/tencent-map/src/Bundle.php @@ -0,0 +1,30 @@ +option = $option; + // 实例化日志 + $this->logger = new Logger("{$log_root}/joypack-tencent-map", $development); + // 实例化请求 + $this->request = new Request($this->logger); + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Bundle/Address.php b/extend/joypack/tencent-map/src/Bundle/Address.php new file mode 100644 index 00000000..48270b1a --- /dev/null +++ b/extend/joypack/tencent-map/src/Bundle/Address.php @@ -0,0 +1,37 @@ +option->setSig($uri); + } + + $data = $this->option->getAll(); + + //$this->request->logger->print($data, true); + + $this->request->uri($uri); + $this->request->query($data); + + return $this->request->get(); + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Bundle/AddressOption.php b/extend/joypack/tencent-map/src/Bundle/AddressOption.php new file mode 100644 index 00000000..92a8cd57 --- /dev/null +++ b/extend/joypack/tencent-map/src/Bundle/AddressOption.php @@ -0,0 +1,29 @@ +option['address'] = $value; + } + + /** + * 指定地址所属城市 + * @param string $value + */ + public function setRegion($value) + { + $this->option['region'] = $value; + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Bundle/Ip.php b/extend/joypack/tencent-map/src/Bundle/Ip.php new file mode 100644 index 00000000..82699cef --- /dev/null +++ b/extend/joypack/tencent-map/src/Bundle/Ip.php @@ -0,0 +1,37 @@ +option->setSig($uri); + } + + $data = $this->option->getAll(); + + //$this->request->logger->print($data, true); + + $this->request->uri($uri); + $this->request->query($data); + + return $this->request->get(); + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Bundle/IpOption.php b/extend/joypack/tencent-map/src/Bundle/IpOption.php new file mode 100644 index 00000000..25098fc3 --- /dev/null +++ b/extend/joypack/tencent-map/src/Bundle/IpOption.php @@ -0,0 +1,20 @@ +option['ip'] = $value; + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Bundle/Location.php b/extend/joypack/tencent-map/src/Bundle/Location.php new file mode 100644 index 00000000..48b98423 --- /dev/null +++ b/extend/joypack/tencent-map/src/Bundle/Location.php @@ -0,0 +1,37 @@ +option->setSig($uri); + } + + $data = $this->option->getAll(); + + //$this->request->logger->print($data, true); + + $this->request->uri($uri); + $this->request->query($data); + + return $this->request->get(); + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Bundle/LocationOption.php b/extend/joypack/tencent-map/src/Bundle/LocationOption.php new file mode 100644 index 00000000..aa73145e --- /dev/null +++ b/extend/joypack/tencent-map/src/Bundle/LocationOption.php @@ -0,0 +1,16 @@ +option['location'] = "{$lat},{$lng}"; + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Bundle/Translate.php b/extend/joypack/tencent-map/src/Bundle/Translate.php new file mode 100644 index 00000000..b2c3da10 --- /dev/null +++ b/extend/joypack/tencent-map/src/Bundle/Translate.php @@ -0,0 +1,37 @@ +option->setSig($uri); + } + + $data = $this->option->getAll(); + + //$this->request->logger->print($data, true); + + $this->request->uri($uri); + $this->request->query($data); + + return $this->request->get(); + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Bundle/TranslateOption.php b/extend/joypack/tencent-map/src/Bundle/TranslateOption.php new file mode 100644 index 00000000..20bde014 --- /dev/null +++ b/extend/joypack/tencent-map/src/Bundle/TranslateOption.php @@ -0,0 +1,58 @@ +option['locations'] = "{$lat},{$lng}"; + } + + /** + * 预转换的坐标,支持批量转换 + * @param array $locations + *

['lat,lng', [lat,lng]]

+ */ + public function setLocations(array $locations) + { + $pieces = []; + foreach ($locations as $item) { + if(is_array($item)) { + $pieces[] = "{$item[0]},{$item[1]}"; + } + } + $this->option['locations'] = implode(';', $pieces); + } + + /** + * 设置坐标类型 + * @param number $value + * 1 GPS坐标 + * 2 sogou经纬度 + * 3 baidu经纬度 + * 4 mapbar经纬度 + * 5 腾讯、google、高德坐标[默认] + * 6 sogou墨卡托 + */ + public function setType($value=self::TYPE_DEFAULT) + { + $this->option['type'] = $value; + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Logger.php b/extend/joypack/tencent-map/src/Logger.php new file mode 100644 index 00000000..248eb9eb --- /dev/null +++ b/extend/joypack/tencent-map/src/Logger.php @@ -0,0 +1,121 @@ +development = $development; + + if($root_path) { + if(is_dir($root_path)) { + $this->rootPath = $root_path; + } else { + if(@mkdir($root_path, 0775, true)) { + $this->rootPath = $root_path; + } + } + } + } + + public function __toString() + { + return __CLASS__; + } + + /** + * 写入 debug 日志 + * @param string $message + * @param string | array $data + */ + public function debug($message, $data=null) + { + $this->save($message, $data, 'DEBUG'); + } + + /** + * 写入 INFO 日志 + * @param string $message + * @param string | array $data + */ + public function info($message, $data=null) + { + $this->save($message, $data, 'INFO'); + } + + /** + * 写入 ERROR 日志 + * @param string $message + * @param string | array $data + */ + public function error($message, $data=null) + { + $this->save($message, $data, 'ERROR'); + } + + /** + * 打印变量 + * @param mixed $args 打印列表 + * 最后一个元素如果是 true 则 exit + */ + public function print(...$args) + { + $args = func_get_args(); + + $length = count($args); + + $exit = false; + + if(is_bool($last_argument = $args[$length-1])) { + if($last_argument) { + $exit = true; + array_pop($args); + } + } + + echo '
';
+        while ($argument = array_shift($args)) {
+            print_r($argument);
+            echo '

'; + } + echo '
'; + + if($exit) { + exit; + } + } + + protected function save($message, $data, $level) + { + if(is_null($this->rootPath)) { + return; + } + + // 生产环境时只记录错误信息 + if(!$this->development) { + if($level != 'ERROR') { + return; + } + } + + $date = date('Y-m-d'); + + $now = date('Y-m-d H:i:s'); + + $filename = "{$this->rootPath}/{$date}.log"; + + if(is_array($data)) { + $data = json_encode($data, JSON_UNESCAPED_UNICODE); + } + + @file_put_contents($filename, "[{$level}] {$now} {$message} {$data}\r\n", FILE_APPEND); + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Option.php b/extend/joypack/tencent-map/src/Option.php new file mode 100644 index 00000000..21f64f2a --- /dev/null +++ b/extend/joypack/tencent-map/src/Option.php @@ -0,0 +1,98 @@ +setKey($key); + $this->setSecret($secret); + } + + public function setSecret($value) + { + $this->secret = $value; + } + + /** + * 开发密钥 + * @param string $value + */ + public function setKey($value) + { + $this->option['key'] = $value; + } + + /** + * 返回格式:支持JSON/JSONP,默认JSON + * @param string $value + */ + public function setOutput($value=self::OUTPUT_JSON) + { + $this->option['output'] = $value; + } + + /** + * JSONP方式回调函数 + * @param string $value + */ + public function setCallback($value) + { + $this->option['callback'] = $value; + } + + /** + * 签名 + * @param string $uri + */ + public function setSig($uri) + { + $this->option['sig'] = $this->buildSig($uri, $this->getAll()); + } + + /** + * 获得所有参数 + * @return array + */ + public function getAll() + { + return $this->option; + } + + /** + * 生成签名 + * @param string $uri + * @return string + */ + protected function buildSig($uri, $option) + { + ksort($option); + + $pieces = []; + foreach ($option as $key => $val) + { + $pieces[] = "{$key}={$val}"; + } + + $str = sprintf('%s?%s', rtrim($uri, '/'), implode('&', $pieces)); + + /* + echo '
';
+         print_r("{$str}{$this->secret}");
+         die;
+         //*/
+        
+        return md5("{$str}{$this->secret}");
+    }
+}
\ No newline at end of file
diff --git a/extend/joypack/tencent-map/src/Request.php b/extend/joypack/tencent-map/src/Request.php
new file mode 100644
index 00000000..78a8f7fe
--- /dev/null
+++ b/extend/joypack/tencent-map/src/Request.php
@@ -0,0 +1,181 @@
+logger = $logger;
+    }
+    
+    public function uri($uri)
+    {
+        $this->url .= '/' . trim($uri, '/');
+        return $this;
+    }
+    
+    public function query($name, $value=null)
+    {
+        if(is_array($name)) {
+            $this->query = array_merge($this->query, $name);
+        } else {
+            $this->query[$name] = $value;
+        }
+        return $this;
+    }
+    
+    public function field($name, $value=null)
+    {
+        if(is_array($name)) {
+            $this->field = array_merge($this->field, $name);
+        } else {
+            $this->field[$name] = $value;
+        }
+        return $this;
+    }
+    
+    /**
+     * method get
+     * @param array $query
+     * @return Response
+     */
+    public function get(array $query=[])
+    {
+        if($query) {
+            $this->query($query);
+        }
+        
+        $url = $this->mergeQuery($this->url, $this->query);
+        
+        return $this->create($url);
+    }
+    
+    public function post(array $fields=[])
+    {
+        if($fields) {
+            $this->field($fields);
+        }
+        
+        $url = $this->mergeQuery($this->url, $this->query);
+        
+        return $this->create($url, 'POST', $this->field);
+    }
+    
+    /**
+     * 

将参数合并到 URL

+ * @param string $url 请求的地址 + * @param array $query 请求参数 + * @param bool $recursive 是否递归合并 + * @return mixed + */ + protected function mergeQuery($url, array $query, bool $recursive=false) + { + if(empty($url)) { + return null; + } + + // 没有设置参数时直接返回地址 + if(empty($query)) { + return $url; + } + + $parsed = parse_url($url); + + // 合并参数 + if(isset($parsed['query'])) { + $url = substr($url, 0, strpos($url, '?')); + + $str_parsed = []; + + parse_str($parsed['query'], $str_parsed); + + if($recursive) { + $query = array_merge_recursive($str_parsed, $query); + } else { + $query = array_merge($str_parsed, $query); + } + } else { + $url = rtrim($url, '/?'); + } + + // 生成 query 字符串 + $url = "{$url}?" . http_build_query($query); + + return $url; + + // 处理锚点 + if(isset($parsed['fragment'])) { + $url .= "#{$parsed['fragment']}"; + } + + return $url; + } + + /** + *

创建请求

+ * @param string $url 请求地址 + * @param string $method 请求方式 + * @param array $data POST 数据 + * @return Response + */ + protected function create($url, $data=null) + { + //$referer = "{$_SERVER['REQUEST_SCHEME']}://{$_SERVER['SERVER_NAME']}"; + + //$header = [ + //"CLIENT-IP: {$_SERVER['REMOTE_ADDR']}", + //"X-FORWARDED-FOR: {$_SERVER['REMOTE_ADDR']}", + //"Content-Type: application/json; charset=utf-8", + //"Accept: */*" + //]; + + $ch = curl_init(); + + curl_setopt($ch, CURLOPT_URL, $url); + //curl_setopt($ch, CURLOPT_HEADER, $header); + curl_setopt($ch, CURLOPT_TIMEOUT, 10); + //curl_setopt($ch, CURLOPT_REFERER, $referer); + curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); + curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, 0); + curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 0); + + if(!is_null($data)) { + curl_setopt($ch, CURLOPT_POST, true); + if($data) { + curl_setopt($ch, CURLOPT_POSTFIELDS, $data); + } + } + + $original = curl_exec($ch); + + $error = null; + if($errno = curl_errno($ch)) { + $error = curl_error($ch); + } + + curl_close($ch); + + $this->logger->info('请求地址', $url); + + if($data) { + $this->logger->info('请求数据', $data); + } + + $this->logger->info('响应数据', $original); + + return new Response($errno, $error, $original, $this->logger); + } +} \ No newline at end of file diff --git a/extend/joypack/tencent-map/src/Response.php b/extend/joypack/tencent-map/src/Response.php new file mode 100644 index 00000000..39d1c956 --- /dev/null +++ b/extend/joypack/tencent-map/src/Response.php @@ -0,0 +1,159 @@ +logger = $logger; + + // 仅成功时 + if(0 === $errno) { + $decode = json_decode($original, true); + if(is_null($decode)) { + // 错误 + $this->setErrorMessage(99); + // 写入日志 + $logger->error('解析失败'); + } else { + $this->original = $original; + $this->decode = $decode; + } + } else { + // 错误 + $this->setErrorMessage($errno); + // 写入日志 + $logger->error($error); + } + } + + /** + * 返回json + * @return string + */ + public function getOriginal() + { + return $this->original; + } + + /** + * 返回数组 + * @return array + */ + public function toArray() + { + return $this->decode; + } + + /** + * 获得某属性时 + * @param string $prop_name + * @return mixed + */ + public function __get($property) + { + return $this->decode[ $property ] ?? null; + } + + public function __toString() + { + return $this->original; + } + + protected function setErrorMessage($errno) + { + $errors = [ + 1=> 'UNSUPPORTED_PROTOCOL', + 2=> 'FAILED_INIT', + 3=> 'URL_MALFORMAT', + 4=> 'URL_MALFORMAT_USER', + 5=> 'COULDNT_RESOLVE_PROXY', + 6=> 'COULDNT_RESOLVE_HOST', + 7=> 'COULDNT_CONNECT', + 8=> 'FTP_WEIRD_SERVER_REPLY', + 9=> 'REMOTE_ACCESS_DENIED', + 11=> 'FTP_WEIRD_PASS_REPLY', + 13=> 'FTP_WEIRD_PASV_REPLY', + 14=>'FTP_WEIRD_227_FORMAT', + 15=> 'FTP_CANT_GET_HOST', + 17=> 'FTP_COULDNT_SET_TYPE', + 18=> 'PARTIAL_FILE', + 19=> 'FTP_COULDNT_RETR_FILE', + 21=> 'QUOTE_ERROR', + 22=> 'HTTP_RETURNED_ERROR', + 23=> 'WRITE_ERROR', + 25=> 'UPLOAD_FAILED', + 26=> 'READ_ERROR', + 27=> 'OUT_OF_MEMORY', + 28=> 'OPERATION_TIMEDOUT', + 30=> 'FTP_PORT_FAILED', + 31=> 'FTP_COULDNT_USE_REST', + 33=> 'RANGE_ERROR', + 34=> 'HTTP_POST_ERROR', + 35=> 'SSL_CONNECT_ERROR', + 36=> 'BAD_DOWNLOAD_RESUME', + 37=> 'FILE_COULDNT_READ_FILE', + 38=> 'LDAP_CANNOT_BIND', + 39=> 'LDAP_SEARCH_FAILED', + 41=> 'FUNCTION_NOT_FOUND', + 42=> 'ABORTED_BY_CALLBACK', + 43=> 'BAD_FUNCTION_ARGUMENT', + 45=> 'INTERFACE_FAILED', + 47=> 'TOO_MANY_REDIRECTS', + 48=> 'UNKNOWN_TELNET_OPTION', + 49=> 'TELNET_OPTION_SYNTAX', + 51=> 'PEER_FAILED_VERIFICATION', + 52=> 'GOT_NOTHING', + 53=> 'SSL_ENGINE_NOTFOUND', + 54=> 'SSL_ENGINE_SETFAILED', + 55=> 'SEND_ERROR', + 56=> 'RECV_ERROR', + 58=> 'SSL_CERTPROBLEM', + 59=> 'SSL_CIPHER', + 60=> 'SSL_CACERT', + 61=> 'BAD_CONTENT_ENCODING', + 62=> 'LDAP_INVALID_URL', + 63=> 'FILESIZE_EXCEEDED', + 64=> 'USE_SSL_FAILED', + 65=> 'SEND_FAIL_REWIND', + 66=> 'SSL_ENGINE_INITFAILED', + 67=> 'LOGIN_DENIED', + 68=> 'TFTP_NOTFOUND', + 69=> 'TFTP_PERM', + 70=> 'REMOTE_DISK_FULL', + 71=> 'TFTP_ILLEGAL', + 72=> 'TFTP_UNKNOWNID', + 73=> 'REMOTE_FILE_EXISTS', + 74=> 'TFTP_NOSUCHUSER', + 75=> 'CONV_FAILED', + 76=> 'CONV_REQD', + 77=> 'SSL_CACERT_BADFILE', + 78=> 'REMOTE_FILE_NOT_FOUND', + 79=> 'SSH', + 80=> 'SSL_SHUTDOWN_FAILED', + 81=> 'AGAIN', + 82=> 'SSL_CRL_BADFILE', + 83=> 'SSL_ISSUER_ERROR', + 84=> 'FTP_PRET_FAILED', + 84=> 'FTP_PRET_FAILED', + 85=> 'RTSP_CSEQ_ERROR', + 86=> 'RTSP_SESSION_ERROR', + 87=> 'FTP_BAD_FILE_LIST', + 88=> 'CHUNK_FAILED', + 99=> 'DECODE_ERROR', + ]; + $this->error = $errors[ $errno ] ?? 'UNKNOWN_ERROR'; + } +} \ No newline at end of file