[
  {
    "path": ".github/FUNDING.yml",
    "content": "github: vil\n"
  },
  {
    "path": ".github/workflows/main.yml",
    "content": "name: Build executables\n\non:\n  workflow_dispatch: # Manually triggered event\n  pull_request: # Trigger the workflow on push or pull request,\n\njobs:\n  build_windows:\n    runs-on: windows-latest\n    steps:\n      - uses: actions/checkout@v4\n      - name: Set up Python 3.12\n        uses: actions/setup-python@v5\n        with:\n          python-version: 3.12\n      - name: Install dependencies\n        run: |\n          python -m pip install --upgrade pip\n          pip install -r requirements.txt\n      - name: Build Windows executable with pyinstaller\n        run: |\n          pyinstaller h4xtools.py --add-data \"resources/*;resources\" --onefile -F --clean\n      - name: Calculate SHA256 hash\n        run: |\n          sha256sum dist/h4xtools.exe > dist/h4xtools.exe.hash\n      - name: Upload Windows artifact\n        uses: actions/upload-artifact@v4\n        with:\n          name: Windows Executable\n          path: dist/\n\n  build_linux:\n    runs-on: ubuntu-latest\n    steps:\n      - uses: actions/checkout@v4\n      - name: Set up Python 3.12\n        uses: actions/setup-python@v5\n        with:\n          python-version: 3.12\n      - name: Install dependencies\n        run: |\n          python -m pip install --upgrade pip\n          pip install -r requirements.txt\n      - name: Build Linux executable with pyinstaller\n        run: |\n          pyinstaller h4xtools.py --add-data \"resources/*:resources\" --onefile -F --clean\n      - name: Calculate SHA256 hash\n        run: |\n          sha256sum dist/h4xtools > dist/h4xtools.hash\n      - name: Upload Linux artifact\n        uses: actions/upload-artifact@v4\n        with:\n          name: Linux Executable\n          path: dist/\n"
  },
  {
    "path": ".gitignore",
    "content": "__pycache__\nbuild\ndist\nscraped_data\nvenv\n.venv\n.env\n.VC.opendb\n.vc"
  },
  {
    "path": "LICENSE",
    "content": "                    GNU GENERAL PUBLIC LICENSE\n                       Version 3, 29 June 2007\n\n Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>\n Everyone is permitted to copy and distribute verbatim copies\n of this license document, but changing it is not allowed.\n\n                            Preamble\n\n  The GNU General Public License is a free, copyleft license for\nsoftware and other kinds of works.\n\n  The licenses for most software and other practical works are designed\nto take away your freedom to share and change the works.  By contrast,\nthe GNU General Public License is intended to guarantee your freedom to\nshare and change all versions of a program--to make sure it remains free\nsoftware for all its users.  We, the Free Software Foundation, use the\nGNU General Public License for most of our software; it applies also to\nany other work released this way by its authors.  You can apply it to\nyour programs, too.\n\n  When we speak of free software, we are referring to freedom, not\nprice.  Our General Public Licenses are designed to make sure that you\nhave the freedom to distribute copies of free software (and charge for\nthem if you wish), that you receive source code or can get it if you\nwant it, that you can change the software or use pieces of it in new\nfree programs, and that you know you can do these things.\n\n  To protect your rights, we need to prevent others from denying you\nthese rights or asking you to surrender the rights.  Therefore, you have\ncertain responsibilities if you distribute copies of the software, or if\nyou modify it: responsibilities to respect the freedom of others.\n\n  For example, if you distribute copies of such a program, whether\ngratis or for a fee, you must pass on to the recipients the same\nfreedoms that you received.  You must make sure that they, too, receive\nor can get the source code.  And you must show them these terms so they\nknow their rights.\n\n  Developers that use the GNU GPL protect your rights with two steps:\n(1) assert copyright on the software, and (2) offer you this License\ngiving you legal permission to copy, distribute and/or modify it.\n\n  For the developers' and authors' protection, the GPL clearly explains\nthat there is no warranty for this free software.  For both users' and\nauthors' sake, the GPL requires that modified versions be marked as\nchanged, so that their problems will not be attributed erroneously to\nauthors of previous versions.\n\n  Some devices are designed to deny users access to install or run\nmodified versions of the software inside them, although the manufacturer\ncan do so.  This is fundamentally incompatible with the aim of\nprotecting users' freedom to change the software.  The systematic\npattern of such abuse occurs in the area of products for individuals to\nuse, which is precisely where it is most unacceptable.  Therefore, we\nhave designed this version of the GPL to prohibit the practice for those\nproducts.  If such problems arise substantially in other domains, we\nstand ready to extend this provision to those domains in future versions\nof the GPL, as needed to protect the freedom of users.\n\n  Finally, every program is threatened constantly by software patents.\nStates should not allow patents to restrict development and use of\nsoftware on general-purpose computers, but in those that do, we wish to\navoid the special danger that patents applied to a free program could\nmake it effectively proprietary.  To prevent this, the GPL assures that\npatents cannot be used to render the program non-free.\n\n  The precise terms and conditions for copying, distribution and\nmodification follow.\n\n                       TERMS AND CONDITIONS\n\n  0. Definitions.\n\n  \"This License\" refers to version 3 of the GNU General Public License.\n\n  \"Copyright\" also means copyright-like laws that apply to other kinds of\nworks, such as semiconductor masks.\n\n  \"The Program\" refers to any copyrightable work licensed under this\nLicense.  Each licensee is addressed as \"you\".  \"Licensees\" and\n\"recipients\" may be individuals or organizations.\n\n  To \"modify\" a work means to copy from or adapt all or part of the work\nin a fashion requiring copyright permission, other than the making of an\nexact copy.  The resulting work is called a \"modified version\" of the\nearlier work or a work \"based on\" the earlier work.\n\n  A \"covered work\" means either the unmodified Program or a work based\non the Program.\n\n  To \"propagate\" a work means to do anything with it that, without\npermission, would make you directly or secondarily liable for\ninfringement under applicable copyright law, except executing it on a\ncomputer or modifying a private copy.  Propagation includes copying,\ndistribution (with or without modification), making available to the\npublic, and in some countries other activities as well.\n\n  To \"convey\" a work means any kind of propagation that enables other\nparties to make or receive copies.  Mere interaction with a user through\na computer network, with no transfer of a copy, is not conveying.\n\n  An interactive user interface displays \"Appropriate Legal Notices\"\nto the extent that it includes a convenient and prominently visible\nfeature that (1) displays an appropriate copyright notice, and (2)\ntells the user that there is no warranty for the work (except to the\nextent that warranties are provided), that licensees may convey the\nwork under this License, and how to view a copy of this License.  If\nthe interface presents a list of user commands or options, such as a\nmenu, a prominent item in the list meets this criterion.\n\n  1. Source Code.\n\n  The \"source code\" for a work means the preferred form of the work\nfor making modifications to it.  \"Object code\" means any non-source\nform of a work.\n\n  A \"Standard Interface\" means an interface that either is an official\nstandard defined by a recognized standards body, or, in the case of\ninterfaces specified for a particular programming language, one that\nis widely used among developers working in that language.\n\n  The \"System Libraries\" of an executable work include anything, other\nthan the work as a whole, that (a) is included in the normal form of\npackaging a Major Component, but which is not part of that Major\nComponent, and (b) serves only to enable use of the work with that\nMajor Component, or to implement a Standard Interface for which an\nimplementation is available to the public in source code form.  A\n\"Major Component\", in this context, means a major essential component\n(kernel, window system, and so on) of the specific operating system\n(if any) on which the executable work runs, or a compiler used to\nproduce the work, or an object code interpreter used to run it.\n\n  The \"Corresponding Source\" for a work in object code form means all\nthe source code needed to generate, install, and (for an executable\nwork) run the object code and to modify the work, including scripts to\ncontrol those activities.  However, it does not include the work's\nSystem Libraries, or general-purpose tools or generally available free\nprograms which are used unmodified in performing those activities but\nwhich are not part of the work.  For example, Corresponding Source\nincludes interface definition files associated with source files for\nthe work, and the source code for shared libraries and dynamically\nlinked subprograms that the work is specifically designed to require,\nsuch as by intimate data communication or control flow between those\nsubprograms and other parts of the work.\n\n  The Corresponding Source need not include anything that users\ncan regenerate automatically from other parts of the Corresponding\nSource.\n\n  The Corresponding Source for a work in source code form is that\nsame work.\n\n  2. Basic Permissions.\n\n  All rights granted under this License are granted for the term of\ncopyright on the Program, and are irrevocable provided the stated\nconditions are met.  This License explicitly affirms your unlimited\npermission to run the unmodified Program.  The output from running a\ncovered work is covered by this License only if the output, given its\ncontent, constitutes a covered work.  This License acknowledges your\nrights of fair use or other equivalent, as provided by copyright law.\n\n  You may make, run and propagate covered works that you do not\nconvey, without conditions so long as your license otherwise remains\nin force.  You may convey covered works to others for the sole purpose\nof having them make modifications exclusively for you, or provide you\nwith facilities for running those works, provided that you comply with\nthe terms of this License in conveying all material for which you do\nnot control copyright.  Those thus making or running the covered works\nfor you must do so exclusively on your behalf, under your direction\nand control, on terms that prohibit them from making any copies of\nyour copyrighted material outside their relationship with you.\n\n  Conveying under any other circumstances is permitted solely under\nthe conditions stated below.  Sublicensing is not allowed; section 10\nmakes it unnecessary.\n\n  3. Protecting Users' Legal Rights From Anti-Circumvention Law.\n\n  No covered work shall be deemed part of an effective technological\nmeasure under any applicable law fulfilling obligations under article\n11 of the WIPO copyright treaty adopted on 20 December 1996, or\nsimilar laws prohibiting or restricting circumvention of such\nmeasures.\n\n  When you convey a covered work, you waive any legal power to forbid\ncircumvention of technological measures to the extent such circumvention\nis effected by exercising rights under this License with respect to\nthe covered work, and you disclaim any intention to limit operation or\nmodification of the work as a means of enforcing, against the work's\nusers, your or third parties' legal rights to forbid circumvention of\ntechnological measures.\n\n  4. Conveying Verbatim Copies.\n\n  You may convey verbatim copies of the Program's source code as you\nreceive it, in any medium, provided that you conspicuously and\nappropriately publish on each copy an appropriate copyright notice;\nkeep intact all notices stating that this License and any\nnon-permissive terms added in accord with section 7 apply to the code;\nkeep intact all notices of the absence of any warranty; and give all\nrecipients a copy of this License along with the Program.\n\n  You may charge any price or no price for each copy that you convey,\nand you may offer support or warranty protection for a fee.\n\n  5. Conveying Modified Source Versions.\n\n  You may convey a work based on the Program, or the modifications to\nproduce it from the Program, in the form of source code under the\nterms of section 4, provided that you also meet all of these conditions:\n\n    a) The work must carry prominent notices stating that you modified\n    it, and giving a relevant date.\n\n    b) The work must carry prominent notices stating that it is\n    released under this License and any conditions added under section\n    7.  This requirement modifies the requirement in section 4 to\n    \"keep intact all notices\".\n\n    c) You must license the entire work, as a whole, under this\n    License to anyone who comes into possession of a copy.  This\n    License will therefore apply, along with any applicable section 7\n    additional terms, to the whole of the work, and all its parts,\n    regardless of how they are packaged.  This License gives no\n    permission to license the work in any other way, but it does not\n    invalidate such permission if you have separately received it.\n\n    d) If the work has interactive user interfaces, each must display\n    Appropriate Legal Notices; however, if the Program has interactive\n    interfaces that do not display Appropriate Legal Notices, your\n    work need not make them do so.\n\n  A compilation of a covered work with other separate and independent\nworks, which are not by their nature extensions of the covered work,\nand which are not combined with it such as to form a larger program,\nin or on a volume of a storage or distribution medium, is called an\n\"aggregate\" if the compilation and its resulting copyright are not\nused to limit the access or legal rights of the compilation's users\nbeyond what the individual works permit.  Inclusion of a covered work\nin an aggregate does not cause this License to apply to the other\nparts of the aggregate.\n\n  6. Conveying Non-Source Forms.\n\n  You may convey a covered work in object code form under the terms\nof sections 4 and 5, provided that you also convey the\nmachine-readable Corresponding Source under the terms of this License,\nin one of these ways:\n\n    a) Convey the object code in, or embodied in, a physical product\n    (including a physical distribution medium), accompanied by the\n    Corresponding Source fixed on a durable physical medium\n    customarily used for software interchange.\n\n    b) Convey the object code in, or embodied in, a physical product\n    (including a physical distribution medium), accompanied by a\n    written offer, valid for at least three years and valid for as\n    long as you offer spare parts or customer support for that product\n    model, to give anyone who possesses the object code either (1) a\n    copy of the Corresponding Source for all the software in the\n    product that is covered by this License, on a durable physical\n    medium customarily used for software interchange, for a price no\n    more than your reasonable cost of physically performing this\n    conveying of source, or (2) access to copy the\n    Corresponding Source from a network server at no charge.\n\n    c) Convey individual copies of the object code with a copy of the\n    written offer to provide the Corresponding Source.  This\n    alternative is allowed only occasionally and noncommercially, and\n    only if you received the object code with such an offer, in accord\n    with subsection 6b.\n\n    d) Convey the object code by offering access from a designated\n    place (gratis or for a charge), and offer equivalent access to the\n    Corresponding Source in the same way through the same place at no\n    further charge.  You need not require recipients to copy the\n    Corresponding Source along with the object code.  If the place to\n    copy the object code is a network server, the Corresponding Source\n    may be on a different server (operated by you or a third party)\n    that supports equivalent copying facilities, provided you maintain\n    clear directions next to the object code saying where to find the\n    Corresponding Source.  Regardless of what server hosts the\n    Corresponding Source, you remain obligated to ensure that it is\n    available for as long as needed to satisfy these requirements.\n\n    e) Convey the object code using peer-to-peer transmission, provided\n    you inform other peers where the object code and Corresponding\n    Source of the work are being offered to the general public at no\n    charge under subsection 6d.\n\n  A separable portion of the object code, whose source code is excluded\nfrom the Corresponding Source as a System Library, need not be\nincluded in conveying the object code work.\n\n  A \"User Product\" is either (1) a \"consumer product\", which means any\ntangible personal property which is normally used for personal, family,\nor household purposes, or (2) anything designed or sold for incorporation\ninto a dwelling.  In determining whether a product is a consumer product,\ndoubtful cases shall be resolved in favor of coverage.  For a particular\nproduct received by a particular user, \"normally used\" refers to a\ntypical or common use of that class of product, regardless of the status\nof the particular user or of the way in which the particular user\nactually uses, or expects or is expected to use, the product.  A product\nis a consumer product regardless of whether the product has substantial\ncommercial, industrial or non-consumer uses, unless such uses represent\nthe only significant mode of use of the product.\n\n  \"Installation Information\" for a User Product means any methods,\nprocedures, authorization keys, or other information required to install\nand execute modified versions of a covered work in that User Product from\na modified version of its Corresponding Source.  The information must\nsuffice to ensure that the continued functioning of the modified object\ncode is in no case prevented or interfered with solely because\nmodification has been made.\n\n  If you convey an object code work under this section in, or with, or\nspecifically for use in, a User Product, and the conveying occurs as\npart of a transaction in which the right of possession and use of the\nUser Product is transferred to the recipient in perpetuity or for a\nfixed term (regardless of how the transaction is characterized), the\nCorresponding Source conveyed under this section must be accompanied\nby the Installation Information.  But this requirement does not apply\nif neither you nor any third party retains the ability to install\nmodified object code on the User Product (for example, the work has\nbeen installed in ROM).\n\n  The requirement to provide Installation Information does not include a\nrequirement to continue to provide support service, warranty, or updates\nfor a work that has been modified or installed by the recipient, or for\nthe User Product in which it has been modified or installed.  Access to a\nnetwork may be denied when the modification itself materially and\nadversely affects the operation of the network or violates the rules and\nprotocols for communication across the network.\n\n  Corresponding Source conveyed, and Installation Information provided,\nin accord with this section must be in a format that is publicly\ndocumented (and with an implementation available to the public in\nsource code form), and must require no special password or key for\nunpacking, reading or copying.\n\n  7. Additional Terms.\n\n  \"Additional permissions\" are terms that supplement the terms of this\nLicense by making exceptions from one or more of its conditions.\nAdditional permissions that are applicable to the entire Program shall\nbe treated as though they were included in this License, to the extent\nthat they are valid under applicable law.  If additional permissions\napply only to part of the Program, that part may be used separately\nunder those permissions, but the entire Program remains governed by\nthis License without regard to the additional permissions.\n\n  When you convey a copy of a covered work, you may at your option\nremove any additional permissions from that copy, or from any part of\nit.  (Additional permissions may be written to require their own\nremoval in certain cases when you modify the work.)  You may place\nadditional permissions on material, added by you to a covered work,\nfor which you have or can give appropriate copyright permission.\n\n  Notwithstanding any other provision of this License, for material you\nadd to a covered work, you may (if authorized by the copyright holders of\nthat material) supplement the terms of this License with terms:\n\n    a) Disclaiming warranty or limiting liability differently from the\n    terms of sections 15 and 16 of this License; or\n\n    b) Requiring preservation of specified reasonable legal notices or\n    author attributions in that material or in the Appropriate Legal\n    Notices displayed by works containing it; or\n\n    c) Prohibiting misrepresentation of the origin of that material, or\n    requiring that modified versions of such material be marked in\n    reasonable ways as different from the original version; or\n\n    d) Limiting the use for publicity purposes of names of licensors or\n    authors of the material; or\n\n    e) Declining to grant rights under trademark law for use of some\n    trade names, trademarks, or service marks; or\n\n    f) Requiring indemnification of licensors and authors of that\n    material by anyone who conveys the material (or modified versions of\n    it) with contractual assumptions of liability to the recipient, for\n    any liability that these contractual assumptions directly impose on\n    those licensors and authors.\n\n  All other non-permissive additional terms are considered \"further\nrestrictions\" within the meaning of section 10.  If the Program as you\nreceived it, or any part of it, contains a notice stating that it is\ngoverned by this License along with a term that is a further\nrestriction, you may remove that term.  If a license document contains\na further restriction but permits relicensing or conveying under this\nLicense, you may add to a covered work material governed by the terms\nof that license document, provided that the further restriction does\nnot survive such relicensing or conveying.\n\n  If you add terms to a covered work in accord with this section, you\nmust place, in the relevant source files, a statement of the\nadditional terms that apply to those files, or a notice indicating\nwhere to find the applicable terms.\n\n  Additional terms, permissive or non-permissive, may be stated in the\nform of a separately written license, or stated as exceptions;\nthe above requirements apply either way.\n\n  8. Termination.\n\n  You may not propagate or modify a covered work except as expressly\nprovided under this License.  Any attempt otherwise to propagate or\nmodify it is void, and will automatically terminate your rights under\nthis License (including any patent licenses granted under the third\nparagraph of section 11).\n\n  However, if you cease all violation of this License, then your\nlicense from a particular copyright holder is reinstated (a)\nprovisionally, unless and until the copyright holder explicitly and\nfinally terminates your license, and (b) permanently, if the copyright\nholder fails to notify you of the violation by some reasonable means\nprior to 60 days after the cessation.\n\n  Moreover, your license from a particular copyright holder is\nreinstated permanently if the copyright holder notifies you of the\nviolation by some reasonable means, this is the first time you have\nreceived notice of violation of this License (for any work) from that\ncopyright holder, and you cure the violation prior to 30 days after\nyour receipt of the notice.\n\n  Termination of your rights under this section does not terminate the\nlicenses of parties who have received copies or rights from you under\nthis License.  If your rights have been terminated and not permanently\nreinstated, you do not qualify to receive new licenses for the same\nmaterial under section 10.\n\n  9. Acceptance Not Required for Having Copies.\n\n  You are not required to accept this License in order to receive or\nrun a copy of the Program.  Ancillary propagation of a covered work\noccurring solely as a consequence of using peer-to-peer transmission\nto receive a copy likewise does not require acceptance.  However,\nnothing other than this License grants you permission to propagate or\nmodify any covered work.  These actions infringe copyright if you do\nnot accept this License.  Therefore, by modifying or propagating a\ncovered work, you indicate your acceptance of this License to do so.\n\n  10. Automatic Licensing of Downstream Recipients.\n\n  Each time you convey a covered work, the recipient automatically\nreceives a license from the original licensors, to run, modify and\npropagate that work, subject to this License.  You are not responsible\nfor enforcing compliance by third parties with this License.\n\n  An \"entity transaction\" is a transaction transferring control of an\norganization, or substantially all assets of one, or subdividing an\norganization, or merging organizations.  If propagation of a covered\nwork results from an entity transaction, each party to that\ntransaction who receives a copy of the work also receives whatever\nlicenses to the work the party's predecessor in interest had or could\ngive under the previous paragraph, plus a right to possession of the\nCorresponding Source of the work from the predecessor in interest, if\nthe predecessor has it or can get it with reasonable efforts.\n\n  You may not impose any further restrictions on the exercise of the\nrights granted or affirmed under this License.  For example, you may\nnot impose a license fee, royalty, or other charge for exercise of\nrights granted under this License, and you may not initiate litigation\n(including a cross-claim or counterclaim in a lawsuit) alleging that\nany patent claim is infringed by making, using, selling, offering for\nsale, or importing the Program or any portion of it.\n\n  11. Patents.\n\n  A \"contributor\" is a copyright holder who authorizes use under this\nLicense of the Program or a work on which the Program is based.  The\nwork thus licensed is called the contributor's \"contributor version\".\n\n  A contributor's \"essential patent claims\" are all patent claims\nowned or controlled by the contributor, whether already acquired or\nhereafter acquired, that would be infringed by some manner, permitted\nby this License, of making, using, or selling its contributor version,\nbut do not include claims that would be infringed only as a\nconsequence of further modification of the contributor version.  For\npurposes of this definition, \"control\" includes the right to grant\npatent sublicenses in a manner consistent with the requirements of\nthis License.\n\n  Each contributor grants you a non-exclusive, worldwide, royalty-free\npatent license under the contributor's essential patent claims, to\nmake, use, sell, offer for sale, import and otherwise run, modify and\npropagate the contents of its contributor version.\n\n  In the following three paragraphs, a \"patent license\" is any express\nagreement or commitment, however denominated, not to enforce a patent\n(such as an express permission to practice a patent or covenant not to\nsue for patent infringement).  To \"grant\" such a patent license to a\nparty means to make such an agreement or commitment not to enforce a\npatent against the party.\n\n  If you convey a covered work, knowingly relying on a patent license,\nand the Corresponding Source of the work is not available for anyone\nto copy, free of charge and under the terms of this License, through a\npublicly available network server or other readily accessible means,\nthen you must either (1) cause the Corresponding Source to be so\navailable, or (2) arrange to deprive yourself of the benefit of the\npatent license for this particular work, or (3) arrange, in a manner\nconsistent with the requirements of this License, to extend the patent\nlicense to downstream recipients.  \"Knowingly relying\" means you have\nactual knowledge that, but for the patent license, your conveying the\ncovered work in a country, or your recipient's use of the covered work\nin a country, would infringe one or more identifiable patents in that\ncountry that you have reason to believe are valid.\n\n  If, pursuant to or in connection with a single transaction or\narrangement, you convey, or propagate by procuring conveyance of, a\ncovered work, and grant a patent license to some of the parties\nreceiving the covered work authorizing them to use, propagate, modify\nor convey a specific copy of the covered work, then the patent license\nyou grant is automatically extended to all recipients of the covered\nwork and works based on it.\n\n  A patent license is \"discriminatory\" if it does not include within\nthe scope of its coverage, prohibits the exercise of, or is\nconditioned on the non-exercise of one or more of the rights that are\nspecifically granted under this License.  You may not convey a covered\nwork if you are a party to an arrangement with a third party that is\nin the business of distributing software, under which you make payment\nto the third party based on the extent of your activity of conveying\nthe work, and under which the third party grants, to any of the\nparties who would receive the covered work from you, a discriminatory\npatent license (a) in connection with copies of the covered work\nconveyed by you (or copies made from those copies), or (b) primarily\nfor and in connection with specific products or compilations that\ncontain the covered work, unless you entered into that arrangement,\nor that patent license was granted, prior to 28 March 2007.\n\n  Nothing in this License shall be construed as excluding or limiting\nany implied license or other defenses to infringement that may\notherwise be available to you under applicable patent law.\n\n  12. No Surrender of Others' Freedom.\n\n  If conditions are imposed on you (whether by court order, agreement or\notherwise) that contradict the conditions of this License, they do not\nexcuse you from the conditions of this License.  If you cannot convey a\ncovered work so as to satisfy simultaneously your obligations under this\nLicense and any other pertinent obligations, then as a consequence you may\nnot convey it at all.  For example, if you agree to terms that obligate you\nto collect a royalty for further conveying from those to whom you convey\nthe Program, the only way you could satisfy both those terms and this\nLicense would be to refrain entirely from conveying the Program.\n\n  13. Use with the GNU Affero General Public License.\n\n  Notwithstanding any other provision of this License, you have\npermission to link or combine any covered work with a work licensed\nunder version 3 of the GNU Affero General Public License into a single\ncombined work, and to convey the resulting work.  The terms of this\nLicense will continue to apply to the part which is the covered work,\nbut the special requirements of the GNU Affero General Public License,\nsection 13, concerning interaction through a network will apply to the\ncombination as such.\n\n  14. Revised Versions of this License.\n\n  The Free Software Foundation may publish revised and/or new versions of\nthe GNU General Public License from time to time.  Such new versions will\nbe similar in spirit to the present version, but may differ in detail to\naddress new problems or concerns.\n\n  Each version is given a distinguishing version number.  If the\nProgram specifies that a certain numbered version of the GNU General\nPublic License \"or any later version\" applies to it, you have the\noption of following the terms and conditions either of that numbered\nversion or of any later version published by the Free Software\nFoundation.  If the Program does not specify a version number of the\nGNU General Public License, you may choose any version ever published\nby the Free Software Foundation.\n\n  If the Program specifies that a proxy can decide which future\nversions of the GNU General Public License can be used, that proxy's\npublic statement of acceptance of a version permanently authorizes you\nto choose that version for the Program.\n\n  Later license versions may give you additional or different\npermissions.  However, no additional obligations are imposed on any\nauthor or copyright holder as a result of your choosing to follow a\nlater version.\n\n  15. Disclaimer of Warranty.\n\n  THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY\nAPPLICABLE LAW.  EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT\nHOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM \"AS IS\" WITHOUT WARRANTY\nOF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,\nTHE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR\nPURPOSE.  THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM\nIS WITH YOU.  SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF\nALL NECESSARY SERVICING, REPAIR OR CORRECTION.\n\n  16. Limitation of Liability.\n\n  IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING\nWILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS\nTHE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY\nGENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE\nUSE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF\nDATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD\nPARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),\nEVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF\nSUCH DAMAGES.\n\n  17. Interpretation of Sections 15 and 16.\n\n  If the disclaimer of warranty and limitation of liability provided\nabove cannot be given local legal effect according to their terms,\nreviewing courts shall apply local law that most closely approximates\nan absolute waiver of all civil liability in connection with the\nProgram, unless a warranty or assumption of liability accompanies a\ncopy of the Program in return for a fee.\n\n                     END OF TERMS AND CONDITIONS\n\n            How to Apply These Terms to Your New Programs\n\n  If you develop a new program, and you want it to be of the greatest\npossible use to the public, the best way to achieve this is to make it\nfree software which everyone can redistribute and change under these terms.\n\n  To do so, attach the following notices to the program.  It is safest\nto attach them to the start of each source file to most effectively\nstate the exclusion of warranty; and each file should have at least\nthe \"copyright\" line and a pointer to where the full notice is found.\n\n    <one line to give the program's name and a brief idea of what it does.>\n    Copyright (C) <year>  <name of author>\n\n    This program is free software: you can redistribute it and/or modify\n    it under the terms of the GNU General Public License as published by\n    the Free Software Foundation, either version 3 of the License, or\n    (at your option) any later version.\n\n    This program is distributed in the hope that it will be useful,\n    but WITHOUT ANY WARRANTY; without even the implied warranty of\n    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n    GNU General Public License for more details.\n\n    You should have received a copy of the GNU General Public License\n    along with this program.  If not, see <https://www.gnu.org/licenses/>.\n\nAlso add information on how to contact you by electronic and paper mail.\n\n  If the program does terminal interaction, make it output a short\nnotice like this when it starts in an interactive mode:\n\n    <program>  Copyright (C) <year>  <name of author>\n    This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.\n    This is free software, and you are welcome to redistribute it\n    under certain conditions; type `show c' for details.\n\nThe hypothetical commands `show w' and `show c' should show the appropriate\nparts of the General Public License.  Of course, your program's commands\nmight be different; for a GUI interface, you would use an \"about box\".\n\n  You should also get your employer (if you work as a programmer) or school,\nif any, to sign a \"copyright disclaimer\" for the program, if necessary.\nFor more information on this, and how to apply and follow the GNU GPL, see\n<https://www.gnu.org/licenses/>.\n\n  The GNU General Public License does not permit incorporating your program\ninto proprietary programs.  If your program is a subroutine library, you\nmay consider it more useful to permit linking proprietary applications with\nthe library.  If this is what you want to do, use the GNU Lesser General\nPublic License instead of this License.  But first, please read\n<https://www.gnu.org/licenses/why-not-lgpl.html>.\n"
  },
  {
    "path": "README.md",
    "content": "# H4X-Tools\n\n[![GitHub latest commit](https://badgen.net/github/last-commit/vil/H4X-Tools)](https://GitHub.com/vil/H4X-Tools/commit/)\n[![GitHub commits](https://badgen.net/github/commits/vil/H4X-Tools)](https://GitHub.com/vil/H4X-Tools/commit/)\n[![GitHub stars](https://badgen.net/github/stars/vil/H4X-Tools)](https://GitHub.com/vil/H4X-Tools/stargazers/)\n[![GitHub forks](https://badgen.net/github/forks/vil/H4X-Tools)](https://GitHub.com/vil/H4X-Tools/network/)\n\nA modular, terminal-based toolkit for OSINT, reconnaissance, and scraping - built in Python, runs on Linux and Windows.\n\nSubmit feature requests and bug reports in the [issues](https://github.com/vil/H4X-Tools/issues) tab.\n\n![](https://github.com/vil/H4X-Tools/blob/master/img/gui-v0.3.5.png)\n\n---\n\n## Tools\n\n| # | Tool | Description |\n|---|------|-------------|\n| 01 | **Ig Scrape** | Two-track Instagram OSINT scraper. **Guest mode** (no login) uses the `ensta` Guest API for public profile data and recent posts. **Authenticated mode** (Instagram `sessionid` cookie) uses [`toutatis`](https://github.com/megadose/toutatis) via Instagram's private mobile API for richer data - business flags, IGTV count, WhatsApp link status, and publicly listed contact details. Both tracks run Toutatis `advanced_lookup` to surface obfuscated email and phone from Instagram's account-recovery flow. Results can be exported to `scraped_data/` as **TXT**, **CSV**, or **JSON**. |\n| 02 | **Web Reconnaissance** | Multi-mode OSINT search powered by the `ddgs` library. Choose from 7 modes: **General** (free-form), **Person** (12 dorks), **Email** (8 dorks), **Domain** (12 recon dorks), **Username** (12 platform dorks), **Phone Number** (8 dorks), or **Custom Dork** (write your own template). Configurable result count, retry/back-off on rate limits. Results can be exported to `scraped_data/` as **TXT**, **CSV**, or **JSON**. |\n| 03 | **Phone Lookup** | Validates and analyses a phone number via the `phonenumbers` library (E.164/national/international formats, country, region, carrier, line type, time zones), then runs [`ignorant`](https://github.com/megadose/ignorant) to check social-media platform registrations. |\n| 04 | **IP Lookup** | Resolves a hostname or IP address and queries [ipinfo.io](https://ipinfo.io) for geolocation data - city, region, country, coordinates, ISP/organization, postal code, and timezone - with a direct OpenStreetMap link. |\n| 05 | **Username Search** | Asynchronously checks a username across hundreds of websites using a bundled site database. All matches (with direct profile URLs) are printed in real time. |\n| 06 | **Email Search** | Checks an email address against 100+ websites and services using [`holehe`](https://github.com/megadose/holehe) to identify where the address is registered. |\n| 07 | **Leak Search** | Multi-source breach and credential intelligence for an **email address**, **domain**, or **username**. Queries [Hudson Rock Cavalier](https://cavalier.hudsonrock.com) for stealer-log records (date of compromise, stealer family, infected machine details, masked credential samples, corporate/user service counts) and, for email targets, cross-references the [ProxyNova COMB](https://api.proxynova.com/comb) dataset (3.2B+ leaked credential lines) for a total hit count. Configurable inline entry limit; results can be exported to `scraped_data/` as **TXT**, **CSV**, or **JSON**. |\n| 08 | **Port Scanner** | Concurrently scans a user-defined TCP port range (1–N) on any IP or hostname using a 50-thread pool. Open ports are reported in real time. |\n| 09 | **WhoIs Lookup** | Performs a WHOIS query on a domain using the `whoisdomain` library and displays registrar, registration/expiry dates, name servers, status, and registrant details. |\n| 10 | **Fake Info Generator** | Generates a complete fake identity using [`Faker`](https://pypi.org/project/Faker/) - name, job, company, email, phone, address, credit card, IBAN, and location. |\n| 11 | **Web Scrape** | Asynchronously harvests all hyperlinks from a target URL. Optionally crawls every discovered page recursively. Results can be exported to `scraped_data/` as **TXT**, **CSV**, or **JSON**. |\n| 12 | **Wi-Fi Finder** | Scans for nearby Wi-Fi networks. Uses `netsh` on Windows and `nmcli` on Linux, reporting SSID, signal strength, and security type. The currently connected network is highlighted. |\n| 13 | **Wi-Fi Vault** | Dumps saved Wi-Fi passwords stored on the local machine - `netsh` on Windows, `nmcli` on Linux. |\n| 14 | **Dir Buster** | Asynchronously bruteforces directory and file paths on a target website using a built-in wordlist, printing every URL that returns HTTP 200. |\n| 15 | **Bluetooth Scanner** | Scans for nearby Bluetooth devices via `bluetoothctl` (Linux) and reports device names and MAC addresses. *(Windows support coming soon.)* |\n| 16 | **Local Users** | Enumerates all local user accounts on the system. On Linux: username, UID, GID, full name, home directory, shell, and group. On Windows: username, terminal, host, session start time, PID, SID, and domain. |\n\n---\n\n## Setup\n\n> [!IMPORTANT]\n> Requires [Python 3.10+](https://www.python.org/downloads/) and [Git](https://git-scm.com/downloads).\n> See the [wiki](https://github.com/vil/H4X-Tools/wiki) for a step-by-step guide.\n\n### Linux\n\n```sh\ngit clone https://github.com/vil/h4x-tools.git\ncd h4x-tools\nsh setup.sh\n```\n\n### Windows\n\n```bat\ngit clone https://github.com/vil/h4x-tools.git\ncd h4x-tools\nsetup.bat\n```\n\nThe setup scripts install all dependencies and optionally build a standalone executable via PyInstaller. You can also run the toolkit directly with:\n\n```sh\npython h4xtools.py\n```\n\nDependencies can be installed manually with:\n\n```sh\npip install -r requirements.txt\n```\n\n### Debug mode\n\nLaunch with the `--debug` flag to enable verbose output:\n\n```sh\npython h4xtools.py --debug\n```\n\n---\n\n## Contributing\n\nContributions are welcome! If you have Python knowledge and want to add a tool or improve an existing one:\n\n1. Fork the repository.\n2. Create a branch: `git checkout -b feature/my-tool`\n3. Write your code and tests.\n4. Open a pull request describing what you added or changed.\n\nPlease keep the style consistent with the existing utilities (use `helper/printer.py` for output, `@timer.timer` for the entry point, etc.).\n\n---\n\n## Security notice\n\nPre-compiled binaries are **not** provided. Downloading pre-built executables from untrusted sources is unsafe - always build from source yourself.\n\n---\n\n## License\n\n> This source code is licensed under the [GNU General Public License v3.0](https://www.gnu.org/licenses/gpl-3.0.txt).\n\n**This toolkit is intended for educational and authorised security research purposes only. Do not use it against systems or accounts you do not own or have explicit permission to test.**\n"
  },
  {
    "path": "h4xtools.py",
    "content": "#!/usr/bin/env python3\n\n\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport socket\nimport time\n\nfrom colorama import Fore, Style\n\nfrom helper import handles, printer\n\nVERSION = \"0.3.5\"\n\n\ndef _internet_check() -> None:\n    \"\"\"\n    Check if the user is connected to the internet by\n    creating a socket connection to a known host and port.\n    \"\"\"\n    try:\n        socket.setdefaulttimeout(3)\n        socket.create_connection((\"gnu.org\", 80))\n        printer.success(\"Internet Connection is Available..!\")\n    except socket.error as sock_error:\n        printer.warning(\n            \"Internet Connection is Unavailable or some other problem occurred..!\\n{}\".format(\n                sock_error\n            )\n        )\n\n\ndef _print_banner() -> None:\n    print(\n        Fore.LIGHTBLACK_EX\n        + f\"\"\"\n ▄ .▄▐▄• ▄ ▄▄▄▄▄            ▄▄▌  .▄▄ ·\n██▪▐█ █▌█▌▪•██  ▪     ▪     ██•  ▐█ ▀.\n██▀▐█ ·██·  ▐█.▪ ▄█▀▄  ▄█▀▄ ██▪  ▄▀▀▀█▄\n██▌▐▀▪▐█·█▌ ▐█▌·▐█▌.▐▌▐█▌.▐▌▐█▌▐▌▐█▄▪▐█\n▀▀▀ ·•▀▀ ▀▀ ▀▀▀  ▀█▄▀▪ ▀█▄▀▪.▀▀▀  ▀▀▀▀\n{Style.RESET_ALL}v{VERSION} / Vili (@vil)\n    \"\"\"\n    )\n\n\ndef _display_help() -> None:\n    print(Fore.LIGHTCYAN_EX)\n    print(\n        \"H4X-Tools v{} - A modular, terminal-based toolkit for OSINT, reconnaissance, and scraping - built in Python, runs on Linux and Windows.\".format(\n            VERSION\n        )\n    )\n    print(\"Repository link: https://github.com/vil/h4x-tools\")\n    print(\"\\nMade in Finland, with love.\\n\")\n\n    print(\"Available Tools:\")\n    print(\"------------------\")\n\n    # Use a loop to print the tools in a nice format\n    tools = {\n        \"Ig Scrape\": (\n            \"Two-track Instagram OSINT scraper. **Guest mode** (no login) uses the `ensta` Guest API for public profile data and recent posts. \"\n            \"**Authenticated mode** (Instagram `sessionid` cookie) uses [`toutatis`](https://github.com/megadose/toutatis) \"\n            \"via Instagram's private mobile API for richer data — business flags, IGTV count, WhatsApp link status, and publicly listed contact details. \"\n            \"Both tracks run Toutatis `advanced_lookup` to surface obfuscated email and phone from Instagram's account-recovery flow. \"\n            \"Results can be exported to `scraped_data/` as **TXT**, **CSV**, or **JSON**.\"\n        ),\n        \"Deep Web Search\": (\n            \"Multi-mode OSINT search powered by the ddgs library. Modes: General (free-form), \"\n            \"Person (12 dorks), Email (8 dorks), Domain (12 recon dorks), Username (12 platform \"\n            \"dorks), Phone Number (8 dorks), or Custom Dork (write your own template). \"\n            \"Results can be exported to `scraped_data/` as **TXT**, **CSV**, or **JSON**.\"\n        ),\n        \"Phone Lookup\": (\n            \"Validates and analyses a phone number — E.164/national/international formats, country, \"\n            \"region, carrier, line type, and time zones via the phonenumbers library. Then runs \"\n            \"ignorant to check social-media platform registrations.\"\n        ),\n        \"IP Lookup\": (\n            \"Resolves a hostname or IP and queries ipinfo.io for geolocation data — city, region, \"\n            \"country, coordinates, ISP/organization, postal code, and timezone — with a direct \"\n            \"OpenStreetMap link.\"\n        ),\n        \"Username Search\": (\n            \"Asynchronously checks a username across hundreds of websites using a bundled site \"\n            \"database. All matches with direct profile URLs are printed in real time.\"\n        ),\n        \"Email Search\": (\n            \"Checks an email address against 100+ websites and services using holehe to identify \"\n            \"where the address is registered. Credits: megadose/holehe.\"\n        ),\n        \"Leak Search\": (\n            \"Multi-source breach and credential intelligence for an **email address**, **domain**, or **username**. \"\n            \"Queries [Hudson Rock Cavalier](https://cavalier.hudsonrock.com) for stealer-log records \"\n            \"(date of compromise, stealer family, infected machine details, masked credential samples, corporate/user service counts) and, for email targets, \"\n            \"cross-references the [ProxyNova COMB](https://api.proxynova.com/comb) dataset (3.2B+ leaked credential lines) for a total hit count. \"\n            \"Configurable inline entry limit; results can be exported to `scraped_data/` as **TXT**, **CSV**, or **JSON**.\"\n        ),\n        \"Port Scanner\": (\n            \"Concurrently scans a user-defined TCP port range (1–N) on any IP or hostname using a \"\n            \"50-thread pool. Open ports are reported in real time.\"\n        ),\n        \"WhoIs Lookup\": (\n            \"Performs a WHOIS query on a domain and displays registrar, registration/expiry dates, \"\n            \"name servers, status flags, and registrant details.\"\n        ),\n        \"Fake Info Generator\": (\n            \"Generates a complete fake identity using Faker — name, job, company, email, phone, \"\n            \"address, credit card details (number, type, expiry, CVV), IBAN, and location.\"\n        ),\n        \"Web Scrape\": (\n            \"Asynchronously harvests all hyperlinks from a target URL. Optionally crawls every \"\n            \"discovered page recursively. Results can be exported to scraped_data/ as TXT, CSV, or JSON.\"\n        ),\n        \"Wi-Fi Finder\": (\n            \"Scans for nearby Wi-Fi networks using netsh (Windows) or nmcli (Linux). Reports SSID, \"\n            \"signal strength, and security type. The currently connected network is highlighted.\"\n        ),\n        \"Wi-Fi Vault\": (\n            \"Dumps saved Wi-Fi passwords stored on the local machine using netsh (Windows) or \"\n            \"nmcli (Linux).\"\n        ),\n        \"Dir Buster\": (\n            \"Asynchronously bruteforces directory and file paths on a target website using a \"\n            \"built-in wordlist, printing every URL that returns HTTP 200.\"\n        ),\n        \"Bluetooth Scanner\": (\n            \"Scans for nearby Bluetooth devices via bluetoothctl (Linux) and reports device names \"\n            \"and MAC addresses. Windows support is coming soon.\"\n        ),\n        \"Local Users\": (\n            \"Enumerates all local user accounts on the system. Linux: username, UID, GID, full \"\n            \"name, home directory, shell, and group. Windows: username, terminal, host, session \"\n            \"start time, PID, SID, and domain.\"\n        ),\n        \"Help\": \"Shows this help menu.\",\n    }\n\n    for tool, description in tools.items():\n        print(\"* {} - {}\".format(tool, description))\n\n    print(\"\\nClosing the Toolkit:\")\n    print(\"----------------------\")\n    print(\"You can close the toolkit using the following commands:\")\n    print(\"* quit\")\n    print(\"* q\")\n    print(\"* kill\")\n    print(\"* exit\")\n\n    print(\"\\nLicense and Credits:\")\n    print(\"---------------------\")\n    print(\n        \"H4X-Tools is under the GNU General Public License v3, made by Vili (@vil).\\n\"\n        \"This toolkit is for educational and authorised security research purposes only.\"\n    )\n\n\ndef _print_menu() -> None:\n    max_option_length = max(\n        len(value.__name__.replace(\"handle_\", \"\").replace(\"_\", \" \").title())\n        for value in MENU_OPTIONS.values()\n    )\n\n    for i, (key, value) in enumerate(MENU_OPTIONS.items(), start=1):\n        option_name = value.__name__.replace(\"handle_\", \"\").replace(\"_\", \" \").title()\n        print(\n            f\"{Fore.LIGHTGREEN_EX}[{key.zfill(2)}]{Style.RESET_ALL} {option_name.ljust(max_option_length)}\",\n            end=\"\",\n        )\n\n        if i % 2 == 0:\n            print()\n        else:\n            print(\" \" * 4, end=\"\")\n\n    print(\"\\n\")\n    print(f\"Type {Style.BRIGHT}?{Style.RESET_ALL} for help.\")\n    print(f\"Type {Style.BRIGHT}exit{Style.RESET_ALL} to close the toolkit...\")\n\n\nMENU_OPTIONS = {\n    \"1\": handles.handle_ig_scrape,\n    \"2\": handles.handle_web_reconnaissance,\n    \"3\": handles.handle_phone_lookup,\n    \"4\": handles.handle_ip_lookup,\n    \"5\": handles.handle_username_search,\n    \"6\": handles.handle_email_search,\n    \"7\": handles.handle_leak_search,\n    \"8\": handles.handle_port_scanner,\n    \"9\": handles.handle_whois_lookup,\n    \"10\": handles.handle_fake_info_generator,\n    \"11\": handles.handle_web_scrape,\n    \"12\": handles.handle_wifi_finder,\n    \"13\": handles.handle_wifi_vault,\n    \"14\": handles.handle_dir_buster,\n    \"15\": handles.handle_bluetooth_scanner,\n    \"16\": handles.handle_local_users,\n}\n\n\ndef main() -> None:\n    _internet_check()\n    time.sleep(0.5)\n\n    printer.debug(\"DEBUG IS ON.\")\n\n    while True:\n        _print_banner()\n        _print_menu()\n        user_input = printer.user_input(\"Tool to execute : \\t\")\n\n        if user_input.lower() in {\"quit\", \"exit\", \"q\", \"kill\"}:\n            # Kill the program.\n            printer.warning(\"Quitting... Goodbye!\")\n            print(Style.RESET_ALL)\n            time.sleep(0.5)\n            break\n\n        if user_input in MENU_OPTIONS:\n            try:\n                MENU_OPTIONS[\n                    user_input\n                ]()  # Call the corresponding function based on the selected option\n            except KeyboardInterrupt:\n                printer.warning(\"Cancelled..!\")\n        elif user_input.lower() == \"?\":\n            _display_help()\n            printer.user_input(\"Done reading? Press the Enter key.\")\n        else:\n            printer.error(\"Invalid option!\")\n            time.sleep(0.5)\n\n\nif __name__ == \"__main__\":\n    while True:\n        try:\n            main()\n            break\n        except ValueError:\n            printer.error(\"Invalid value inputted..!\")\n        except KeyboardInterrupt:\n            print(\"\\n\")\n            printer.warning(\"Quitting... Goodbye!\")\n            print(Style.RESET_ALL)\n            exit(0)\n"
  },
  {
    "path": "helper/handles.py",
    "content": "\"\"\"\nCopyright (c) 2023-2025. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nfrom helper import printer\nfrom utils import (\n    bluetooth_scanner,\n    dirbuster,\n    email_search,\n    fake_info_generator,\n    ig_scrape,\n    ip_lookup,\n    leak_search,\n    local_users,\n    phonenumber_lookup,\n    port_scanner,\n    search_username,\n    web_reconnaissance,\n    web_scrape,\n    whois_lookup,\n    wifi_finder,\n    wifi_vault,\n)\n\n\ndef handle_bluetooth_scanner() -> None:\n    \"\"\"Handles the Bluetooth Scanner util.\"\"\"\n    scan_duration = int(printer.user_input(\"Enter a scan duration (seconds) : \\t\"))\n    bluetooth_scanner.scan_nearby_bluetooth(duration=scan_duration)\n\n\ndef handle_ig_scrape() -> None:\n    \"\"\"Handles the IG Scrape util.\"\"\"\n    target = str(printer.user_input(\"Enter a target username : \\t\")).replace(\" \", \"_\")\n    ig_scrape.scrape(target=target)\n\n\ndef handle_web_reconnaissance() -> None:\n    \"\"\"Handles the Deep Web Search util.\"\"\"\n    web_reconnaissance.websearch()\n\n\ndef handle_phone_lookup() -> None:\n    \"\"\"Handles the Phone number Lookup util.\"\"\"\n    printer.info(\"Include the country code, e.g. +358501234567 or +12025550123\")\n    no = str(printer.user_input(\"Enter a phone-number with country code : \\t\"))\n    phonenumber_lookup.lookup(phone_number=no)\n\n\ndef handle_ip_lookup() -> None:\n    \"\"\"Handles the IP/Domain Lookup util.\"\"\"\n    ip = str(printer.user_input(\"Enter a IP address OR domain : \\t\"))\n    ip_lookup.lookup(ip_address=ip)\n\n\ndef handle_username_search() -> None:\n    \"\"\"Handles the Username Search util.\"\"\"\n    username = str(printer.user_input(\"Enter a target username : \\t\")).replace(\" \", \"_\")\n    search_username.search(username=username)\n\n\ndef handle_email_search() -> None:\n    \"\"\"Handles the Email Search util.\"\"\"\n    printer.info(\n        \"holehe will check the address against 100+ websites and show where it is registered.\"\n    )\n    email = str(printer.user_input(\"Enter an email address : \\t\"))\n    email_search.search(email=email)\n\n\ndef handle_port_scanner() -> None:\n    \"\"\"Handles the Port Scanner util.\"\"\"\n    ip = str(printer.user_input(\"Enter a IP address OR domain : \\t\"))\n    port_range = int(printer.user_input(\"Enter number of ports to scan : \\t\"))\n    port_scanner.scan(ip=ip, port_range=port_range)\n\n\ndef handle_whois_lookup() -> None:\n    \"\"\"Handles the WhoIs Lookup util.\"\"\"\n    domain = str(printer.user_input(\"Enter a domain : \\t\"))\n    whois_lookup.check_whois(domain=domain)\n\n\ndef handle_fake_info_generator() -> None:\n    \"\"\"Handles the Fake Info Generator util.\"\"\"\n    fake_info_generator.generate()\n\n\ndef handle_web_scrape() -> None:\n    \"\"\"Handles the Web Scrape util.\"\"\"\n    url = str(printer.user_input(\"Enter a URL : \\t\"))\n    web_scrape.scrape(url=url)\n\n\ndef handle_wifi_finder() -> None:\n    \"\"\"Handles the Wi-Fi Finder util.\"\"\"\n    printer.info(\"Scanning for nearby Wi-Fi networks...\")\n    wifi_finder.scan_nearby_wifi()\n\n\ndef handle_wifi_vault() -> None:\n    \"\"\"Handles the Wi-Fi Password Getter util.\"\"\"\n    printer.info(\"Scanning for locally saved Wi-Fi passwords...\")\n    wifi_vault.get_local_passwords()\n\n\ndef handle_dir_buster() -> None:\n    \"\"\"Handles the Dir Buster util.\"\"\"\n    domain = printer.user_input(\"Enter a domain : \\t\")\n    dirbuster.bust(domain=domain)\n\n\ndef handle_local_users() -> None:\n    \"\"\"Handles the Local User Enum.\"\"\"\n    printer.info(\"Scanning for local accounts...\")\n    local_users.scan_for_local_users()\n\n\ndef handle_leak_search() -> None:\n    \"\"\"Handles the Cybercrime Intelligence util.\"\"\"\n    target = printer.user_input(\"Enter a target (email/domain) : \\t\")\n    leak_search.lookup(target=target)\n"
  },
  {
    "path": "helper/printer.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport re\nimport sys\n\nfrom colorama import Fore, Style\n\nANSI_ESCAPE = re.compile(r\"\\x1B(?:[@-Z\\\\-_]|\\[[0-?]*[ -/]*[@-~])\")\n\n\ndef _print_colored(message: str, color: str, prefix: str, *args, **kwargs) -> None:\n    \"\"\"\n    Print colored message with specified color and prefix.\n\n    :param message: message to print\n    :param color: color code from colorama.Fore\n    :param prefix: prefix for the message\n    :param args: arguments if any\n    :param kwargs: keyword arguments if any\n    \"\"\"\n    print(f\"{color}{prefix}{Style.RESET_ALL} {message}\", *args, **kwargs)\n\n\ndef info(message, *args, **kwargs) -> None:\n    _print_colored(message, Fore.LIGHTBLUE_EX, \"[*]\", *args, **kwargs)\n\n\ndef success(message, *args, **kwargs) -> None:\n    _print_colored(message, Fore.LIGHTGREEN_EX, \"[+]\", *args, **kwargs)\n\n\ndef error(message, *args, **kwargs) -> None:\n    _print_colored(message, Fore.LIGHTRED_EX, \"[!]\", *args, **kwargs)\n\n\ndef warning(message, *args, **kwargs) -> None:\n    _print_colored(message, Fore.LIGHTYELLOW_EX, \"[-]\", *args, **kwargs)\n\n\ndef debug(message, *args, **kwargs) -> None:\n    if \"--debug\" in sys.argv:\n        _print_colored(message, Fore.LIGHTMAGENTA_EX, \"[>]\", *args, **kwargs)\n\n\ndef section(title: str) -> None:\n    \"\"\"\n    Print a consistently formatted section header.\n\n    Output looks like:\n        [*] ─── Title ──────────────────────────────────────\n\n    The total content width (after the ``[*] `` prefix) is always 50 characters,\n    regardless of the title length.\n\n    :param title: The section title to display.\n    \"\"\"\n    _TOTAL = 50\n    left = \"─── \"\n    fill = _TOTAL - len(left) - len(title) - 1\n    right = \" \" + \"─\" * max(fill, 3)\n    _print_colored(left + title + right, Fore.LIGHTBLUE_EX, \"[*]\")\n\n\ndef noprefix(message, *args, **kwargs) -> None:\n    print(message, *args, **kwargs)\n\n\ndef user_input(prompt, *args, **kwargs) -> str:\n    _print_colored(prompt, Fore.LIGHTBLUE_EX, \"[?]\", end=\"\", *args, **kwargs)\n    return input()\n\n\ndef ansi_escape(output: str) -> str:\n    \"\"\"\n    Strips ANSI escapes from output.\n\n    :retrun clean_output: ANSI escape stripped output.\n    \"\"\"\n    clean_output = ANSI_ESCAPE.sub(\"\", output)\n\n    return clean_output\n"
  },
  {
    "path": "helper/randomuser.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport random\n\n_USER_AGENTS = [\n    \"Mozilla/5.0 (Windows NT 6.1; rv:94.0) Gecko/20100101 Firefox/94.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64; rv:94.0) Gecko/20100101 Firefox/94.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:94.0) Gecko/20100101 Firefox/94.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; rv:94.0) Gecko/20100101 Firefox/94.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64; rv:94.0) Gecko/20100101 Firefox/94.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64; rv:94.0) Gecko/20100101 Firefox/94.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; rv:94.0) Gecko/20100101 Firefox/94.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64; rv:94.0) Gecko/20100101 Firefox/94.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:94.0) Gecko/20100101 Firefox/94.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; rv:93.0) Gecko/20100101 Firefox/93.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64; rv:93.0) Gecko/20100101 Firefox/93.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:93.0) Gecko/20100101 Firefox/93.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; rv:93.0) Gecko/20100101 Firefox/93.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64; rv:93.0) Gecko/20100101 Firefox/93.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64; rv:93.0) Gecko/20100101 Firefox/93.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; rv:93.0) Gecko/20100101 Firefox/93.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64; rv:93.0) Gecko/20100101 Firefox/93.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:93.0) Gecko/20100101 Firefox/93.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; rv:92.0) Gecko/20100101 Firefox/92.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64; rv:92.0) Gecko/20100101 Firefox/92.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:92.0) Gecko/20100101 Firefox/92.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; rv:92.0) Gecko/20100101 Firefox/92.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64; rv:92.0) Gecko/20100101 Firefox/92.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64; rv:92.0) Gecko/20100101 Firefox/92.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; rv:92.0) Gecko/20100101 Firefox/92.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64; rv:92.0) Gecko/20100101 Firefox/92.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:92.0) Gecko/20100101 Firefox/92.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; rv:91.0) Gecko/20100101 Firefox/91.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64; rv:91.0) Gecko/20100101 Firefox/91.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:91.0) Gecko/20100101 Firefox/91.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; rv:91.0) Gecko/20100101 Firefox/91.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64; rv:91.0) Gecko/20100101 Firefox/91.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64; rv:91.0) Gecko/20100101 Firefox/91.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; rv:91.0) Gecko/20100101 Firefox/91.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64; rv:91.0) Gecko/20100101 Firefox/91.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:91.0) Gecko/20100101 Firefox/91.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; rv:90.0) Gecko/20100101 Firefox/90.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64; rv:90.0) Gecko/20100101 Firefox/90.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:90.0) Gecko/20100101 Firefox/90.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; rv:90.0) Gecko/20100101 Firefox/90.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64; rv:90.0) Gecko/20100101 Firefox/90.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64; rv:90.0) Gecko/20100101 Firefox/90.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; rv:90.0) Gecko/20100101 Firefox/90.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64; rv:90.0) Gecko/20100101 Firefox/90.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:90.0) Gecko/20100101 Firefox/90.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; rv:89.0) Gecko/20100101 Firefox/89.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64; rv:89.0) Gecko/20100101 Firefox/89.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:89.0) Gecko/20100101 Firefox/89.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; rv:89.0) Gecko/20100101 Firefox/89.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64; rv:89.0) Gecko/20100101 Firefox/89.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64; rv:89.0) Gecko/20100101 Firefox/89.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; rv:89.0) Gecko/20100101 Firefox/89.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64; rv:89.0) Gecko/20100101 Firefox/89.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:89.0) Gecko/20100101 Firefox/89.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; rv:88.0) Gecko/20100101 Firefox/88.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64; rv:88.0) Gecko/20100101 Firefox/88.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:88.0) Gecko/20100101 Firefox/88.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; rv:88.0) Gecko/20100101 Firefox/88.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64; rv:88.0) Gecko/20100101 Firefox/88.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64; rv:88.0) Gecko/20100101 Firefox/88.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; rv:88.0) Gecko/20100101 Firefox/88.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64; rv:88.0) Gecko/20100101 Firefox/88.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:88.0) Gecko/20100101 Firefox/88.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; rv:87.0) Gecko/20100101 Firefox/87.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64; rv:87.0) Gecko/20100101 Firefox/87.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:87.0) Gecko/20100101 Firefox/87.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; rv:87.0) Gecko/20100101 Firefox/87.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64; rv:87.0) Gecko/20100101 Firefox/87.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64; rv:87.0) Gecko/20100101 Firefox/87.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; rv:87.0) Gecko/20100101 Firefox/87.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64; rv:87.0) Gecko/20100101 Firefox/87.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:87.0) Gecko/20100101 Firefox/87.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; rv:86.0) Gecko/20100101 Firefox/86.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64; rv:86.0) Gecko/20100101 Firefox/86.0\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64; rv:86.0) Gecko/20100101 Firefox/86.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; rv:86.0) Gecko/20100101 Firefox/86.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64; rv:86.0) Gecko/20100101 Firefox/86.0\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64; rv:86.0) Gecko/20100101 Firefox/86.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; rv:86.0) Gecko/20100101 Firefox/86.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64; rv:86.0) Gecko/20100101 Firefox/86.0\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:86.0) Gecko/20100101 Firefox/86.0\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/94.0.4606.54 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.159 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.131 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.131 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.131 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.131 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.131 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.131 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.131 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.131 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.131 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.107 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.107 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.107 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.107 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.107 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.107 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.107 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.107 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/92.0.4515.107 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.164 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.164 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.164 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.164 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.164 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.164 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.164 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.164 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.164 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.124 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.106 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.77 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.212 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.93 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.85 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.85 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.85 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.85 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.85 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.85 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.85 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.85 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.85 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.72 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.72 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.72 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.72 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.72 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.72 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.72 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.72 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/90.0.4430.72 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.128 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.128 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.128 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.128 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.128 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.128 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.128 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.128 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.128 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.114 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.114 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.114 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.114 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.114 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.114 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.114 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.114 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.114 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/89.0.4389.90 Safari/537.36\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36 OPR/81.0.4196.54\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36 OPR/81.0.4196.54\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36 OPR/81.0.4196.54\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36 OPR/81.0.4196.54\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36 OPR/81.0.4196.54\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36 OPR/81.0.4196.54\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36 OPR/81.0.4196.54\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36 OPR/81.0.4196.54\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.69 Safari/537.36 OPR/81.0.4196.54\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.37\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.37\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.37\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.37\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.37\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.37\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.37\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.37\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.37\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.31\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.31\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.31\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.31\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.31\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.31\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.31\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.31\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/95.0.4638.54 Safari/537.36 OPR/81.0.4196.31\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.72\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.72\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.72\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.72\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.72\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.72\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.72\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.72\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.72\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.63\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.63\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.63\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.63\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.63\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.63\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.63\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.63\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.81 Safari/537.36 OPR/80.0.4170.63\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36 OPR/80.0.4170.40\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36 OPR/80.0.4170.40\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36 OPR/80.0.4170.40\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36 OPR/80.0.4170.40\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36 OPR/80.0.4170.40\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36 OPR/80.0.4170.40\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36 OPR/80.0.4170.40\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36 OPR/80.0.4170.40\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.71 Safari/537.36 OPR/80.0.4170.40\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36 OPR/80.0.4170.16\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36 OPR/80.0.4170.16\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36 OPR/80.0.4170.16\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36 OPR/80.0.4170.16\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36 OPR/80.0.4170.16\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36 OPR/80.0.4170.16\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36 OPR/80.0.4170.16\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36 OPR/80.0.4170.16\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/94.0.4606.61 Safari/537.36 OPR/80.0.4170.16\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.72\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.72\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.72\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.72\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.72\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.72\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.72\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.72\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.72\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.66\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.66\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.66\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.66\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.66\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.66\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.66\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.66\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.66\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.56\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.56\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.56\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.56\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.56\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.56\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.56\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.56\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.56\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.50\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.50\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.50\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.50\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.50\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.50\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.50\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.50\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36 OPR/79.0.4143.50\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36 OPR/79.0.4143.22\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36 OPR/79.0.4143.22\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36 OPR/79.0.4143.22\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36 OPR/79.0.4143.22\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36 OPR/79.0.4143.22\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36 OPR/79.0.4143.22\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36 OPR/79.0.4143.22\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36 OPR/79.0.4143.22\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/93.0.4577.63 Safari/537.36 OPR/79.0.4143.22\",\n    \"Mozilla/5.0 (Windows NT 6.1) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36 OPR/77.0.4054.90\",\n    \"Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36 OPR/77.0.4054.90\",\n    \"Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36 OPR/77.0.4054.90\",\n    \"Mozilla/5.0 (Windows NT 6.3) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36 OPR/77.0.4054.90\",\n    \"Mozilla/5.0 (Windows NT 6.3; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36 OPR/77.0.4054.90\",\n    \"Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36 OPR/77.0.4054.90\",\n    \"Mozilla/5.0 (Windows NT 10.0) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36 OPR/77.0.4054.90\",\n    \"Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36 OPR/77.0.4054.90\",\n    \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.43 (KHTML, like Gecko) Chrome/91.0.4472.101 Safari/537.36 OPR/77.0.4054.90\",\n]\n\n\nclass GetUser:\n    \"\"\"\n    Return a random user agent from the list.\n    \"\"\"\n\n    def __init__(self) -> None:\n        self.user_agent = random.choice(_USER_AGENTS)\n\n    def __str__(self) -> str:\n        return self.user_agent\n"
  },
  {
    "path": "helper/timer.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport functools\nimport time\nfrom types import FunctionType\n\nfrom helper import printer\n\n\ndef timer(require_input: bool) -> FunctionType:\n    \"\"\"\n    A timer decorator to measure the execution time of a function and optionally\n    require user input after execution.\n\n    :param require_input: Boolean flag to determine if input is required after execution\n    \"\"\"\n\n    def decorator(func):\n        @functools.wraps(func)\n        def wrapper(*args, **kwargs) -> str:\n            start_time = time.time()  # Start timing\n            result = func(*args, **kwargs)  # Execute the wrapped function\n            end_time = time.time()  # End timing\n\n            elapsed_time = end_time - start_time  # Calculate elapsed time\n            printer.info(\n                f\"Completed in {elapsed_time:.4f} seconds.\"\n            )  # Print the elapsed time\n\n            # Prompt the user for input after execution\n            if require_input:\n                printer.user_input(\"Press Enter key to continue...\")  # Prompt for input\n\n            return result  # Return the result of the wrapped function\n\n        return wrapper\n\n    return decorator\n"
  },
  {
    "path": "helper/url_helper.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport json\nimport os\nimport sys\n\nfrom helper import printer\n\n\ndef read_local_content(path) -> str | dict | None:\n    \"\"\"\n    Reads file content from a local file.\n\n    :param path: path to the file\n    :return: Parsed dict for JSON files, raw string for text files,\n             or None if an error occurs.\n    \"\"\"\n    try:\n        with open(_resource_path(path), \"r\") as file:\n            if path.endswith(\".json\"):\n                content = json.load(file)\n            else:\n                content = file.read()\n        return content\n    except Exception as e:\n        printer.error(f\"An error occurred: {str(e)}\")\n        return None\n\n\n# I hate pyinstaller.\ndef _resource_path(relative_path) -> str:\n    if hasattr(sys, \"_MEIPASS\"):\n        return os.path.join(sys._MEIPASS, relative_path)\n    return os.path.join(os.path.abspath(\".\"), relative_path)\n"
  },
  {
    "path": "requirements.txt",
    "content": "colorama\nddgs\nphonenumbers\nrequests\nbeautifulsoup4\nurllib3\npyinstaller\nwhoisdomain\ngit+https://github.com/diezo/Ensta.git\nsnscrape\nfaker\nholehe\nignorant\ntoutatis\nhttpx\naiohttp\npsutil\n"
  },
  {
    "path": "resources/data.json",
    "content": "{\n  \"sites\": [\n    {\n      \"app\": \"Facebook\",\n      \"id\": 1,\n      \"method\": \"GET\",\n      \"url\": \"https://www.facebook.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"YouTube\",\n      \"id\": 2,\n      \"method\": \"GET\",\n      \"url\": \"https://www.youtube.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'name\\\" content=' in responseContent\"\n    },\n    {\n      \"app\": \"Twitter\",\n      \"id\": 3,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('a', class_='profile-card-fullname')['title']\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='profile-bio').string\"\n        },\n        {\n          \"key\": \"Site\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='profile-website').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Member since\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='profile-joindate').find('span')['title']\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https://nitter.net'+soup.find('a', class_='profile-card-avatar')['href']\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.select_one('.profile-location:nth-of-type(2)').text.strip('\\\\t\\\\r\\\\n')\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://nitter.net/{username}\",\n      \"valid\": \"response.status == 200 and ') | nitter</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Telegram\",\n      \"id\": 4,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span', dir='auto').string\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img', class_='tgme_page_photo_image')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://t.me/{username}\",\n      \"valid\": \"'You can contact' in soup.find('meta', property='og:description')['content'] and 'tgme_page_title' in responseContent\"\n    },\n    {\n      \"app\": \"TikTok\",\n      \"id\": 5,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h2', attrs={'data-e2e':'user-bio'}).text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.tiktok.com/@{username}\",\n      \"valid\": \"response.status == 200 and 'title=\\\"Following\\\"' in responseContent\"\n    },\n    {\n      \"app\": \"Tinder\",\n      \"id\": 6,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta', property='profile:first_name')['content']\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('meta', property='og:image')['content']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://tinder.com/@{username}\",\n      \"valid\": \"'@' in soup.find('meta', property='og:title')['content'] and ') | Tinder</title><meta' in responseContent\"\n    },\n    {\n      \"app\": \"Instagram\",\n      \"id\": 7,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h2', class_='profile-name-bottom').string\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div', class_='profile-description').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Followers\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span', class_='followed_by').string\"\n        },\n        {\n          \"key\": \"Following\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span', class_='follows').string\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('div', class_='profile-avatar').find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.picuki.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Instagram profile with posts and stories' in responseContent\"\n    },\n    {\n      \"app\": \"Pinterest\",\n      \"id\": 8,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').string \"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta', property='og:description')['content']\"\n        },\n        {\n          \"key\": \"Followers\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta', property='pinterestapp:followers')['content']\"\n        },\n        {\n          \"key\": \"Following\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta', property='pinterestapp:following')['content']\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://pinterest.com/{username}/\",\n      \"valid\": \"response.status == 200 and ' - Profile | Pinterest' in responseContent\"\n    },\n    {\n      \"app\": \"Snapchat\",\n      \"id\": 9,\n      \"metadata\": [\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('image')['xlink:href']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://feelinsonice.appspot.com/web/deeplink/snapcode?username={username}&size=400&type=SVG\",\n      \"valid\": \"soup.find('defs') != None and '</clipPath>' in responseContent\"\n    },\n    {\n      \"app\": \"Reddit\",\n      \"id\": 10,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"jsonData['data']['subreddit']['title']\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"jsonData['data']['subreddit']['public_description']\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"jsonData['data']['snoovatar_img']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.reddit.com/user/{username}/about.json\",\n      \"valid\": \"response.status == 200 and 'total_karma' in responseContent\"\n    },\n    {\n      \"app\": \"Soundcloud\",\n      \"id\": 11,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('header').find('p').string\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('meta',property='og:locality')['content']+', '+soup.find('meta',property='og:country-name')['content']\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('meta',property='twitter:image')['content']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://soundcloud.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Github\",\n      \"id\": 12,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span',class_='p-name').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Nickname\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span',class_='p-nickname').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Site\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('a',rel='nofollow me').text\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('span',class_='p-label').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('meta',property='og:image')['content']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://github.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Steam\",\n      \"id\": 13,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='header_real_name').find('bdi').text\"\n        },\n        {\n          \"key\": \"Nickname\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span',class_='actual_persona_name').string\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta',property='og:description')['content']\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('img',class_='profile_flag').next_sibling.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('link',rel='image_src')['href']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://steamcommunity.com/id/{username}/\",\n      \"valid\": \"'Error' not in soup.find('title').string and 'g_rgProfileData =' in responseContent\"\n    },\n    {\n      \"app\": \"Linktree\",\n      \"id\": 14,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').string\"\n        },\n        {\n          \"key\": \"Description\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta',attrs={'name':'description'})['content']\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('meta',property='og:image')['content']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://linktr.ee/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Xbox Gamertag\",\n      \"id\": 15,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').string\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https:' + soup.find('img',class_='rounded img-thumbnail')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.xboxgamertag.com/search/{username}\",\n      \"valid\": \"response.status == 200 and 'Games Played' in responseContent\"\n    },\n    {\n      \"app\": \"Twitter Archived\",\n      \"id\": 16,\n      \"method\": \"GET\",\n      \"url\": \"http://archive.org/wayback/available?url=https://twitter.com/{username}\",\n      \"valid\": \"'available' in responseContent\"\n    },\n    {\n      \"app\": \"Xvideos\",\n      \"id\": 17,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',id='profile-title').find('strong').text\"\n        },\n        {\n          \"key\": \"Gender\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('p',id='pinfo-sex').find('span').text\"\n        },\n        {\n          \"key\": \"Age\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('p',id='pinfo-age').find('span').text\"\n        },\n        {\n          \"key\": \"Member since\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('p',id='pinfo-signedup').find('span').text\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('p',id='pinfo-city').find('span').text+', '+soup.find('p',id='pinfo-country').find('span').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.xvideos.com/profiles/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"PornHub\",\n      \"id\": 18,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='profileUserName').find('a').string\"\n        },\n        {\n          \"key\": \"Gender\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('dl',class_='moreInformation').select_one('dd:nth-of-type(1)').string\"\n        },\n        {\n          \"key\": \"Last login\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('dl',class_='moreInformation').select_one('dd:nth-of-type(2)').string\"\n        },\n        {\n          \"key\": \"Relationship Status\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('dl',class_='moreInformation').select_one('dd:nth-of-type(3)').string\"\n        },\n        {\n          \"key\": \"Interested In\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('dl',class_='moreInformation').select_one('dd:nth-of-type(4)').string\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('dl',class_='moreInformation').select_one('dd:nth-of-type(5)').string+', '+soup.find('dl',class_='moreInformation').select_one('dd:nth-of-type(6)').string\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img',id='getAvatar')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.pornhub.com/users/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Xhamster\",\n      \"id\": 19,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='user-name').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Gender\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='i-am').find(class_='value').text\"\n        },\n        {\n          \"key\": \"Last login\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='offline').text\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('div',class_='from').find(class_='value').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img',class_='xh-avatar')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://xhamster.com/users/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Periscope\",\n      \"id\": 20,\n      \"method\": \"GET\",\n      \"url\": \"https://www.periscope.tv/{username}\",\n      \"valid\": \"response.status == 200 and '<label>Followers' in responseContent\"\n    },\n    {\n      \"app\": \"AskFM\",\n      \"id\": 21,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='icon-bio').text\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('div',class_='icon-location').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img',class_='userAvatar')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://ask.fm/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Vimeo\",\n      \"id\": 22,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta', property='og:title')['content']\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta', property='og:description')['content']\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('meta', property='og:image')['content']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://vimeo.com/{username}\",\n      \"valid\": \"response.status == 200 and soup.find('title').text != 'VimeOhOh' and 'is a member of Vimeo, the' in responseContent\"\n    },\n    {\n      \"app\": \"Pastebin\",\n      \"id\": 23,\n      \"metadata\": [\n        {\n          \"key\": \"Member since\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span',class_='date-text')['title']\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('span',class_='location').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https://pastebin.com'+soup.find('div',class_='user-icon').find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://pastebin.com/u/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"WordPress Profile\",\n      \"id\": 24,\n      \"method\": \"GET\",\n      \"url\": \"https://profiles.wordpress.org/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"WordPress Site\",\n      \"id\": 25,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.wordpress.com/\",\n      \"valid\": \"'Do you want to register' not in responseContent and 'cannot be registered' not in responseContent\"\n    },\n    {\n      \"app\": \"AllMyLinks\",\n      \"id\": 26,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span',class_='profile-username').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('p',class_='profile-bio').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('meta',property='og:image')['content']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://allmylinks.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Buzzfeed\",\n      \"id\": 27,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta', attrs={'name':'og:title'})['content']\"\n        },\n        {\n          \"key\": \"Member since\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('dd').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https://img.buzzfeed.com/buzzfeed-static'+soup.find('meta',attrs={'name':'og:image'})['content']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.buzzfeed.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"JsFiddle\",\n      \"id\": 28,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='profileDetails').find('a').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('div',class_='avatar').find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://jsfiddle.net/user/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Sourceforge\",\n      \"id\": 29,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Member since\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('dl',class_='personal-data').select_one('dd:nth-of-type(2)').string.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Gender\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('dl',class_='personal-data').select_one('dd:nth-of-type(4)').string.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('dl',class_='personal-data').select_one('dd:nth-of-type(3)').string.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img',class_='project_icon')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://sourceforge.net/u/{username}/profile\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Kickstarter\",\n      \"id\": 30,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h2').text.strip('\\\\n')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"html.unescape(soup.find('div',id='profile_avatar').find('img')['src'])\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.kickstarter.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'projects' in responseContent\"\n    },\n    {\n      \"app\": \"Smule\",\n      \"id\": 31,\n      \"method\": \"GET\",\n      \"url\": \"https://www.smule.com/{username}\",\n      \"valid\": \"'404' not in soup.find('title').string\"\n    },\n    {\n      \"app\": \"Blogspot\",\n      \"id\": 32,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.blogspot.com/\",\n      \"valid\": \"response.status == 200 and 'Blogger Template Style' in responseContent\"\n    },\n    {\n      \"app\": \"Tradingview\",\n      \"id\": 33,\n      \"method\": \"GET\",\n      \"url\": \"https://www.tradingview.com/u/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Internet Archive\",\n      \"id\": 34,\n      \"method\": \"GET\",\n      \"url\": \"https://archive.org/details/@{username}\",\n      \"valid\": \"'cannot find account' not in soup.find('title').string\"\n    },\n    {\n      \"app\": \"Alura\",\n      \"id\": 35,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h2').string\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('p',class_='profile-header-bio').string.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img',class_='profile-header-avatar')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://cursos.alura.com.br/user/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Behance\",\n      \"id\": 36,\n      \"method\": \"GET\",\n      \"url\": \"https://www.behance.net/{username}/\",\n      \"valid\": \"response.status == 200 and '<title>Behance</title>' in responseContent\"\n    },\n    {\n      \"app\": \"MySpace\",\n      \"id\": 37,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',id='aboutMe').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Site\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='website').string.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('div',class_='location').string.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('a',id='profileImage').find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://myspace.com/{username}\",\n      \"valid\": \"response.status == 200 and '<!-- Profile -->' in responseContent\"\n    },\n    {\n      \"app\": \"Disqus\",\n      \"id\": 38,\n      \"method\": \"GET\",\n      \"url\": \"https://disqus.com/by/{username}/\",\n      \"valid\": \"response.status == 200 and '<title>Disqus Profile' in responseContent\"\n    },\n    {\n      \"app\": \"Slideshare\",\n      \"id\": 39,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span',class_='description').text\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('span',class_='city').text+', '+soup.find('span',class_='country-name').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img',class_='user-photo')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.slideshare.net/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Rumble\",\n      \"id\": 40,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://rumble.com/user/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Ebay\",\n      \"id\": 41,\n      \"method\": \"GET\",\n      \"url\": \"https://www.ebay.com/usr/{username}\",\n      \"valid\": \"'error' not in soup.find('title').string and response.status == 200\"\n    },\n    {\n      \"app\": \"RedBubble\",\n      \"id\": 42,\n      \"method\": \"GET\",\n      \"url\": \"https://www.redbubble.com/people/{username}/shop?asc=u\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Kik\",\n      \"id\": 43,\n      \"method\": \"GET\",\n      \"url\": \"https://ws2.kik.com/user/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Roblox\",\n      \"id\": 44,\n      \"method\": \"GET\",\n      \"url\": \"https://auth.roblox.com/v1/usernames/validate?username={username}&birthday=2019-12-31T23:00:00.000Z\",\n      \"valid\": \"'Username is already in use' in jsonData['message']\"\n    },\n    {\n      \"app\": \"Armor Games\",\n      \"id\": 45,\n      \"method\": \"GET\",\n      \"url\": \"https://armorgames.com/user/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Fortnite Tracker\",\n      \"id\": 46,\n      \"method\": \"GET\",\n      \"url\": \"https://fortnitetracker.com/profile/all/{username}\",\n      \"valid\": \"response.status == 200 and 's Fortnite Stats - Fortnite Tracker' in responseContent\"\n    },\n    {\n      \"app\": \"Duolingo\",\n      \"id\": 47,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"json['users'][0]['name']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.duolingo.com/2017-06-30/users?username={username}&_=1628308619574\",\n      \"valid\": \"len(jsonData['users']) > 0\"\n    },\n    {\n      \"app\": \"Chess\",\n      \"id\": 48,\n      \"method\": \"GET\",\n      \"url\": \"https://www.chess.com/member/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Shopify\",\n      \"id\": 49,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.myshopify.com/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Untappd\",\n      \"id\": 50,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('p',class_='location').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('div',class='avatar-holder').find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://untappd.com/user/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Last FM\",\n      \"id\": 51,\n      \"method\": \"GET\",\n      \"url\": \"https://www.last.fm/user/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Cash APP\",\n      \"id\": 52,\n      \"method\": \"GET\",\n      \"url\": \"https://cash.app/${username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Imgur\",\n      \"id\": 53,\n      \"method\": \"GET\",\n      \"url\": \"https://api.imgur.com/account/v1/accounts/{username}?client_id=546c25a59c58ad7&include=trophies%2Cmedallions\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Trello\",\n      \"id\": 54,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"jsonData['fullName']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://trello.com/1/Members/{username}?fields=activityBlocked%2CavatarUrl%2Cbio%2CbioData%2Cconfirmed%2CfullName%2CidEnterprise%2CidMemberReferrer%2Cinitials%2CmemberType%2CnonPublic%2Cproducts%2Curl%2Cusername\",\n      \"valid\": \"response.status == 200 and 'avatarUrl' in responseContent\"\n    },\n    {\n      \"app\": \"Minecraft\",\n      \"id\": 55,\n      \"metadata\": [\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"jsonData['data']['player']['avatar']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://playerdb.co/api/player/minecraft/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Patreon\",\n      \"id\": 56,\n      \"method\": \"GET\",\n      \"url\": \"https://www.patreon.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Become a patron of' in responseContent\"\n    },\n    {\n      \"app\": \"DockerHub\",\n      \"id\": 57,\n      \"method\": \"GET\",\n      \"url\": \"https://hub.docker.com/v2/users/{username}/\",\n      \"valid\": \"response.status == 200 and 'username' in responseContent\"\n    },\n    {\n      \"app\": \"Kongregate\",\n      \"id\": 58,\n      \"method\": \"GET\",\n      \"url\": \"https://www.kongregate.com/accounts/{username}\",\n      \"valid\": \"response.status == 200 and 'Member Since' in responseContent\"\n    },\n    {\n      \"app\": \"Vine\",\n      \"id\": 59,\n      \"method\": \"GET\",\n      \"url\": \"https://vine.co/api/users/profiles/vanity/{username}\",\n      \"valid\": \"response.status == 200 and 'userId' in responseContent\"\n    },\n    {\n      \"app\": \"Gamespot\",\n      \"id\": 60,\n      \"method\": \"GET\",\n      \"url\": \"https://www.gamespot.com/profile/{username}/\",\n      \"valid\": \"response.status == 200 and 's Profile - GameSpot</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Shutterstock\",\n      \"id\": 61,\n      \"method\": \"GET\",\n      \"url\": \"https://www.shutterstock.com/pt/g/{username}/about\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Chaturbate\",\n      \"id\": 62,\n      \"method\": \"GET\",\n      \"url\": \"https://chaturbate.com/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"ProtonMail\",\n      \"headers\": \"{'X-Pm-Appversion': 'web-account@4.28.2','X-Pm-Appversion': 'web-account@4.28.2'}\",\n      \"id\": 63,\n      \"method\": \"GET\",\n      \"url\": \"https://account.protonmail.com/api/users/available?Name={username}\",\n      \"valid\": \"response.status == 409\"\n    },\n    {\n      \"app\": \"TripAdvisor\",\n      \"id\": 64,\n      \"method\": \"GET\",\n      \"url\": \"https://www.tripadvisor.com/Profile/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"RapidAPI\",\n      \"id\": 65,\n      \"method\": \"GET\",\n      \"url\": \"https://rapidapi.com/user/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"HackTheBox\",\n      \"id\": 66,\n      \"json\": \"{{\\\"type\\\": \\\"username\\\",\\\"input\\\": \\\"{username}\\\"}}\",\n      \"method\": \"POST\",\n      \"url\": \"https://www.hackthebox.com/api/v4/register/check\",\n      \"valid\": \"jsonData['message']['found'] == True\"\n    },\n    {\n      \"app\": \"Wikipedia\",\n      \"id\": 67,\n      \"method\": \"GET\",\n      \"url\": \"https://en.wikipedia.org/w/api.php?action=query&format=json&list=users&ususers={username}&usprop=cancreate&formatversion=2&errorformat=html&errorsuselocal=true&uselang=en\",\n      \"valid\": \"response.status == 200 and 'userid' in responseContent\"\n    },\n    {\n      \"app\": \"Buymeacoffe\",\n      \"id\": 68,\n      \"method\": \"GET\",\n      \"url\": \"https://www.buymeacoffee.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Arduino\",\n      \"id\": 69,\n      \"method\": \"GET\",\n      \"url\": \"https://create.arduino.cc/projecthub/{username}\",\n      \"valid\": \"response.status == 200 and '- Arduino Project Hub' in responseContent\"\n    },\n    {\n      \"app\": \"League of Legends Tracker\",\n      \"id\": 70,\n      \"method\": \"GET\",\n      \"url\": \"https://tracker.gg/lol/profile/riot/NA/{username}/overview\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Lego Ideas\",\n      \"id\": 71,\n      \"method\": \"GET\",\n      \"url\": \"https://ideas.lego.com/profile/{username}/entries?query=&sort=top\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Fiverr\",\n      \"id\": 72,\n      \"method\": \"GET\",\n      \"url\": \"https://www.fiverr.com/{username}\",\n      \"valid\": \"response.status == 200 and 'member-since' in responseContent\"\n    },\n    {\n      \"app\": \"Redtube\",\n      \"id\": 73,\n      \"method\": \"GET\",\n      \"url\": \"https://www.redtube.com.br/users/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Dribble\",\n      \"id\": 74,\n      \"method\": \"GET\",\n      \"url\": \"https://dribbble.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Packet Storm Security\",\n      \"id\": 75,\n      \"method\": \"GET\",\n      \"url\": \"https://packetstormsecurity.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 'User Profile for' in responseContent\"\n    },\n    {\n      \"app\": \"Ello\",\n      \"id\": 76,\n      \"method\": \"GET\",\n      \"url\": \"https://ello.co/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Medium\",\n      \"id\": 77,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.medium.com/about\",\n      \"valid\": \"response.status == 200 and 'Medium member since' in responseContent\"\n    },\n    {\n      \"app\": \"Hackaday\",\n      \"id\": 78,\n      \"method\": \"GET\",\n      \"url\": \"https://hackaday.io/{username}\",\n      \"valid\": \"response.status == 200 and 's Profile | Hackaday.io' in responseContent\"\n    },\n    {\n      \"app\": \"Keybase\",\n      \"id\": 79,\n      \"method\": \"GET\",\n      \"url\": \"https://keybase.io/{username}\",\n      \"valid\": \"response.status == 200 and 'username' in responseContent\"\n    },\n    {\n      \"app\": \"HackerOne\",\n      \"id\": 80,\n      \"method\": \"GET\",\n      \"url\": \"https://hackerone.com/{username}?type=user\",\n      \"valid\": \"response.status == 200 and 'profile that highlights' in responseContent\"\n    },\n    {\n      \"app\": \"BugCrowd\",\n      \"id\": 81,\n      \"method\": \"GET\",\n      \"url\": \"https://bugcrowd.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"OneCompiler\",\n      \"id\": 82,\n      \"method\": \"GET\",\n      \"url\": \"https://onecompiler.com/api/users/{username}\",\n      \"valid\": \"response.status == 200 and 'name' in responseContent\"\n    },\n    {\n      \"app\": \"TryHackMe\",\n      \"id\": 83,\n      \"method\": \"GET\",\n      \"url\": \"https://tryhackme.com/p/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Lyrics Training\",\n      \"id\": 84,\n      \"method\": \"GET\",\n      \"url\": \"https://lyricstraining.com/search?user={username}\",\n      \"valid\": \"response.status == 200 and 'Sorry, there are no results for your search.' not in responseContent\"\n    },\n    {\n      \"app\": \"Expo\",\n      \"id\": 85,\n      \"method\": \"GET\",\n      \"url\": \"https://forums.expo.dev/u/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"RAWG\",\n      \"id\": 86,\n      \"method\": \"GET\",\n      \"url\": \"https://rawg.io/@{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Coroflot\",\n      \"id\": 87,\n      \"method\": \"GET\",\n      \"url\": \"https://www.coroflot.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Cloudflare\",\n      \"id\": 88,\n      \"method\": \"GET\",\n      \"url\": \"https://community.cloudflare.com/u/{username}\",\n      \"valid\": \"response.status == 200 and '- Cloudflare Community' in responseContent\"\n    },\n    {\n      \"app\": \"Wattpad\",\n      \"id\": 89,\n      \"method\": \"GET\",\n      \"url\": \"https://www.wattpad.com/user/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Mixlr\",\n      \"id\": 90,\n      \"method\": \"GET\",\n      \"url\": \"http://api.mixlr.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 'username' in responseContent\"\n    },\n    {\n      \"app\": \"ImageShack\",\n      \"id\": 91,\n      \"method\": \"GET\",\n      \"url\": \"https://imageshack.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 's Images</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Freelancer\",\n      \"id\": 92,\n      \"method\": \"GET\",\n      \"url\": \"https://www.freelancer.com/u/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Dev To\",\n      \"id\": 93,\n      \"method\": \"GET\",\n      \"url\": \"https://dev.to/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"BitBucket\",\n      \"id\": 94,\n      \"method\": \"GET\",\n      \"url\": \"https://bitbucket.org/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Ko Fi\",\n      \"id\": 95,\n      \"method\": \"GET\",\n      \"url\": \"https://ko-fi.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Become a supporter of' in responseContent\"\n    },\n    {\n      \"app\": \"Flickr\",\n      \"id\": 96,\n      \"method\": \"GET\",\n      \"url\": \"https://www.flickr.com/photos/{username}/\",\n      \"valid\": \"response.status == 200 and '| Flickr' in responseContent\"\n    },\n    {\n      \"app\": \"HackerEarth\",\n      \"id\": 97,\n      \"method\": \"GET\",\n      \"url\": \"https://www.hackerearth.com/@{username}\",\n      \"valid\": \"response.status == 200 and '404 | HackerEarth' not in responseContent\"\n    },\n    {\n      \"app\": \"Spotify\",\n      \"id\": 98,\n      \"method\": \"GET\",\n      \"url\": \"https://open.spotify.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'on Spotify</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Snapchat Stories\",\n      \"id\": 99,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h4').string\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://story.snapchat.com/s/{username}\",\n      \"valid\": \"response.status == 200 and 'is on Snapchat!' in responseContent\"\n    },\n    {\n      \"app\": \"Audio Jungle\",\n      \"id\": 100,\n      \"method\": \"GET\",\n      \"url\": \"https://audiojungle.net/user/{username}\",\n      \"valid\": \"response.status == 200 and 's profile on AudioJungle' in responseContent\"\n    },\n    {\n      \"app\": \"Avid Community\",\n      \"id\": 101,\n      \"method\": \"GET\",\n      \"url\": \"https://community.avid.com/members/{username}/default.aspx\",\n      \"valid\": \"response.status == 200 and 'My Activity' in responseContent\"\n    },\n    {\n      \"app\": \"Bandlab\",\n      \"id\": 102,\n      \"method\": \"GET\",\n      \"url\": \"https://www.bandlab.com/api/v1.3/users/{username}\",\n      \"valid\": \"response.status == 200 and 'about' in responseContent\"\n    },\n    {\n      \"app\": \"Carrd\",\n      \"id\": 103,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.carrd.co/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"CastingCallClub\",\n      \"id\": 104,\n      \"method\": \"GET\",\n      \"url\": \"https://www.castingcall.club/{username}\",\n      \"valid\": \"response.status == 200 and '| Casting Call Club' in responseContent\"\n    },\n    {\n      \"app\": \"Coderwall\",\n      \"id\": 105,\n      \"method\": \"GET\",\n      \"url\": \"https://coderwall.com/{username}/\",\n      \"valid\": \"response.status == 200 and 's profile |' in responseContent\"\n    },\n    {\n      \"app\": \"Codewars\",\n      \"id\": 106,\n      \"method\": \"GET\",\n      \"url\": \"https://www.codewars.com/users/{username}\",\n      \"valid\": \"response.status == 200 and '| Codewars' in responseContent\"\n    },\n    {\n      \"app\": \"F3\",\n      \"id\": 107,\n      \"method\": \"GET\",\n      \"url\": \"https://f3.cool/{username}\",\n      \"valid\": \"response.status == 200 and '<title>@' in responseContent\"\n    },\n    {\n      \"app\": \"Gab\",\n      \"id\": 108,\n      \"method\": \"GET\",\n      \"url\": \"https://gab.com/api/v1/account_by_username/{username}\",\n      \"valid\": \"response.status == 200 and 'followers_count' in responseContent\"\n    },\n    {\n      \"app\": \"Issuu\",\n      \"id\": 109,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('h2').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https:'+soup.find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://issuu.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Steemit\",\n      \"id\": 110,\n      \"method\": \"GET\",\n      \"url\": \"https://steemit.com/@{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Venmo\",\n      \"id\": 111,\n      \"method\": \"GET\",\n      \"url\": \"https://account.venmo.com/u/{username}\",\n      \"valid\": \"response.status == 200 and '\\\"profileInfo_username__' in responseContent\"\n    },\n    {\n      \"app\": \"MODDB\",\n      \"id\": 112,\n      \"method\": \"GET\",\n      \"url\": \"https://www.moddb.com/members/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"COLOURlovers\",\n      \"id\": 113,\n      \"method\": \"GET\",\n      \"url\": \"https://www.colourlovers.com/lover/{username}\",\n      \"valid\": \"response.status == 200 and 'since' in responseContent\"\n    },\n    {\n      \"app\": \"Scheme Color\",\n      \"id\": 114,\n      \"method\": \"GET\",\n      \"url\": \"https://www.schemecolor.com/author/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Roblox Trade\",\n      \"id\": 115,\n      \"method\": \"GET\",\n      \"url\": \"https://rblx.trade/p/{username}\",\n      \"valid\": \"response.status == 200 and 'Currently Wearing' in responseContent\"\n    },\n    {\n      \"app\": \"Aetherhub\",\n      \"id\": 116,\n      \"metadata\": [\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img', class_='ae-profile-image-card')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://aetherhub.com/User/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"BugBounty\",\n      \"id\": 117,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img', class_='avatar user-319-avatar avatar-450 photo')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://bugbounty.gg/members/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Huntr\",\n      \"id\": 118,\n      \"method\": \"GET\",\n      \"url\": \"https://huntr.dev/users/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Universocraft\",\n      \"id\": 119,\n      \"method\": \"GET\",\n      \"url\": \"https://stats.universocraft.com/stats.php?player={username}\",\n      \"valid\": \"response.status == 200 and 'No se ha encontrad' not in responseContent\"\n    },\n    {\n      \"app\": \"Wireclub\",\n      \"id\": 120,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta', property='og:title')['content']\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('meta', property='og:image')['content']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.wireclub.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 'Chat With' in responseContent\"\n    },\n    {\n      \"app\": \"AminoApps\",\n      \"id\": 121,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span', class_='NvNickname').text\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div', class_='bio').string.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https:' + soup.find('img', alt='Avatar')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://aminoapps.com/u/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Trakt\",\n      \"id\": 122,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').find('a').text\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h4').text\"\n        },\n        {\n          \"key\": \"Age\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('span',title='Age').text\"\n        },\n        {\n          \"key\": \"Gender\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h3',class_='emojis-supported').contents[2]['title']\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('h3',class_='emojis-supported').contents[1]\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img', class_='avatar')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://trakt.tv/users/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Giphy\",\n      \"id\": 123,\n      \"method\": \"GET\",\n      \"url\": \"https://giphy.com/channel/{username}\",\n      \"valid\": \"response.status == 200 and 'Share on GIPHY</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Minecraft List\",\n      \"id\": 124,\n      \"metadata\": [\n        {\n          \"key\": \"Servers\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='flex items-center bg-white p-2 mb-3').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https://minecraftlist.com' + soup.find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://minecraftlist.com/players/{username}\",\n      \"valid\": \"response.status == 200 and '-->was seen on<!--' in responseContent\"\n    },\n    {\n      \"app\": \"SEOClerks\",\n      \"id\": 125,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='profile-link').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Member since\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='user-data').find('span').text\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('div',class_='flag-pill')['data-content']\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('div',class_='user-avatar').find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.seoclerks.com/user/{username}\",\n      \"valid\": \"response.status == 200 and '<div class=\\\"user-info container\\\">' in responseContent\"\n    },\n    {\n      \"app\": \"Mix\",\n      \"id\": 126,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='IntoUserProfileLayout__name').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('meta',property='og:image')['content']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://mix.com/{username}/\",\n      \"valid\": \"response.status == 200 and '<title>@' in responseContent\"\n    },\n    {\n      \"app\": \"Codecademy\",\n      \"id\": 127,\n      \"method\": \"GET\",\n      \"url\": \"https://discuss.codecademy.com/u/{username}/summary\",\n      \"valid\": \"response.status == 200 and '<title>  Profile - ' in responseContent\"\n    },\n    {\n      \"app\": \"Bandcamp\",\n      \"id\": 128,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find(class_='blurb-text').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Site\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find(class_='website').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('meta',property='og:image')['content']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://bandcamp.com/{username}\",\n      \"valid\": \"response.status == 200 and ' collection | Bandcamp</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Poshmark\",\n      \"id\": 129,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text\"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('div',class_='d--fl ai--c').find('span').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('div',class_='closet__header__info').find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://poshmark.com/closet/{username}\",\n      \"valid\": \"response.status == 200 and ' is using Poshmark to sell items from their closet.' in responseContent\"\n    },\n    {\n      \"app\": \"hackster\",\n      \"id\": 130,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find_all('img')[2]['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.hackster.io/{username}\",\n      \"valid\": \"response.status == 200 and '- Hackster.io' in responseContent\"\n    },\n    {\n      \"app\": \"BodyBuilding\",\n      \"id\": 131,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='infoSummary').find('ul').find_all('li')[0].text.replace('REAL NAME', '')\"\n        },\n        {\n          \"key\": \"Age\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='infoSummary').find('ul').find_all('li')[1].text.replace('AGE', '')\"\n        },\n        {\n          \"key\": \"Gender\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='infoSummary').find('ul').find_all('li')[2].text.replace('GENDER', '') \"\n        },\n        {\n          \"key\": \"location\",\n          \"type\": \"location\",\n          \"value\": \"soup.find('div',class_='infoSummary').find('ul').find_all('li')[3].text.replace('LOCATION', '')\"\n        },\n        {\n          \"key\": \"Member since\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('div',class_='infoSummary').find('ul').find_all('li')[4].text.replace('member since', '')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https:' + soup.find_all('img',class_='BodyBanner-profile-pic__img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://bodyspace.bodybuilding.com/{username}/\",\n      \"valid\": \"response.status == 200 and 'infoSummary' in responseContent\"\n    },\n    {\n      \"app\": \"Mastodon\",\n      \"id\": 132,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('strong',class_='display-name__html').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img',id='profile_page_avatar')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://mastodon.social/@{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"IFTTT\",\n      \"id\": 133,\n      \"method\": \"GET\",\n      \"url\": \"https://ifttt.com/p/{username}\",\n      \"valid\": \"response.status == 200 and 'Joined' in responseContent\"\n    },\n    {\n      \"app\": \"Anime Planet\",\n      \"id\": 134,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('section',class_='profBio userContent').find('p').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https://www.anime-planet.com' + soup.find('img',id='user-avatar')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.anime-planet.com/users/{username}\",\n      \"valid\": \"'Profile' in soup.find('title').text and response.status == 200\"\n    },\n    {\n      \"app\": \"Destructoid\",\n      \"id\": 135,\n      \"metadata\": [\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https://www.destructoid.com/'+soup.find('div',id='userinfo_photo').find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.destructoid.com/?name={username}\",\n      \"valid\": \"response.status == 200 and 'Follow' in responseContent\"\n    },\n    {\n      \"app\": \"Gitee\",\n      \"id\": 136,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h2').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('p',class_='bio').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img',class_='ui image avatar')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://gitee.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Teknik\",\n      \"id\": 137,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text.strip('\\\\t\\\\r\\\\n')\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://user.teknik.io/{username}\",\n      \"valid\": \"response.status == 200 and 'Public Key' in responseContent\"\n    },\n    {\n      \"app\": \"BitChute\",\n      \"id\": 138,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('p',class_='name').text.strip('\\\\t\\\\r\\\\n')\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('img',class_='image lazyloaded')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.bitchute.com/channel/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"The Tatto Forum\",\n      \"id\": 139,\n      \"method\": \"GET\",\n      \"url\": \"https://www.thetattooforum.com/members/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"NPM\",\n      \"id\": 140,\n      \"method\": \"GET\",\n      \"url\": \"https://www.npmjs.com/~{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"PyPI\",\n      \"id\": 141,\n      \"method\": \"GET\",\n      \"url\": \"https://pypi.org/user/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"HackenProof\",\n      \"id\": 142,\n      \"method\": \"GET\",\n      \"url\": \"https://hackenproof.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"VKontakte\",\n      \"id\": 143,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1',class_='page_name').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"'https://vk.com/' + soup.find('img',class_='page_avatar_img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://vk.com/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"About me\",\n      \"id\": 144,\n      \"method\": \"GET\",\n      \"url\": \"https://about.me/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Dissenter\",\n      \"id\": 145,\n      \"method\": \"GET\",\n      \"url\": \"https://dissenter.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Dissenter | The Comment Section of the Internet' in responseContent\"\n    },\n    {\n      \"app\": \"Designspiration\",\n      \"id\": 146,\n      \"method\": \"GET\",\n      \"url\": \"https://www.designspiration.com/{username}/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Fark\",\n      \"id\": 147,\n      \"method\": \"GET\",\n      \"url\": \"https://www.fark.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 'bio' in responseContent\"\n    },\n    {\n      \"app\": \"mmorpg\",\n      \"id\": 148,\n      \"method\": \"GET\",\n      \"url\": \"https://forums.mmorpg.com/profile/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Pikabu\",\n      \"id\": 149,\n      \"method\": \"GET\",\n      \"url\": \"https://pikabu.ru/@{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Playstation Network\",\n      \"id\": 150,\n      \"method\": \"GET\",\n      \"url\": \"https://psnprofiles.com/xhr/search/users?q={username}\",\n      \"valid\": \"response.status == 200 and '<div class=\\\"progress-bar small level\\\">' in responseContent\"\n    },\n    {\n      \"app\": \"Warrior Forum\",\n      \"id\": 151,\n      \"method\": \"GET\",\n      \"url\": \"https://www.warriorforum.com/members/{username}.html\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Pxilart\",\n      \"id\": 152,\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('h1').text\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('div', class_='user-profile-image mb-0 mb-md-0').find('a').find('img')['src']\"\n        }\n      ],\n      \"method\": \"GET\",\n      \"url\": \"https://www.pixilart.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"2Dimensions\",\n      \"id\": 153,\n      \"method\": \"GET\",\n      \"url\": \"https://2Dimensions.com/a/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"3dnews\",\n      \"id\": 154,\n      \"method\": \"GET\",\n      \"url\": \"http://forum.3dnews.ru/member.php?username={username}\",\n      \"valid\": \"response.status == 200 and '\\u041f\\u043e\\u043b\\u044c\\u0437\\u043e\\u0432\\u0430\\u0442\\u0435\\u043b\\u044c \\u043d\\u0435 \\u0437\\u0430\\u0440\\u0435\\u0433\\u0438\\u0441\\u0442\\u0440\\u0438\\u0440\\u043e\\u0432\\u0430\\u043d \\u0438 \\u043d\\u0435 \\u0438\\u043c\\u0435\\u0435\\u0442 \\u043f\\u0440\\u043e\\u0444\\u0438\\u043b\\u044f \\u0434\\u043b\\u044f \\u043f\\u0440\\u043e\\u0441\\u043c\\u043e\\u0442\\u0440\\u0430.' not in responseContent\"\n    },\n    {\n      \"app\": \"7Cups\",\n      \"id\": 155,\n      \"method\": \"GET\",\n      \"url\": \"https://www.7cups.com/@{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"9GAG\",\n      \"id\": 156,\n      \"method\": \"GET\",\n      \"url\": \"https://www.9gag.com/u/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Academia.edu\",\n      \"id\": 157,\n      \"method\": \"GET\",\n      \"url\": \"https://independent.academia.edu/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Airbit\",\n      \"id\": 158,\n      \"method\": \"GET\",\n      \"url\": \"https://airbit.com/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Airliners\",\n      \"id\": 159,\n      \"method\": \"GET\",\n      \"url\": \"https://www.airliners.net/user/{username}/profile/photos\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Alik.cz\",\n      \"id\": 160,\n      \"method\": \"GET\",\n      \"url\": \"https://www.alik.cz/u/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Apple Developer\",\n      \"id\": 161,\n      \"method\": \"GET\",\n      \"url\": \"https://developer.apple.com/forums/profile/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Apple Discussions\",\n      \"id\": 162,\n      \"method\": \"GET\",\n      \"url\": \"https://discussions.apple.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'The page you tried was not found. You may have used an outdated link or may have typed the address (URL) incorrectly.' not in responseContent\"\n    },\n    {\n      \"app\": \"Asciinema\",\n      \"id\": 163,\n      \"method\": \"GET\",\n      \"url\": \"https://asciinema.org/~{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Ask Fedora\",\n      \"id\": 164,\n      \"method\": \"GET\",\n      \"url\": \"https://ask.fedoraproject.org/u/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Audiojungle\",\n      \"id\": 165,\n      \"method\": \"GET\",\n      \"url\": \"https://audiojungle.net/user/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Autofrage\",\n      \"id\": 166,\n      \"method\": \"GET\",\n      \"url\": \"https://www.autofrage.net/nutzer/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"BLIP.fm\",\n      \"id\": 167,\n      \"method\": \"GET\",\n      \"url\": \"https://blip.fm/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Bazar.cz\",\n      \"id\": 168,\n      \"method\": \"GET\",\n      \"url\": \"https://www.bazar.cz/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Bezuzyteczna\",\n      \"id\": 169,\n      \"method\": \"GET\",\n      \"url\": \"https://bezuzyteczna.pl/uzytkownicy/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Bikemap\",\n      \"id\": 170,\n      \"method\": \"GET\",\n      \"url\": \"https://www.bikemap.net/en/u/{username}/routes/created/\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"BioHacking\",\n      \"id\": 171,\n      \"method\": \"GET\",\n      \"url\": \"https://forum.dangerousthings.com/u/{username}\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"Bitwarden Forum\",\n      \"id\": 172,\n      \"method\": \"GET\",\n      \"url\": \"https://community.bitwarden.com/u/{username}/summary\",\n      \"valid\": \"response.status == 200\"\n    },\n    {\n      \"app\": \"101010 pl\",\n      \"id\": 173,\n      \"method\": \"GET\",\n      \"url\": \"https://101010.pl/@{username}\",\n      \"valid\": \"response.status == 200 and '@101010.pl' in responseContent\"\n    },\n    {\n      \"app\": \"3DNews\",\n      \"id\": 174,\n      \"method\": \"GET\",\n      \"url\": \"http://forum.3dnews.ru/member.php?username={username}\",\n      \"valid\": \"response.status == 200 and '<title>\\u0424\\u043e\\u0440\\u0443\\u043c 3DNews - \\u041f\\u0440\\u043e\\u0441\\u043c\\u043e\\u0442\\u0440 \\u043f\\u0440\\u043e\\u0444\\u0438\\u043b\\u044f:' in responseContent\"\n    },\n    {\n      \"app\": \"7cup\",\n      \"id\": 175,\n      \"method\": \"GET\",\n      \"url\": \"https://www.7cups.com/@{username}\",\n      \"valid\": \"response.status == 200 and 'Profile - 7 Cups' in responseContent\"\n    },\n    {\n      \"app\": \"21buttons\",\n      \"id\": 176,\n      \"method\": \"GET\",\n      \"url\": \"https://www.21buttons.com/buttoner/{username}\",\n      \"valid\": \"response.status == 200 and 'profile_user_followers' in responseContent\"\n    },\n    {\n      \"app\": \"about me\",\n      \"id\": 177,\n      \"method\": \"GET\",\n      \"url\": \"https://about.me/{username}\",\n      \"valid\": \"response.status == 200 and ' | about.me' in responseContent\"\n    },\n    {\n      \"app\": \"Adult_Forum\",\n      \"id\": 178,\n      \"method\": \"GET\",\n      \"url\": \"https://adultforum.gr/{username}-glamour-escorts/\",\n      \"valid\": \"response.status == 200 and 'Glamour Escorts ' in responseContent\"\n    },\n    {\n      \"app\": \"ADVFN\",\n      \"id\": 179,\n      \"method\": \"GET\",\n      \"url\": \"https://uk.advfn.com/forum/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Profile | ADVFN' in responseContent\"\n    },\n    {\n      \"app\": \"aflam\",\n      \"id\": 180,\n      \"method\": \"GET\",\n      \"url\": \"https://www.aflam4you.net/profile.html?u={username}\",\n      \"valid\": \"response.status == 200 and ') on \\u0628\\u062b \\u062d\\u064a \\u0648 \\u0645\\u0628\\u0627\\u0634\\u0631' in responseContent\"\n    },\n    {\n      \"app\": \"akniga\",\n      \"id\": 181,\n      \"method\": \"GET\",\n      \"url\": \"https://akniga.org/profile/{username}\",\n      \"valid\": \"response.status == 200 and ' - \\u0410\\u0443\\u0434\\u0438\\u043e\\u043a\\u043d\\u0438\\u0433\\u0438 \\u041a\\u043b\\u0443\\u0431</title' in responseContent\"\n    },\n    {\n      \"app\": \"Albicla\",\n      \"id\": 182,\n      \"method\": \"GET\",\n      \"url\": \"https://albicla.com/{username}/post/1\",\n      \"valid\": \"response.status == 500 and '500 Post tymczasowo niedost\\u0119pny' in responseContent\"\n    },\n    {\n      \"app\": \"allesovercrypto\",\n      \"id\": 183,\n      \"method\": \"GET\",\n      \"url\": \"https://allesovercrypto.nl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Favoriete coins' in responseContent\"\n    },\n    {\n      \"app\": \"allmylinks\",\n      \"id\": 184,\n      \"method\": \"GET\",\n      \"url\": \"https://allmylinks.com/{username}\",\n      \"valid\": \"response.status == 200 and 'message' in responseContent\"\n    },\n    {\n      \"app\": \"Alloannonces\",\n      \"id\": 185,\n      \"method\": \"GET\",\n      \"url\": \"https://www.alloannonces.ma/{username}/\",\n      \"valid\": \"response.status == 200 and 'Vendeurs/Agents' in responseContent\"\n    },\n    {\n      \"app\": \"AllTrails\",\n      \"id\": 186,\n      \"method\": \"GET\",\n      \"url\": \"https://www.alltrails.com/members/{username}\",\n      \"valid\": \"response.status == 200 and 's Profile | ' in responseContent\"\n    },\n    {\n      \"app\": \"Ameblo\",\n      \"id\": 187,\n      \"method\": \"GET\",\n      \"url\": \"https://ameblo.jp/{username}\",\n      \"valid\": \"response.status == 200 and '\\u753b\\u50cf\\u4e00\\u89a7' in responseContent\"\n    },\n    {\n      \"app\": \"AmericanThinker\",\n      \"id\": 188,\n      \"method\": \"GET\",\n      \"url\": \"https://www.americanthinker.com/author/{username}/\",\n      \"valid\": \"response.status == 200 and 'Articles &amp;' in responseContent\"\n    },\n    {\n      \"app\": \"Aminoapps\",\n      \"id\": 189,\n      \"method\": \"GET\",\n      \"url\": \"https://aminoapps.com/u/{username}\",\n      \"valid\": \"response.status == 200 and 'Stories' in responseContent\"\n    },\n    {\n      \"app\": \"AnimePlanet\",\n      \"id\": 190,\n      \"method\": \"GET\",\n      \"url\": \"https://www.anime-planet.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 'Joined' in responseContent\"\n    },\n    {\n      \"app\": \"aNobii\",\n      \"id\": 191,\n      \"method\": \"GET\",\n      \"url\": \"https://api.anobii.com/users/eid/{username}\",\n      \"valid\": \"response.status == 200 and 'Anobian since' in responseContent\"\n    },\n    {\n      \"app\": \"anonup\",\n      \"id\": 192,\n      \"method\": \"GET\",\n      \"url\": \"https://anonup.com/@{username}\",\n      \"valid\": \"response.status == 200 and 'Following</span>' in responseContent\"\n    },\n    {\n      \"app\": \"Apex Legends\",\n      \"id\": 193,\n      \"method\": \"GET\",\n      \"url\": \"https://apex.tracker.gg/apex/profile/origin/{username}/overview\",\n      \"valid\": \"response.status == 200 and 'Overview' in responseContent\"\n    },\n    {\n      \"app\": \"Appian\",\n      \"id\": 194,\n      \"method\": \"GET\",\n      \"url\": \"https://community.appian.com/members/{username}\",\n      \"valid\": \"response.status == 200 and 'User Profile' in responseContent\"\n    },\n    {\n      \"app\": \"Archive Of Our Own Account\",\n      \"id\": 195,\n      \"method\": \"GET\",\n      \"url\": \"https://archiveofourown.org/users/{username}\",\n      \"valid\": \"response.status == 200 and '>Profile<' in responseContent\"\n    },\n    {\n      \"app\": \"ArmorGames\",\n      \"id\": 196,\n      \"method\": \"GET\",\n      \"url\": \"https://armorgames.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'about' in responseContent\"\n    },\n    {\n      \"app\": \"ArtBreeder\",\n      \"id\": 197,\n      \"method\": \"GET\",\n      \"url\": \"https://www.artbreeder.com/{username}\",\n      \"valid\": \"response.status == 200 and '<title>' in responseContent\"\n    },\n    {\n      \"app\": \"Artists & Clients\",\n      \"id\": 198,\n      \"method\": \"GET\",\n      \"url\": \"https://artistsnclients.com/people/{username}\",\n      \"valid\": \"response.status == 200 and 'Member Since' in responseContent\"\n    },\n    {\n      \"app\": \"asciinema\",\n      \"id\": 199,\n      \"method\": \"GET\",\n      \"url\": \"https://asciinema.org/~{username}\",\n      \"valid\": \"response.status == 200 and 's profile - asciinema' in responseContent\"\n    },\n    {\n      \"app\": \"ask fm\",\n      \"id\": 200,\n      \"method\": \"GET\",\n      \"url\": \"https://ask.fm/{username}\",\n      \"valid\": \"response.status == 200 and 'answers,' in responseContent\"\n    },\n    {\n      \"app\": \"au ru\",\n      \"id\": 201,\n      \"method\": \"GET\",\n      \"url\": \"https://au.ru/user/{username}/\",\n      \"valid\": \"response.status == 200 and '\\u041b\\u043e\\u0442\\u044b \\u043f\\u043e\\u043b\\u044c\\u0437\\u043e\\u0432\\u0430\\u0442\\u0435\\u043b\\u044f ' in responseContent\"\n    },\n    {\n      \"app\": \"authorSTREAM\",\n      \"id\": 202,\n      \"method\": \"GET\",\n      \"url\": \"http://www.authorstream.com/{username}/\",\n      \"valid\": \"response.status == 200 and 'Presentations on authorSTREAM' in responseContent\"\n    },\n    {\n      \"app\": \"Ayfal\",\n      \"id\": 203,\n      \"method\": \"GET\",\n      \"url\": \"https://afyal.com/users/{username}\",\n      \"valid\": \"response.status == 200 and '\\u0633\\u0648\\u0642 \\u0623\\u0641\\u064a\\u0627\\u0644 \\u0642\\u0637\\u0639 \\u063a\\u064a\\u0627\\u0631 \\u0627\\u0644\\u0633\\u064a\\u0627\\u0631\\u0627\\u062a \\u0648\\u0645\\u062a\\u0639\\u0644\\u0642\\u0627\\u062a\\u0647\\u0627 -' in responseContent\"\n    },\n    {\n      \"app\": \"bblog_ru\",\n      \"id\": 204,\n      \"method\": \"GET\",\n      \"url\": \"https://www.babyblog.ru/user/{username}\",\n      \"valid\": \"response.status == 200 and '@' in responseContent\"\n    },\n    {\n      \"app\": \"BDSMLR\",\n      \"id\": 205,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.bdsmlr.com\",\n      \"valid\": \"response.status == 200 and 'login' in responseContent\"\n    },\n    {\n      \"app\": \"bdsmsingles\",\n      \"id\": 206,\n      \"method\": \"GET\",\n      \"url\": \"https://www.bdsmsingles.com/members/{username}/\",\n      \"valid\": \"response.status == 200 and '<title>Profile' in responseContent\"\n    },\n    {\n      \"app\": \"Bentbox\",\n      \"id\": 207,\n      \"method\": \"GET\",\n      \"url\": \"https://bentbox.co/{username}\",\n      \"valid\": \"response.status == 200 and 'BentBox photos and videos' in responseContent and 'This user is currently not available' not in responseContent\"\n    },\n    {\n      \"app\": \"BiggerPockets\",\n      \"id\": 208,\n      \"method\": \"GET\",\n      \"url\": \"https://www.biggerpockets.com/users/{username}\",\n      \"valid\": \"response.status == 200 and '| BiggerPockets</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Bimpos\",\n      \"id\": 209,\n      \"method\": \"GET\",\n      \"url\": \"https://ask.bimpos.com/user/{username}\",\n      \"valid\": \"response.status == 200 and '<title>User' in responseContent\"\n    },\n    {\n      \"app\": \"Bitbucket\",\n      \"id\": 210,\n      \"method\": \"GET\",\n      \"url\": \"https://bitbucket.org/{username}/\",\n      \"valid\": \"response.status == 200 and 'Repositories' in responseContent\"\n    },\n    {\n      \"app\": \"Bitchute\",\n      \"id\": 211,\n      \"method\": \"GET\",\n      \"url\": \"https://www.bitchute.com/channel/{username}/\",\n      \"valid\": \"response.status == 200 and 'subscribers' in responseContent\"\n    },\n    {\n      \"app\": \"bitcoin forum\",\n      \"id\": 212,\n      \"method\": \"GET\",\n      \"url\": \"https://bitcoinforum.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Profile of' in responseContent\"\n    },\n    {\n      \"app\": \"BLIP fm\",\n      \"id\": 213,\n      \"method\": \"GET\",\n      \"url\": \"https://blip.fm/{username}\",\n      \"valid\": \"response.status == 200 and 'recommended' in responseContent\"\n    },\n    {\n      \"app\": \"Blogger\",\n      \"id\": 214,\n      \"method\": \"GET\",\n      \"url\": \"https://www.blogger.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and '>On Blogger since' in responseContent\"\n    },\n    {\n      \"app\": \"blogi pl\",\n      \"id\": 215,\n      \"method\": \"GET\",\n      \"url\": \"https://www.blogi.pl/osoba,{username}.html\",\n      \"valid\": \"response.status == 200 and 'Informacje og\\u00f3lne' in responseContent\"\n    },\n    {\n      \"app\": \"Blogmarks\",\n      \"id\": 216,\n      \"method\": \"GET\",\n      \"url\": \"http://blogmarks.net/user/{username}\",\n      \"valid\": \"response.status == 200 and 'class=\\\"mark\\\"' in responseContent\"\n    },\n    {\n      \"app\": \"BodyBuilding com\",\n      \"id\": 217,\n      \"method\": \"GET\",\n      \"url\": \"http://api.bodybuilding.com/api-proxy/bbc/get?slug={username}\",\n      \"valid\": \"response.status == 200 and 'username' in responseContent\"\n    },\n    {\n      \"app\": \"Bookcrossing\",\n      \"id\": 218,\n      \"method\": \"GET\",\n      \"url\": \"https://www.bookcrossing.com/mybookshelf/{username}\",\n      \"valid\": \"response.status == 200 and 'Recent Book Activity' in responseContent\"\n    },\n    {\n      \"app\": \"Booth\",\n      \"id\": 219,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.booth.pm/\",\n      \"valid\": \"response.status == 200 and '- BOOTH</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Brickset\",\n      \"id\": 220,\n      \"method\": \"GET\",\n      \"url\": \"https://forum.brickset.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Activity</h2>' in responseContent\"\n    },\n    {\n      \"app\": \"Bugcrowd\",\n      \"id\": 221,\n      \"method\": \"GET\",\n      \"url\": \"https://bugcrowd.com/{username}\",\n      \"valid\": \"response.status == 200 and 's researcher profile on Bugcrowd' in responseContent\"\n    },\n    {\n      \"app\": \"buymeacoffee\",\n      \"id\": 222,\n      \"method\": \"GET\",\n      \"url\": \"https://www.buymeacoffee.com/{username}\",\n      \"valid\": \"response.status == 200 and 'supporters' in responseContent\"\n    },\n    {\n      \"app\": \"BuzzFeed\",\n      \"id\": 223,\n      \"method\": \"GET\",\n      \"url\": \"https://www.buzzfeed.com/{username}\",\n      \"valid\": \"response.status == 200 and 'memberSince' in responseContent\"\n    },\n    {\n      \"app\": \"Buzznet\",\n      \"id\": 224,\n      \"method\": \"GET\",\n      \"url\": \"https://www.buzznet.com/author/{username}/\",\n      \"valid\": \"response.status == 200 and '<title>Author:' in responseContent\"\n    },\n    {\n      \"app\": \"Carbonmade\",\n      \"id\": 225,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.carbonmade.com/\",\n      \"valid\": \"response.status == 200 and 's online portfolio' in responseContent\"\n    },\n    {\n      \"app\": \"Career habr\",\n      \"id\": 226,\n      \"method\": \"GET\",\n      \"url\": \"https://career.habr.com/{username}\",\n      \"valid\": \"response.status == 200 and '\\u2014 \\u0425\\u0430\\u0431\\u0440 \\u041a\\u0430\\u0440\\u044c\\u0435\\u0440\\u0430</title>' in responseContent\"\n    },\n    {\n      \"app\": \"CaringBridge\",\n      \"id\": 227,\n      \"method\": \"GET\",\n      \"url\": \"https://www.caringbridge.org/visit/{username}\",\n      \"valid\": \"response.status == 200 and '| CaringBridge' in responseContent\"\n    },\n    {\n      \"app\": \"carrd co\",\n      \"id\": 228,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.carrd.co\",\n      \"valid\": \"response.status == 200 and '( Made with Carrd )' in responseContent\"\n    },\n    {\n      \"app\": \"cash app\",\n      \"id\": 229,\n      \"method\": \"GET\",\n      \"url\": \"https://cash.app/${username}\",\n      \"valid\": \"response.status == 200 and ' on Cash App</title>' in responseContent\"\n    },\n    {\n      \"app\": \"CD-Action\",\n      \"id\": 230,\n      \"method\": \"GET\",\n      \"url\": \"https://cdaction.pl/uzytkownicy/{username}\",\n      \"valid\": \"response.status == 200 and 'Lista gier:' in responseContent\"\n    },\n    {\n      \"app\": \"cda pl\",\n      \"id\": 231,\n      \"method\": \"GET\",\n      \"url\": \"https://www.cda.pl/{username}\",\n      \"valid\": \"response.status == 200 and 'Foldery' in responseContent\"\n    },\n    {\n      \"app\": \"championat\",\n      \"id\": 232,\n      \"method\": \"GET\",\n      \"url\": \"https://www.championat.com/user/{username}/\",\n      \"valid\": \"response.status == 200 and '\\u041b\\u0438\\u0447\\u043d\\u044b\\u0439 \\u043f\\u0440\\u043e\\u0444\\u0438\\u043b' in responseContent\"\n    },\n    {\n      \"app\": \"Chaos social\",\n      \"id\": 233,\n      \"method\": \"GET\",\n      \"url\": \"https://chaos.social/@{username}\",\n      \"valid\": \"response.status == 200 and '- chaos.social' in responseContent\"\n    },\n    {\n      \"app\": \"cHEEZburger\",\n      \"id\": 234,\n      \"method\": \"GET\",\n      \"url\": \"https://profile.cheezburger.com/{username}\",\n      \"valid\": \"response.status == 200 and 'profile-header' in responseContent\"\n    },\n    {\n      \"app\": \"Chamsko\",\n      \"id\": 235,\n      \"method\": \"GET\",\n      \"url\": \"https://www.chamsko.pl/profil/{username}\",\n      \"valid\": \"response.status == 200 and 'W serwisie od' in responseContent\"\n    },\n    {\n      \"app\": \"Chess com\",\n      \"id\": 236,\n      \"method\": \"GET\",\n      \"url\": \"https://www.chess.com/member/{username}\",\n      \"valid\": \"response.status == 200 and 'Last Online' in responseContent\"\n    },\n    {\n      \"app\": \"Chomikuj pl\",\n      \"id\": 237,\n      \"method\": \"GET\",\n      \"url\": \"https://chomikuj.pl/{username}/\",\n      \"valid\": \"response.status == 200 and 'Foldery' in responseContent\"\n    },\n    {\n      \"app\": \"Chyoa\",\n      \"id\": 238,\n      \"method\": \"GET\",\n      \"url\": \"https://chyoa.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'When I\\\\'m not reading erotica I like to read' in responseContent\"\n    },\n    {\n      \"app\": \"clusterdafrica\",\n      \"id\": 239,\n      \"method\": \"GET\",\n      \"url\": \"https://clusterdafrica.com/@{username}\",\n      \"valid\": \"response.status == 200 and 'Membre depuis -' in responseContent\"\n    },\n    {\n      \"app\": \"cnet\",\n      \"id\": 240,\n      \"method\": \"GET\",\n      \"url\": \"https://www.cnet.com/profiles/{username}/\",\n      \"valid\": \"response.status == 200 and 'Member Since:' in responseContent\"\n    },\n    {\n      \"app\": \"codeforces\",\n      \"id\": 241,\n      \"method\": \"GET\",\n      \"url\": \"https://codeforces.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and '- Codeforces' in soup.find('meta', property='og:title')['content']\"\n    },\n    {\n      \"app\": \"codementor\",\n      \"id\": 242,\n      \"method\": \"GET\",\n      \"url\": \"https://www.codementor.io/@{username}\",\n      \"valid\": \"response.status == 200 and 'ABOUT ME' in responseContent\"\n    },\n    {\n      \"app\": \"contactos sex\",\n      \"id\": 243,\n      \"method\": \"GET\",\n      \"url\": \"https://www.contactossex.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Informaci\\u00f3n Personal' in responseContent\"\n    },\n    {\n      \"app\": \"coroflot\",\n      \"id\": 244,\n      \"method\": \"GET\",\n      \"url\": \"https://www.coroflot.com/{username}\",\n      \"valid\": \"response.status == 200 and 'portfolio' in responseContent\"\n    },\n    {\n      \"app\": \"cracked_io\",\n      \"id\": 245,\n      \"method\": \"GET\",\n      \"url\": \"https://cracked.io/{username}\",\n      \"valid\": \"response.status == 200 and 'Cracked.io - Profile of' in responseContent\"\n    },\n    {\n      \"app\": \"Cracked\",\n      \"id\": 246,\n      \"method\": \"GET\",\n      \"url\": \"https://www.cracked.com/members/{username}\",\n      \"valid\": \"response.status == 200 and 'Member Since' in responseContent\"\n    },\n    {\n      \"app\": \"crevado\",\n      \"id\": 247,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.crevado.com/\",\n      \"valid\": \"response.status == 200 and 'Portfolio' in responseContent\"\n    },\n    {\n      \"app\": \"crowdin\",\n      \"id\": 248,\n      \"method\": \"GET\",\n      \"url\": \"https://crowdin.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and ') \\u2013 Crowdin' in responseContent\"\n    },\n    {\n      \"app\": \"Cults3D\",\n      \"id\": 249,\n      \"method\": \"GET\",\n      \"url\": \"https://cults3d.com/en/users/{username}/creations\",\n      \"valid\": \"response.status == 200 and 'All the 3D models of' in responseContent\"\n    },\n    {\n      \"app\": \"Cytoid\",\n      \"id\": 250,\n      \"method\": \"GET\",\n      \"url\": \"https://cytoid.io/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Joined' in responseContent\"\n    },\n    {\n      \"app\": \"Dailymotion\",\n      \"id\": 251,\n      \"method\": \"GET\",\n      \"url\": \"https://www.dailymotion.com/{username}\",\n      \"valid\": \"response.status == 200 and 'og:url' in responseContent\"\n    },\n    {\n      \"app\": \"darudar\",\n      \"id\": 252,\n      \"method\": \"GET\",\n      \"url\": \"https://darudar.org/users/{username}/\",\n      \"valid\": \"response.status == 200 and '. \\u0414\\u0430\\u0440\\u0443\\u0434\\u0430\\u0440' in responseContent\"\n    },\n    {\n      \"app\": \"dateinasia\",\n      \"id\": 253,\n      \"method\": \"GET\",\n      \"url\": \"https://www.dateinasia.com/{username}\",\n      \"valid\": \"response.status == 200 and 'About me' in responseContent\"\n    },\n    {\n      \"app\": \"datezone\",\n      \"id\": 254,\n      \"method\": \"GET\",\n      \"url\": \"https://www.datezone.com/users/{username}/\",\n      \"valid\": \"response.status == 200 and ' - Users - Datezone' in responseContent\"\n    },\n    {\n      \"app\": \"Dating ru\",\n      \"id\": 255,\n      \"method\": \"GET\",\n      \"url\": \"https://dating.ru/{username}/\",\n      \"valid\": \"response.status == 200 and '| dating.ru' in responseContent\"\n    },\n    {\n      \"app\": \"Demotywatory\",\n      \"id\": 256,\n      \"method\": \"GET\",\n      \"url\": \"https://demotywatory.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Z nami od:' in responseContent\"\n    },\n    {\n      \"app\": \"Designspriation\",\n      \"id\": 257,\n      \"method\": \"GET\",\n      \"url\": \"https://www.designspiration.com/{username}/\",\n      \"valid\": \"response.status == 200 and 'has discovered on Designspiration' in responseContent\"\n    },\n    {\n      \"app\": \"DeviantArt\",\n      \"id\": 258,\n      \"method\": \"GET\",\n      \"url\": \"https://www.deviantart.com/{username}\",\n      \"valid\": \"response.status == 200 and ' | DeviantArt</title>' in responseContent\"\n    },\n    {\n      \"app\": \"dfgames\",\n      \"id\": 259,\n      \"method\": \"GET\",\n      \"url\": \"https://www.dfgames.com.br/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Reputa' in responseContent\"\n    },\n    {\n      \"app\": \"dev to\",\n      \"id\": 260,\n      \"method\": \"GET\",\n      \"url\": \"https://dev.to/{username}\",\n      \"valid\": \"response.status == 200 and '- DEV' in responseContent\"\n    },\n    {\n      \"app\": \"devRant\",\n      \"id\": 261,\n      \"method\": \"GET\",\n      \"url\": \"https://devrant.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 'Joined devRant on' in responseContent\"\n    },\n    {\n      \"app\": \"Diablo\",\n      \"id\": 262,\n      \"method\": \"GET\",\n      \"url\": \"https://diablo2.io/member/{username}/\",\n      \"valid\": \"response.status == 200 and 'Viewing profile - ' in responseContent\"\n    },\n    {\n      \"app\": \"diigo\",\n      \"id\": 263,\n      \"method\": \"GET\",\n      \"url\": \"https://www.diigo.com/interact_api/load_profile_info?name={username}\",\n      \"valid\": \"response.status == 200 and 'regist_at' in responseContent\"\n    },\n    {\n      \"app\": \"Digitalspy\",\n      \"id\": 264,\n      \"method\": \"GET\",\n      \"url\": \"https://forums.digitalspy.com/profile/discussions/{username}\",\n      \"valid\": \"response.status == 200 and 'About' in responseContent\"\n    },\n    {\n      \"app\": \"Discogs\",\n      \"id\": 265,\n      \"method\": \"GET\",\n      \"url\": \"https://www.discogs.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Joined on' in responseContent\"\n    },\n    {\n      \"app\": \"Discourse\",\n      \"id\": 266,\n      \"method\": \"GET\",\n      \"url\": \"https://meta.discourse.org/u/{username}/summary.json\",\n      \"valid\": \"response.status == 200 and 'topics' in responseContent\"\n    },\n    {\n      \"app\": \"discuss elastic co\",\n      \"id\": 267,\n      \"method\": \"GET\",\n      \"url\": \"https://discuss.elastic.co/u/{username}\",\n      \"valid\": \"response.status == 200 and '<title>  Profile' in responseContent\"\n    },\n    {\n      \"app\": \"Dojoverse\",\n      \"id\": 268,\n      \"method\": \"GET\",\n      \"url\": \"https://dojoverse.com/members/{username}/\",\n      \"valid\": \"response.status == 200 and 'Joined' in responseContent\"\n    },\n    {\n      \"app\": \"Dribbble\",\n      \"id\": 269,\n      \"method\": \"GET\",\n      \"url\": \"https://dribbble.com/{username}\",\n      \"valid\": \"response.status == 200 and ' | Dribbble' in responseContent\"\n    },\n    {\n      \"app\": \"Droners\",\n      \"id\": 270,\n      \"method\": \"GET\",\n      \"url\": \"https://droners.io/accounts/{username}/\",\n      \"valid\": \"response.status == 200 and '- Professional Drone Pilot' in responseContent\"\n    },\n    {\n      \"app\": \"easyen\",\n      \"id\": 271,\n      \"method\": \"GET\",\n      \"url\": \"https://easyen.ru/index/8-0-{username}\",\n      \"valid\": \"response.status == 200 and '\\u0414\\u0435\\u043d\\u044c \\u0440\\u043e\\u0436\\u0434\\u0435\\u043d\\u0438\\u044f' in responseContent\"\n    },\n    {\n      \"app\": \"eBay\",\n      \"id\": 272,\n      \"method\": \"GET\",\n      \"url\": \"https://www.ebay.com/usr/{username}\",\n      \"valid\": \"response.status == 200 and 'on eBay</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Elftown\",\n      \"id\": 273,\n      \"method\": \"GET\",\n      \"url\": \"http://elftown.com/{username}\",\n      \"valid\": \"response.status == 303\"\n    },\n    {\n      \"app\": \"Ello co\",\n      \"id\": 274,\n      \"method\": \"GET\",\n      \"url\": \"https://ello.co/{username}\",\n      \"valid\": \"response.status == 200 and '| Ello' in responseContent\"\n    },\n    {\n      \"app\": \"Engadget\",\n      \"id\": 275,\n      \"method\": \"GET\",\n      \"url\": \"https://www.engadget.com/about/editors/{username}/\",\n      \"valid\": \"response.status == 200 and '- Engadget</title>' in responseContent\"\n    },\n    {\n      \"app\": \"EPORNER\",\n      \"id\": 276,\n      \"method\": \"GET\",\n      \"url\": \"https://www.eporner.com/profile/{username}/\",\n      \"valid\": \"response.status == 200 and 'Video/Pics views' in responseContent\"\n    },\n    {\n      \"app\": \"Etsy\",\n      \"id\": 277,\n      \"method\": \"GET\",\n      \"url\": \"https://www.etsy.com/people/{username}\",\n      \"valid\": \"response.status == 200 and ' on Etsy</title>' in responseContent\"\n    },\n    {\n      \"app\": \"EU_Voice\",\n      \"id\": 278,\n      \"method\": \"GET\",\n      \"url\": \"https://social.network.europa.eu/@{username}\",\n      \"valid\": \"response.status == 200 and 'social.network.europa.eu' in responseContent\"\n    },\n    {\n      \"app\": \"ExtraLunchMoney\",\n      \"id\": 279,\n      \"method\": \"GET\",\n      \"url\": \"https://extralunchmoney.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Public Profile Page' in responseContent\"\n    },\n    {\n      \"app\": \"Eyeem\",\n      \"id\": 280,\n      \"method\": \"GET\",\n      \"url\": \"https://www.eyeem.com/u/{username}\",\n      \"valid\": \"response.status == 200 and '| EyeEm Photographer' in responseContent\"\n    },\n    {\n      \"app\": \"Fabswingers\",\n      \"id\": 281,\n      \"method\": \"GET\",\n      \"url\": \"https://www.fabswingers.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'View Profile' in responseContent\"\n    },\n    {\n      \"app\": \"Facenama\",\n      \"id\": 282,\n      \"method\": \"GET\",\n      \"url\": \"https://facenama.com/{username}\",\n      \"valid\": \"response.status == 200 and '\\u0634\\u0628\\u06a9\\u0647 \\u0627\\u062c\\u062a\\u0645\\u0627\\u0639\\u06cc \\u0641\\u06cc\\u0633 \\u0646\\u0645\\u0627</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Faktopedia\",\n      \"id\": 283,\n      \"method\": \"GET\",\n      \"url\": \"https://faktopedia.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Zamieszcza fakty od:' in responseContent\"\n    },\n    {\n      \"app\": \"FanCentro\",\n      \"id\": 284,\n      \"method\": \"GET\",\n      \"url\": \"https://fancentro.com/api/profile.get?profileAlias={username}&limit=1\",\n      \"valid\": \"response.status == 200 and '\\\"status\\\":true' in responseContent\"\n    },\n    {\n      \"app\": \"Fandom\",\n      \"id\": 285,\n      \"method\": \"GET\",\n      \"url\": \"https://www.fandom.com/u/{username}\",\n      \"valid\": \"response.status == 200 and '| Profile | Fandom' in responseContent\"\n    },\n    {\n      \"app\": \"fanpop\",\n      \"id\": 286,\n      \"method\": \"GET\",\n      \"url\": \"https://www.fanpop.com/fans/{username}\",\n      \"valid\": \"response.status == 200 and 'Fanpopping since' in responseContent\"\n    },\n    {\n      \"app\": \"fansly\",\n      \"id\": 287,\n      \"method\": \"GET\",\n      \"url\": \"https://apiv2.fansly.com/api/v1/account?usernames={username}\",\n      \"valid\": \"response.status == 200 and 'username' in responseContent\"\n    },\n    {\n      \"app\": \"FatSecret\",\n      \"id\": 288,\n      \"method\": \"GET\",\n      \"url\": \"https://www.fatsecret.com/member/{username}\",\n      \"valid\": \"response.status == 200 and '- Member</title>' in responseContent\"\n    },\n    {\n      \"app\": \"fcv\",\n      \"id\": 289,\n      \"method\": \"GET\",\n      \"url\": \"https://fcv.if.ua/index.php/component/comprofiler/userprofile/{username}/\",\n      \"valid\": \"response.status == 200 and '\\u0441\\u0442\\u043e\\u0440\\u0456\\u043d\\u043a\\u0430 \\u043f\\u0440\\u043e\\u0444\\u0456\\u043b\\u044e' in responseContent\"\n    },\n    {\n      \"app\": \"fedi lewactwo pl\",\n      \"id\": 290,\n      \"method\": \"GET\",\n      \"url\": \"https://fedi.lewactwo.pl/@{username}\",\n      \"valid\": \"response.status == 200 and '@lewactwo.pl' in responseContent\"\n    },\n    {\n      \"app\": \"Filmweb\",\n      \"id\": 291,\n      \"method\": \"GET\",\n      \"url\": \"https://www.filmweb.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Na filmwebie od' in responseContent\"\n    },\n    {\n      \"app\": \"Flipboard\",\n      \"id\": 292,\n      \"method\": \"GET\",\n      \"url\": \"https://flipboard.com/@{username}\",\n      \"valid\": \"response.status == 200 and ') on Flipboard' in responseContent\"\n    },\n    {\n      \"app\": \"Fodors Forum\",\n      \"id\": 293,\n      \"method\": \"GET\",\n      \"url\": \"https://www.fodors.com/community/profile/{username}/forum-activity\",\n      \"valid\": \"response.status == 200 and 'Member since' in responseContent\"\n    },\n    {\n      \"app\": \"forumprawne org\",\n      \"id\": 294,\n      \"method\": \"GET\",\n      \"url\": \"https://forumprawne.org/members/{username}.html\",\n      \"valid\": \"response.status == 200 and 'Wiadomo\\u015b\\u0107' in responseContent\"\n    },\n    {\n      \"app\": \"fotka\",\n      \"id\": 295,\n      \"method\": \"GET\",\n      \"url\": \"https://api.fotka.com/v2/user/dataStatic?login={username}\",\n      \"valid\": \"response.status == 200 and 'profil' in responseContent\"\n    },\n    {\n      \"app\": \"Foursquare\",\n      \"id\": 296,\n      \"method\": \"GET\",\n      \"url\": \"https://foursquare.com/{username}\",\n      \"valid\": \"response.status == 200 and 'on Foursquare</title>' in responseContent\"\n    },\n    {\n      \"app\": \"freelancer\",\n      \"id\": 297,\n      \"method\": \"GET\",\n      \"url\": \"https://www.freelancer.com/u/{username}\",\n      \"valid\": \"response.status == 200 and 'qualifications' in responseContent\"\n    },\n    {\n      \"app\": \"freesound\",\n      \"id\": 298,\n      \"method\": \"GET\",\n      \"url\": \"https://freesound.org/people/{username}/\",\n      \"valid\": \"response.status == 200 and 'START of Content area' in responseContent\"\n    },\n    {\n      \"app\": \"FriendFinder\",\n      \"id\": 299,\n      \"method\": \"GET\",\n      \"url\": \"https://friendfinder.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Last Visit:' in responseContent\"\n    },\n    {\n      \"app\": \"FriendFinder-X\",\n      \"id\": 300,\n      \"method\": \"GET\",\n      \"url\": \"https://www.friendfinder-x.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and '\\\\'s Dating Profile on FriendFinder-x' in responseContent\"\n    },\n    {\n      \"app\": \"Friendweb\",\n      \"id\": 301,\n      \"method\": \"GET\",\n      \"url\": \"https://friendweb.nl/{username}\",\n      \"valid\": \"response.status == 200 and 'friendweb.nl' in responseContent\"\n    },\n    {\n      \"app\": \"FurAffinity\",\n      \"id\": 302,\n      \"method\": \"GET\",\n      \"url\": \"https://www.furaffinity.net/user/{username}\",\n      \"valid\": \"response.status == 200 and '<title>Userpage of' in responseContent\"\n    },\n    {\n      \"app\": \"Furiffic\",\n      \"id\": 303,\n      \"method\": \"GET\",\n      \"url\": \"https://www.furiffic.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Registered Since' in responseContent\"\n    },\n    {\n      \"app\": \"game_debate\",\n      \"id\": 304,\n      \"method\": \"GET\",\n      \"url\": \"https://www.game-debate.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and '| , , GB pc game performance' in responseContent\"\n    },\n    {\n      \"app\": \"Garmin connect\",\n      \"id\": 305,\n      \"method\": \"GET\",\n      \"url\": \"https://connect.garmin.com/modern/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'window.ERROR_VIEW = null' in responseContent\"\n    },\n    {\n      \"app\": \"Geocaching\",\n      \"id\": 306,\n      \"method\": \"GET\",\n      \"url\": \"https://www.geocaching.com/p/?u={username}\",\n      \"valid\": \"response.status == 200 and 'Groundspeak - User Profile' in responseContent\"\n    },\n    {\n      \"app\": \"getmonero\",\n      \"id\": 307,\n      \"method\": \"GET\",\n      \"url\": \"https://forum.getmonero.org/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Monero | User' in responseContent\"\n    },\n    {\n      \"app\": \"Gettr\",\n      \"id\": 308,\n      \"method\": \"GET\",\n      \"url\": \"https://api.gettr.com/s/user/{username}/exist\",\n      \"valid\": \"response.status == 200 and 'success\\\":true' in responseContent\"\n    },\n    {\n      \"app\": \"Gigapan\",\n      \"id\": 309,\n      \"method\": \"GET\",\n      \"url\": \"https://www.gigapan.com/profiles/{username}\",\n      \"valid\": \"response.status == 200 and 'width=\\\"100\\\"' in responseContent\"\n    },\n    {\n      \"app\": \"Girlfriendsmeet\",\n      \"id\": 310,\n      \"method\": \"GET\",\n      \"url\": \"http://www.girlfriendsmeet.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'online dating profile' in responseContent\"\n    },\n    {\n      \"app\": \"gitea\",\n      \"id\": 311,\n      \"method\": \"GET\",\n      \"url\": \"https://gitea.com/{username}\",\n      \"valid\": \"response.status == 200 and '(Git with a cup of tea)' in responseContent\"\n    },\n    {\n      \"app\": \"GitHub\",\n      \"id\": 312,\n      \"method\": \"GET\",\n      \"url\": \"https://github.com/{username}\",\n      \"valid\": \"response.status == 200 and 'p-nickname vcard-username d-block' in responseContent\"\n    },\n    {\n      \"app\": \"GitLab\",\n      \"id\": 313,\n      \"method\": \"GET\",\n      \"url\": \"https://gitlab.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Member since' in responseContent\"\n    },\n    {\n      \"app\": \"gitee\",\n      \"id\": 314,\n      \"method\": \"GET\",\n      \"url\": \"https://gitee.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Commits, issues, and pull requests will appear' in responseContent\"\n    },\n    {\n      \"app\": \"gloria tv\",\n      \"id\": 315,\n      \"method\": \"GET\",\n      \"url\": \"https://gloria.tv/{username}\",\n      \"valid\": \"response.status == 200 and 'Last online' in responseContent\"\n    },\n    {\n      \"app\": \"gnome_extensions\",\n      \"id\": 316,\n      \"method\": \"GET\",\n      \"url\": \"https://extensions.gnome.org/accounts/profile/{username}\",\n      \"valid\": \"response.status == 200 and 's Profile - GNOME Shell Extensions' in responseContent\"\n    },\n    {\n      \"app\": \"gpodder net\",\n      \"id\": 317,\n      \"method\": \"GET\",\n      \"url\": \"https://gpodder.net/user/{username}/\",\n      \"valid\": \"response.status == 200 and 'mdash; gpodder.net' in responseContent\"\n    },\n    {\n      \"app\": \"grandprof\",\n      \"id\": 318,\n      \"method\": \"GET\",\n      \"url\": \"https://grandprof.org/communaute/{username}\",\n      \"valid\": \"response.status == 200 and 's Profile' in responseContent\"\n    },\n    {\n      \"app\": \"Gravatar\",\n      \"id\": 319,\n      \"method\": \"GET\",\n      \"url\": \"http://en.gravatar.com/profiles/{username}.json\",\n      \"valid\": \"response.status == 200 and 'entry' in responseContent\"\n    },\n    {\n      \"app\": \"gumroad\",\n      \"id\": 320,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.gumroad.com/\",\n      \"valid\": \"response.status == 200 and 's profile picture' in responseContent\"\n    },\n    {\n      \"app\": \"Hacker News\",\n      \"id\": 321,\n      \"method\": \"GET\",\n      \"url\": \"https://news.ycombinator.com/user?id={username}\",\n      \"valid\": \"response.status == 200 and 'created:' in responseContent\"\n    },\n    {\n      \"app\": \"Hackernoon\",\n      \"id\": 322,\n      \"method\": \"GET\",\n      \"url\": \"https://hackernoon.com/_next/data/foL6JC7ro2FEEMD-gMKgQ/u/{username}.json\",\n      \"valid\": \"response.status == 200 and '\\\"profile\\\"' in responseContent\"\n    },\n    {\n      \"app\": \"hackerearth\",\n      \"id\": 323,\n      \"method\": \"GET\",\n      \"url\": \"https://www.hackerearth.com/@{username}\",\n      \"valid\": \"response.status == 200 and '| Developer Profile on HackerEarth' in responseContent\"\n    },\n    {\n      \"app\": \"hamaha\",\n      \"id\": 324,\n      \"method\": \"GET\",\n      \"url\": \"https://hamaha.net/{username}\",\n      \"valid\": \"response.status == 200 and '- \\u0442\\u0440\\u0435\\u0439\\u0434\\u0438\\u043d\\u0433 \\u0444\\u043e\\u0440\\u0435\\u043a\\u0441 \\u0444\\u044c\\u044e\\u0447\\u0435\\u0440\\u0441\\u044b \\u0430\\u043a\\u0446\\u0438\\u0438 \\u0444\\u043e\\u043d\\u0434\\u043e\\u0432\\u044b\\u0439 \\u0440\\u044b\\u043d\\u043e\\u043a ' in responseContent\"\n    },\n    {\n      \"app\": \"Heylink\",\n      \"id\": 325,\n      \"method\": \"GET\",\n      \"url\": \"https://heylink.me/{username}/\",\n      \"valid\": \"response.status == 200 and 'HeyLink.me |' in responseContent\"\n    },\n    {\n      \"app\": \"hiberworld\",\n      \"id\": 326,\n      \"method\": \"GET\",\n      \"url\": \"https://hiberworld.com/u/{username}\",\n      \"valid\": \"response.status == 200 and 'Creations by ' in responseContent\"\n    },\n    {\n      \"app\": \"HomeDesign3D\",\n      \"id\": 327,\n      \"method\": \"GET\",\n      \"url\": \"https://en.homedesign3d.net/user/{username}\",\n      \"valid\": \"response.status == 200 and 'userspace' in responseContent\"\n    },\n    {\n      \"app\": \"Houzz\",\n      \"id\": 328,\n      \"method\": \"GET\",\n      \"url\": \"https://www.houzz.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Followers' in responseContent\"\n    },\n    {\n      \"app\": \"HubPages\",\n      \"id\": 329,\n      \"method\": \"GET\",\n      \"url\": \"https://hubpages.com/@{username}\",\n      \"valid\": \"response.status == 200 and 'name\\\">Followers' in responseContent\"\n    },\n    {\n      \"app\": \"Hubski\",\n      \"id\": 330,\n      \"method\": \"GET\",\n      \"url\": \"https://hubski.com/user/{username}\",\n      \"valid\": \"response.status == 200 and '\\\\'s profile' in responseContent\"\n    },\n    {\n      \"app\": \"hugging_face\",\n      \"id\": 331,\n      \"method\": \"GET\",\n      \"url\": \"https://huggingface.co/{username}\",\n      \"valid\": \"response.status == 200 and 'thumbnails.huggingface.co/social-thumbnails/' in responseContent\"\n    },\n    {\n      \"app\": \"Iconfinder\",\n      \"id\": 332,\n      \"method\": \"GET\",\n      \"url\": \"https://www.iconfinder.com/{username}\",\n      \"valid\": \"response.status == 200 and 'iconsets' in responseContent\"\n    },\n    {\n      \"app\": \"icq-chat\",\n      \"id\": 333,\n      \"method\": \"GET\",\n      \"url\": \"https://icq.icqchat.co/members/{username}/\",\n      \"valid\": \"response.status == 200 and 'ICQ chat' in responseContent\"\n    },\n    {\n      \"app\": \"ifunny\",\n      \"id\": 334,\n      \"method\": \"GET\",\n      \"url\": \"https://ifunny.co/user/{username}\",\n      \"valid\": \"response.status == 200 and 'subscribers' in responseContent\"\n    },\n    {\n      \"app\": \"igromania\",\n      \"id\": 335,\n      \"method\": \"GET\",\n      \"url\": \"http://forum.igromania.ru/member.php?username={username}\",\n      \"valid\": \"response.status == 200 and '\\u0424\\u043e\\u0440\\u0443\\u043c \\u0418\\u0433\\u0440\\u043e\\u043c\\u0430\\u043d\\u0438\\u0438 - \\u041f\\u0440\\u043e\\u0441\\u043c\\u043e\\u0442\\u0440 \\u043f\\u0440\\u043e\\u0444\\u0438\\u043b\\u044f:' in responseContent\"\n    },\n    {\n      \"app\": \"ilovegrowingmarijuana\",\n      \"id\": 336,\n      \"method\": \"GET\",\n      \"url\": \"https://support.ilovegrowingmarijuana.com/u/{username}\",\n      \"valid\": \"response.status == 200 and '<title>  Profile - ' in responseContent\"\n    },\n    {\n      \"app\": \"imagefap\",\n      \"id\": 337,\n      \"method\": \"GET\",\n      \"url\": \"https://www.imagefap.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 's Profile' in responseContent\"\n    },\n    {\n      \"app\": \"iMGSRC RU\",\n      \"id\": 338,\n      \"method\": \"GET\",\n      \"url\": \"https://imgsrc.ru/main/user.php?lang=ru&user={username}\",\n      \"valid\": \"response.status == 200 and '\\u041f\\u0440\\u0438\\u0441\\u043e\\u0435\\u0434\\u0438\\u043d\\u0438\\u043b\\u0441\\u044f' in responseContent\"\n    },\n    {\n      \"app\": \"imgur\",\n      \"id\": 339,\n      \"method\": \"GET\",\n      \"url\": \"https://api.imgur.com/account/v1/accounts/{username}?client_id=546c25a59c58ad7&include=trophies%2Cmedallions\",\n      \"valid\": \"response.status == 200 and 'created_at' in responseContent\"\n    },\n    {\n      \"app\": \"Independent academia\",\n      \"id\": 340,\n      \"method\": \"GET\",\n      \"url\": \"https://independent.academia.edu/{username}\",\n      \"valid\": \"response.status == 200 and '- Academia.edu' in responseContent\"\n    },\n    {\n      \"app\": \"InkBunny\",\n      \"id\": 341,\n      \"method\": \"GET\",\n      \"url\": \"https://inkbunny.net/{username}\",\n      \"valid\": \"response.status == 200 and 'Profile | Inkbunny, the Furry Art Community</title>' in responseContent\"\n    },\n    {\n      \"app\": \"InsaneJournal\",\n      \"id\": 342,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.insanejournal.com/profile\",\n      \"valid\": \"response.status == 200 and 'User:' in responseContent\"\n    },\n    {\n      \"app\": \"instructables\",\n      \"id\": 343,\n      \"method\": \"GET\",\n      \"url\": \"https://www.instructables.com/member/{username}/\",\n      \"valid\": \"response.status == 200 and '>Joined' in responseContent\"\n    },\n    {\n      \"app\": \"Internet Archive Account\",\n      \"id\": 344,\n      \"method\": \"GET\",\n      \"url\": \"https://archive.org/details/@{username}\",\n      \"valid\": \"response.status == 200 and '<title>User Account' in responseContent\"\n    },\n    {\n      \"app\": \"Internet Archive User Search\",\n      \"id\": 345,\n      \"method\": \"GET\",\n      \"url\": \"https://archive.org/search.php?query={username}\",\n      \"valid\": \"response.status == 200 and '<!--/.item-ia-->' in responseContent\"\n    },\n    {\n      \"app\": \"interpals\",\n      \"id\": 346,\n      \"method\": \"GET\",\n      \"url\": \"https://www.interpals.net/{username}\",\n      \"valid\": \"response.status == 200 and 'Looking for' in responseContent\"\n    },\n    {\n      \"app\": \"ipolska pl\",\n      \"id\": 347,\n      \"method\": \"GET\",\n      \"url\": \"https://ipolska.pl/@{username}\",\n      \"valid\": \"response.status == 200 and '@ipolska.pl' in responseContent\"\n    },\n    {\n      \"app\": \"issuu\",\n      \"id\": 348,\n      \"method\": \"GET\",\n      \"url\": \"https://issuu.com/{username}\",\n      \"valid\": \"response.status == 200 and '- Issuu' in responseContent\"\n    },\n    {\n      \"app\": \"JBZD\",\n      \"id\": 349,\n      \"method\": \"GET\",\n      \"url\": \"https://jbzd.com.pl/uzytkownik/{username}\",\n      \"valid\": \"response.status == 200 and 'Dzidy u\\u017cytkownika' in responseContent\"\n    },\n    {\n      \"app\": \"jeja pl\",\n      \"id\": 350,\n      \"method\": \"GET\",\n      \"url\": \"https://www.jeja.pl/user,{username}\",\n      \"valid\": \"response.status == 200 and 'Profil u\\u017cytkownika' in responseContent\"\n    },\n    {\n      \"app\": \"Jeuxvideo\",\n      \"id\": 351,\n      \"method\": \"GET\",\n      \"url\": \"https://www.jeuxvideo.com/profil/{username}?mode=infos\",\n      \"valid\": \"response.status == 200 and '- jeuxvideo.com' in responseContent\"\n    },\n    {\n      \"app\": \"Joe Monster\",\n      \"id\": 352,\n      \"method\": \"GET\",\n      \"url\": \"https://joemonster.org/bojownik/{username}\",\n      \"valid\": \"response.status == 200 and 'jest prywatny' in responseContent\"\n    },\n    {\n      \"app\": \"JSFiddle\",\n      \"id\": 353,\n      \"method\": \"GET\",\n      \"url\": \"https://jsfiddle.net/user/{username}/\",\n      \"valid\": \"response.status == 200 and 'Settings - JSFiddle - Code Playground' in responseContent\"\n    },\n    {\n      \"app\": \"Justforfans\",\n      \"id\": 354,\n      \"method\": \"GET\",\n      \"url\": \"https://justfor.fans/{username}\",\n      \"valid\": \"response.status == 200 and ' @ JustFor.Fans' in responseContent\"\n    },\n    {\n      \"app\": \"kaggle\",\n      \"id\": 355,\n      \"method\": \"GET\",\n      \"url\": \"https://www.kaggle.com/{username}\",\n      \"valid\": \"response.status == 200 and '| Kaggle' in responseContent\"\n    },\n    {\n      \"app\": \"karab in\",\n      \"id\": 356,\n      \"method\": \"GET\",\n      \"url\": \"https://karab.in/u/{username}\",\n      \"valid\": \"response.status == 200 and 'Do\\u0142\\u0105czy\\u0142:' in responseContent\"\n    },\n    {\n      \"app\": \"kik\",\n      \"id\": 357,\n      \"method\": \"GET\",\n      \"url\": \"https://ws2.kik.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'firstName' in responseContent\"\n    },\n    {\n      \"app\": \"Ko-Fi\",\n      \"id\": 358,\n      \"method\": \"GET\",\n      \"url\": \"https://ko-fi.com/{username}\",\n      \"valid\": \"response.status == 200 and '>  Buy a Coffee for' in responseContent\"\n    },\n    {\n      \"app\": \"Kotburger\",\n      \"id\": 359,\n      \"method\": \"GET\",\n      \"url\": \"https://kotburger.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Zamieszcza kotburgery od:' in responseContent\"\n    },\n    {\n      \"app\": \"kwejk pl\",\n      \"id\": 360,\n      \"method\": \"GET\",\n      \"url\": \"https://kwejk.pl/uzytkownik/{username}#/tablica/\",\n      \"valid\": \"response.status == 200 and 'Kwejki u\\u017cytkownika' in responseContent\"\n    },\n    {\n      \"app\": \"LibraryThing\",\n      \"id\": 361,\n      \"method\": \"GET\",\n      \"url\": \"https://www.librarything.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Collections' in responseContent\"\n    },\n    {\n      \"app\": \"lichess\",\n      \"id\": 362,\n      \"method\": \"GET\",\n      \"url\": \"https://lichess.org/@/{username}\",\n      \"valid\": \"response.status == 200 and 'Activity' in responseContent\"\n    },\n    {\n      \"app\": \"LINE\",\n      \"id\": 363,\n      \"method\": \"GET\",\n      \"url\": \"https://line.me/R/ti/p/@{username}?from=page\",\n      \"valid\": \"response.status == 200 and 'Add LINE Friends via QR Code' in responseContent\"\n    },\n    {\n      \"app\": \"linux org ru\",\n      \"id\": 364,\n      \"method\": \"GET\",\n      \"url\": \"https://www.linux.org.ru/people/{username}/profile\",\n      \"valid\": \"response.status == 200 and '\\u0414\\u0430\\u0442\\u0430 \\u0440\\u0435\\u0433\\u0438\\u0441\\u0442\\u0440\\u0430\\u0446\\u0438\\u0438' in responseContent\"\n    },\n    {\n      \"app\": \"Livejournal\",\n      \"id\": 365,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.livejournal.com\",\n      \"valid\": \"response.status == 200 and '<link rel=\\\"canonical\\\" href=\\\"' in responseContent\"\n    },\n    {\n      \"app\": \"lobste rs\",\n      \"id\": 366,\n      \"method\": \"GET\",\n      \"url\": \"https://lobste.rs/u/{username}\",\n      \"valid\": \"response.status == 200 and 'Joined' in responseContent\"\n    },\n    {\n      \"app\": \"lowcygier pl\",\n      \"id\": 367,\n      \"method\": \"GET\",\n      \"url\": \"https://bazar.lowcygier.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Zarejestrowany' in responseContent\"\n    },\n    {\n      \"app\": \"MAGABOOK\",\n      \"id\": 368,\n      \"method\": \"GET\",\n      \"url\": \"https://magabook.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Recent Updates' in responseContent\"\n    },\n    {\n      \"app\": \"MAGA-CHAT\",\n      \"id\": 369,\n      \"method\": \"GET\",\n      \"url\": \"https://maga-chat.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Recent Updates' in responseContent\"\n    },\n    {\n      \"app\": \"Magix\",\n      \"id\": 370,\n      \"method\": \"GET\",\n      \"url\": \"https://www.magix.info/us/users/profile/{username}/\",\n      \"valid\": \"response.status == 200 and 'About me' in responseContent\"\n    },\n    {\n      \"app\": \"MapMyTracks\",\n      \"id\": 371,\n      \"method\": \"GET\",\n      \"url\": \"https://www.mapmytracks.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Daily distance this week' in responseContent\"\n    },\n    {\n      \"app\": \"Maroc_nl\",\n      \"id\": 372,\n      \"method\": \"GET\",\n      \"url\": \"https://www.maroc.nl/forums/members/{username}.html\",\n      \"valid\": \"response.status == 200 and 'Bekijk Profiel:' in responseContent\"\n    },\n    {\n      \"app\": \"Marshmallow\",\n      \"id\": 373,\n      \"method\": \"GET\",\n      \"url\": \"https://marshmallow-qa.com/{username}\",\n      \"valid\": \"response.status == 200 and '\\u3055\\u3093\\u306b\\u30e1\\u30c3\\u30bb\\u30fc\\u30b8\\u3092\\u304a\\u304f\\u308b' in responseContent\"\n    },\n    {\n      \"app\": \"Martech\",\n      \"id\": 374,\n      \"method\": \"GET\",\n      \"url\": \"https://martech.org/author/{username}/\",\n      \"valid\": \"response.status == 200 and 'twitter:site' in responseContent\"\n    },\n    {\n      \"app\": \"Massage Anywhere\",\n      \"id\": 375,\n      \"method\": \"GET\",\n      \"url\": \"https://www.massageanywhere.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and '<title>MassageAnywhere.com Profile for ' in responseContent\"\n    },\n    {\n      \"app\": \"mastodon\",\n      \"id\": 376,\n      \"method\": \"GET\",\n      \"url\": \"https://mastodon.social/@{username}\",\n      \"valid\": \"response.status == 200 and 'profile:username' in responseContent\"\n    },\n    {\n      \"app\": \"MCUUID (Minecraft)\",\n      \"id\": 377,\n      \"method\": \"GET\",\n      \"url\": \"https://playerdb.co/api/player/minecraft/{username}\",\n      \"valid\": \"response.status == 200 and 'Successfully found player by given ID.' in responseContent\"\n    },\n    {\n      \"app\": \"medyczka pl\",\n      \"id\": 378,\n      \"method\": \"GET\",\n      \"url\": \"http://medyczka.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Lista uzytkownikow' in responseContent\"\n    },\n    {\n      \"app\": \"meet me\",\n      \"id\": 379,\n      \"method\": \"GET\",\n      \"url\": \"https://www.meetme.com/{username}\",\n      \"valid\": \"response.status == 200 and '<title>Meet people like ' in responseContent\"\n    },\n    {\n      \"app\": \"megamodels pl\",\n      \"id\": 380,\n      \"method\": \"GET\",\n      \"url\": \"http://megamodels.pl/{username}\",\n      \"valid\": \"response.status == 200 and 'Portfolio' in responseContent\"\n    },\n    {\n      \"app\": \"memrise\",\n      \"id\": 381,\n      \"method\": \"GET\",\n      \"url\": \"https://app.memrise.com/user/{username}/\",\n      \"valid\": \"response.status == 200 and 'followers' in responseContent\"\n    },\n    {\n      \"app\": \"Microsoft Technet Community\",\n      \"id\": 382,\n      \"method\": \"GET\",\n      \"url\": \"https://social.technet.microsoft.com/profile/{username}/\",\n      \"valid\": \"response.status == 200 and 's Profile</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Minds\",\n      \"id\": 383,\n      \"method\": \"GET\",\n      \"url\": \"https://www.minds.com/{username}/\",\n      \"valid\": \"response.status == 200 and ') | Minds</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Mistrzowie\",\n      \"id\": 384,\n      \"method\": \"GET\",\n      \"url\": \"https://mistrzowie.org/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Profil u\\u017cytkownika' in responseContent\"\n    },\n    {\n      \"app\": \"Mixi\",\n      \"id\": 385,\n      \"method\": \"GET\",\n      \"url\": \"https://mixi.jp/view_community.pl?id={username}\",\n      \"valid\": \"response.status == 200 and '| mixi\\u30b3\\u30df\\u30e5\\u30cb\\u30c6\\u30a3</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Mmorpg\",\n      \"id\": 386,\n      \"method\": \"GET\",\n      \"url\": \"https://forums.mmorpg.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'MMORPG.com Forums' in responseContent\"\n    },\n    {\n      \"app\": \"Mod DB\",\n      \"id\": 387,\n      \"method\": \"GET\",\n      \"url\": \"https://www.moddb.com/members/{username}\",\n      \"valid\": \"response.status == 200 and 'joined <time' in responseContent\"\n    },\n    {\n      \"app\": \"Moneysavingexpert\",\n      \"id\": 388,\n      \"method\": \"GET\",\n      \"url\": \"https://forums.moneysavingexpert.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Profile' in responseContent\"\n    },\n    {\n      \"app\": \"Motokiller\",\n      \"id\": 389,\n      \"method\": \"GET\",\n      \"url\": \"https://mklr.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Zamieszcza materia\\u0142y od:' in responseContent\"\n    },\n    {\n      \"app\": \"moxfield\",\n      \"id\": 390,\n      \"method\": \"GET\",\n      \"url\": \"https://www.moxfield.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 'Moxfield Profile' in responseContent\"\n    },\n    {\n      \"app\": \"Muck Rack\",\n      \"id\": 391,\n      \"method\": \"GET\",\n      \"url\": \"https://muckrack.com/{username}\",\n      \"valid\": \"response.status == 200 and 'on Muck Rack' in responseContent\"\n    },\n    {\n      \"app\": \"MyAnimeList\",\n      \"id\": 392,\n      \"method\": \"GET\",\n      \"url\": \"https://myanimelist.net/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Profile - MyAnimeList.net' in responseContent\"\n    },\n    {\n      \"app\": \"MyBuilder com\",\n      \"id\": 393,\n      \"method\": \"GET\",\n      \"url\": \"https://www.mybuilder.com/profile/view/{username}\",\n      \"valid\": \"response.status == 200 and 'feedback' in responseContent\"\n    },\n    {\n      \"app\": \"MyFitnessPal\",\n      \"id\": 394,\n      \"method\": \"GET\",\n      \"url\": \"https://www.myfitnesspal.com/user/{username}/status\",\n      \"valid\": \"response.status == 200 and 's profile | MyFitnessPal.com</title>' in responseContent\"\n    },\n    {\n      \"app\": \"my_instants\",\n      \"id\": 395,\n      \"method\": \"GET\",\n      \"url\": \"https://www.myinstants.com/en/profile/{username}/\",\n      \"valid\": \"response.status == 200 and ' | Myinstants</title>' in responseContent\"\n    },\n    {\n      \"app\": \"MyLot\",\n      \"id\": 396,\n      \"method\": \"GET\",\n      \"url\": \"https://www.mylot.com/{username}\",\n      \"valid\": \"response.status == 200 and 'on myLot</title>' in responseContent\"\n    },\n    {\n      \"app\": \"mym fans\",\n      \"id\": 397,\n      \"method\": \"GET\",\n      \"url\": \"https://mym.fans/{username}\",\n      \"valid\": \"response.status == 200 and 'posts' in responseContent\"\n    },\n    {\n      \"app\": \"NameMC\",\n      \"id\": 398,\n      \"method\": \"GET\",\n      \"url\": \"https://namemc.com/search?q={username}\",\n      \"valid\": \"response.status == 200 and 'Status: Unavailable' in responseContent\"\n    },\n    {\n      \"app\": \"naija_planet\",\n      \"id\": 399,\n      \"method\": \"GET\",\n      \"url\": \"https://naijaplanet.com/{username}\",\n      \"valid\": \"response.status == 200 and 'dating Profile, ' in responseContent\"\n    },\n    {\n      \"app\": \"nairaland\",\n      \"id\": 400,\n      \"method\": \"GET\",\n      \"url\": \"https://www.nairaland.com/{username}\",\n      \"valid\": \"response.status == 200 and 's Profile' in responseContent\"\n    },\n    {\n      \"app\": \"NaturalNews\",\n      \"id\": 401,\n      \"method\": \"GET\",\n      \"url\": \"https://naturalnews.com/author/{username}/\",\n      \"valid\": \"response.status == 200 and 'All posts by' in responseContent\"\n    },\n    {\n      \"app\": \"Naver\",\n      \"id\": 402,\n      \"method\": \"GET\",\n      \"url\": \"https://blog.naver.com/{username}\",\n      \"valid\": \"response.status == 200 and ': \\ub124\\uc774\\ubc84 \\ube14\\ub85c\\uadf8' in responseContent\"\n    },\n    {\n      \"app\": \"netvibes\",\n      \"id\": 403,\n      \"method\": \"GET\",\n      \"url\": \"https://www.netvibes.com/{username}\",\n      \"valid\": \"response.status == 200 and 'userId' in responseContent\"\n    },\n    {\n      \"app\": \"Newgrounds\",\n      \"id\": 404,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.newgrounds.com/\",\n      \"valid\": \"response.status == 200 and 'fans' in responseContent\"\n    },\n    {\n      \"app\": \"newmeet\",\n      \"id\": 405,\n      \"method\": \"GET\",\n      \"url\": \"https://www.newmeet.com/en/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'The profile of' in responseContent\"\n    },\n    {\n      \"app\": \"NotABug\",\n      \"id\": 406,\n      \"method\": \"GET\",\n      \"url\": \"https://notabug.org/{username}\",\n      \"valid\": \"response.status == 200 and 'followers and is following' in responseContent\"\n    },\n    {\n      \"app\": \"oglaszamy24h pl\",\n      \"id\": 407,\n      \"method\": \"GET\",\n      \"url\": \"https://oglaszamy24h.pl/profil,{username}\",\n      \"valid\": \"response.status == 200 and 'Profil u\\u017cytkownika:' in responseContent\"\n    },\n    {\n      \"app\": \"ok ru\",\n      \"id\": 408,\n      \"method\": \"GET\",\n      \"url\": \"https://ok.ru/{username}\",\n      \"valid\": \"response.status == 200 and '| OK' in responseContent and 'This page does not exist on OK' not in responseContent\"\n    },\n    {\n      \"app\": \"okidoki\",\n      \"id\": 409,\n      \"method\": \"GET\",\n      \"url\": \"https://m.okidoki.ee/ru/users/{username}/\",\n      \"valid\": \"response.status == 200 and '\\u041f\\u043e\\u043b\\u044c\\u0437\\u043e\\u0432\\u0430\\u0442\\u0435\\u043b\\u044c' in responseContent\"\n    },\n    {\n      \"app\": \"olx\",\n      \"id\": 410,\n      \"method\": \"GET\",\n      \"url\": \"https://www.olx.pl/oferty/uzytkownik/{username}/\",\n      \"valid\": \"response.status == 200 and 'Obserwuj wyszukiwanie' in responseContent\"\n    },\n    {\n      \"app\": \"Opencollective\",\n      \"id\": 411,\n      \"method\": \"GET\",\n      \"url\": \"https://opencollective.com/{username}\",\n      \"valid\": \"response.status == 200 and '- Open Collective' in responseContent\"\n    },\n    {\n      \"app\": \"OpenStreetMap\",\n      \"id\": 412,\n      \"method\": \"GET\",\n      \"url\": \"https://www.openstreetmap.org/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Mapper since:' in responseContent\"\n    },\n    {\n      \"app\": \"OPGG\",\n      \"id\": 413,\n      \"method\": \"GET\",\n      \"url\": \"https://eune.op.gg/summoners/eune/{username}\",\n      \"valid\": \"response.status == 200 and '- Summoner Stats - League of Legends' in responseContent\"\n    },\n    {\n      \"app\": \"Orbys\",\n      \"id\": 414,\n      \"method\": \"GET\",\n      \"url\": \"https://orbys.net/{username}\",\n      \"valid\": \"response.status == 200 and 'profile_user_image' in responseContent\"\n    },\n    {\n      \"app\": \"osu!\",\n      \"id\": 415,\n      \"method\": \"GET\",\n      \"url\": \"https://osu.ppy.sh/users/{username}\",\n      \"valid\": \"response.status == 302 and '\\\\' in responseContent\"\n    },\n    {\n      \"app\": \"Our Freedom Book\",\n      \"id\": 416,\n      \"method\": \"GET\",\n      \"url\": \"https://www.ourfreedombook.com/{username}\",\n      \"valid\": \"response.status == 200 and 'meta property=\\\"og:' in responseContent\"\n    },\n    {\n      \"app\": \"ow ly\",\n      \"id\": 417,\n      \"method\": \"GET\",\n      \"url\": \"http://ow.ly/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Images' in responseContent\"\n    },\n    {\n      \"app\": \"palnet\",\n      \"id\": 418,\n      \"method\": \"GET\",\n      \"url\": \"https://www.palnet.io/@{username}\",\n      \"valid\": \"response.status == 200 and ' - PALnet' in responseContent\"\n    },\n    {\n      \"app\": \"Parler\",\n      \"id\": 419,\n      \"method\": \"GET\",\n      \"url\": \"https://parler.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'People to Follow' in responseContent\"\n    },\n    {\n      \"app\": \"Parler archived profile\",\n      \"id\": 420,\n      \"method\": \"GET\",\n      \"url\": \"http://archive.org/wayback/available?url=https://parler.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and '\\\"archived_snapshots\\\": {\\\"closest\\\"' in responseContent\"\n    },\n    {\n      \"app\": \"Parler archived posts\",\n      \"id\": 421,\n      \"method\": \"GET\",\n      \"url\": \"http://archive.org/wayback/available?url=https://parler.com/profile/{username}/posts\",\n      \"valid\": \"response.status == 200 and '\\\"archived_snapshots\\\": {\\\"closest\\\"' in responseContent\"\n    },\n    {\n      \"app\": \"PatientsLikeMe\",\n      \"id\": 422,\n      \"method\": \"GET\",\n      \"url\": \"https://www.patientslikeme.com/members/{username}\",\n      \"valid\": \"response.status == 200 and 's profile | PatientsLikeMe</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Patronite\",\n      \"id\": 423,\n      \"method\": \"GET\",\n      \"url\": \"https://patronite.pl/{username}\",\n      \"valid\": \"response.status == 200 and 'Zosta\\u0144 Patronem' in responseContent\"\n    },\n    {\n      \"app\": \"PCGamer\",\n      \"id\": 424,\n      \"method\": \"GET\",\n      \"url\": \"https://forums.pcgamer.com/members/{username}/\",\n      \"valid\": \"response.status == 200 and 'Joined' in responseContent\"\n    },\n    {\n      \"app\": \"PCPartPicker\",\n      \"id\": 425,\n      \"method\": \"GET\",\n      \"url\": \"https://pcpartpicker.com/user/{username}/\",\n      \"valid\": \"response.status == 200 and 'class=\\\"active\\\"' in responseContent\"\n    },\n    {\n      \"app\": \"Pewex\",\n      \"id\": 426,\n      \"method\": \"GET\",\n      \"url\": \"https://retro.pewex.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Zamieszcza eksponaty od:' in responseContent\"\n    },\n    {\n      \"app\": \"Photoblog\",\n      \"id\": 427,\n      \"method\": \"GET\",\n      \"url\": \"https://www.photoblog.com/{username}/\",\n      \"valid\": \"response.status == 200 and '/following/' in responseContent\"\n    },\n    {\n      \"app\": \"PhotoBucket\",\n      \"id\": 428,\n      \"method\": \"GET\",\n      \"url\": \"https://app.photobucket.com/u/{username}\",\n      \"valid\": \"response.status == 200 and 'content=\\\\'My Library' in responseContent\"\n    },\n    {\n      \"app\": \"Picsart\",\n      \"id\": 429,\n      \"method\": \"GET\",\n      \"url\": \"https://picsart.com/u/{username}\",\n      \"valid\": \"response.status == 200 and 'Profiles on Picsart' in responseContent\"\n    },\n    {\n      \"app\": \"Piekielni\",\n      \"id\": 430,\n      \"method\": \"GET\",\n      \"url\": \"https://piekielni.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Zamieszcza historie od:' in responseContent\"\n    },\n    {\n      \"app\": \"pikabu\",\n      \"id\": 431,\n      \"method\": \"GET\",\n      \"url\": \"https://pikabu.ru/@{username}\",\n      \"valid\": \"response.status == 200 and '\\u2014 \\u0432\\u0441\\u0435 \\u043f\\u043e\\u0441\\u0442\\u044b \\u043f\\u043e\\u043b\\u044c\\u0437\\u043e\\u0432\\u0430\\u0442\\u0435\\u043b\\u044f' in responseContent\"\n    },\n    {\n      \"app\": \"PinkBike\",\n      \"id\": 432,\n      \"method\": \"GET\",\n      \"url\": \"https://www.pinkbike.com/u/{username}/\",\n      \"valid\": \"response.status == 200 and 'on Pinkbike</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Plurk\",\n      \"id\": 433,\n      \"method\": \"GET\",\n      \"url\": \"https://www.plurk.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Profile views' in responseContent\"\n    },\n    {\n      \"app\": \"Pokec\",\n      \"id\": 434,\n      \"method\": \"GET\",\n      \"url\": \"https://pokec.azet.sk/{username}\",\n      \"valid\": \"response.status == 200 and 'idReportedUser' in responseContent\"\n    },\n    {\n      \"app\": \"pokemonshowdown\",\n      \"id\": 435,\n      \"method\": \"GET\",\n      \"url\": \"https://pokemonshowdown.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 'Official ladder' in responseContent\"\n    },\n    {\n      \"app\": \"Pokerstrategy\",\n      \"id\": 436,\n      \"method\": \"GET\",\n      \"url\": \"http://www.pokerstrategy.net/user/{username}/profile/\",\n      \"valid\": \"response.status == 200 and 'User profile for' in responseContent\"\n    },\n    {\n      \"app\": \"Polchat pl\",\n      \"id\": 437,\n      \"method\": \"GET\",\n      \"url\": \"https://polczat.pl/forum/profile/{username}/\",\n      \"valid\": \"response.status == 200 and 'Historia wpis\\u00f3w' in responseContent\"\n    },\n    {\n      \"app\": \"policja2009\",\n      \"id\": 438,\n      \"method\": \"GET\",\n      \"url\": \"http://www.policja2009.fora.pl/search.php?search_author={username}\",\n      \"valid\": \"response.status == 200 and 'Znaleziono' in responseContent\"\n    },\n    {\n      \"app\": \"Poll Everywhere\",\n      \"id\": 439,\n      \"method\": \"GET\",\n      \"url\": \"https://pollev.com/proxy/api/users/{username}\",\n      \"valid\": \"response.status == 200 and 'name' in responseContent\"\n    },\n    {\n      \"app\": \"pol social\",\n      \"id\": 440,\n      \"method\": \"GET\",\n      \"url\": \"https://pol.social/@{username}\",\n      \"valid\": \"response.status == 200 and '@pol.social' in responseContent\"\n    },\n    {\n      \"app\": \"polygon\",\n      \"id\": 441,\n      \"method\": \"GET\",\n      \"url\": \"https://www.polygon.com/users/{username}\",\n      \"valid\": \"response.status == 200 and '- Polygon' in responseContent\"\n    },\n    {\n      \"app\": \"popl\",\n      \"id\": 442,\n      \"method\": \"GET\",\n      \"url\": \"https://poplme.co/{username}\",\n      \"valid\": \"response.status == 200 and 'MuiTypography-root MuiTypography-body1 css-kj7pvm' in responseContent\"\n    },\n    {\n      \"app\": \"Pornhub Porn Stars\",\n      \"id\": 443,\n      \"method\": \"GET\",\n      \"url\": \"https://www.pornhub.com/pornstar/{username}\",\n      \"valid\": \"response.status == 200 and 'Pornstar Rank' in responseContent\"\n    },\n    {\n      \"app\": \"Pornhub Users\",\n      \"id\": 444,\n      \"method\": \"GET\",\n      \"url\": \"https://www.pornhub.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 's Profile - Pornhub.com</title>' in responseContent\"\n    },\n    {\n      \"app\": \"postcrossing\",\n      \"id\": 445,\n      \"method\": \"GET\",\n      \"url\": \"https://www.postcrossing.com/user/{username}\",\n      \"valid\": \"response.status == 200 and ', from' in responseContent\"\n    },\n    {\n      \"app\": \"Producthunt\",\n      \"id\": 446,\n      \"method\": \"GET\",\n      \"url\": \"https://www.producthunt.com/@{username}\",\n      \"valid\": \"response.status == 200 and 's profile on Product Hunt' in responseContent\"\n    },\n    {\n      \"app\": \"promodj\",\n      \"id\": 447,\n      \"method\": \"GET\",\n      \"url\": \"https://promodj.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Favorite styles' in responseContent\"\n    },\n    {\n      \"app\": \"prv pl\",\n      \"id\": 448,\n      \"method\": \"GET\",\n      \"url\": \"https://www.prv.pl/osoba/{username}\",\n      \"valid\": \"response.status == 200 and 'LOGIN' in responseContent\"\n    },\n    {\n      \"app\": \"public\",\n      \"id\": 449,\n      \"method\": \"GET\",\n      \"url\": \"https://public.com/@{username}\",\n      \"valid\": \"response.status == 200 and ') Investment Portfolio on Public' in responseContent\"\n    },\n    {\n      \"app\": \"QUEER\",\n      \"id\": 450,\n      \"method\": \"GET\",\n      \"url\": \"https://queer.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Spo\\u0142eczno\\u015b\\u0107' in responseContent\"\n    },\n    {\n      \"app\": \"quitter pl\",\n      \"id\": 451,\n      \"method\": \"GET\",\n      \"url\": \"https://quitter.pl/profile/{username}\",\n      \"valid\": \"response.status == 200 and '@quitter.pl' in responseContent\"\n    },\n    {\n      \"app\": \"Quora\",\n      \"id\": 452,\n      \"method\": \"GET\",\n      \"url\": \"https://www.quora.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Credentials' in responseContent\"\n    },\n    {\n      \"app\": \"ReblogMe\",\n      \"id\": 453,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.reblogme.com\",\n      \"valid\": \"response.status == 200 and 'blogbody' in responseContent\"\n    },\n    {\n      \"app\": \"redbubble\",\n      \"id\": 454,\n      \"method\": \"GET\",\n      \"url\": \"https://www.redbubble.com/people/{username}/shop?ref=artist_title_name%2F\",\n      \"valid\": \"response.status == 200 and 'Shop | Redbubble' in responseContent\"\n    },\n    {\n      \"app\": \"Researchgate\",\n      \"id\": 455,\n      \"method\": \"GET\",\n      \"url\": \"https://www.researchgate.net/profile/{username}\",\n      \"valid\": \"response.status == 200 and ' | ' in responseContent\"\n    },\n    {\n      \"app\": \"rigcz club\",\n      \"id\": 456,\n      \"method\": \"GET\",\n      \"url\": \"https://rigcz.club/@{username}\",\n      \"valid\": \"response.status == 200 and '@rigcz.club' in responseContent\"\n    },\n    {\n      \"app\": \"risk ru\",\n      \"id\": 457,\n      \"method\": \"GET\",\n      \"url\": \"https://risk.ru/people/{username}\",\n      \"valid\": \"response.status == 200 and '\\u2014 \\u041b\\u044e\\u0434\\u0438 \\u2014 Risk.ru' in responseContent\"\n    },\n    {\n      \"app\": \"rsi\",\n      \"id\": 458,\n      \"method\": \"GET\",\n      \"url\": \"https://robertsspaceindustries.com/citizens/{username}\",\n      \"valid\": \"response.status == 200 and 'CITIZEN DOSSIER' in responseContent\"\n    },\n    {\n      \"app\": \"Ruby Dating\",\n      \"id\": 459,\n      \"method\": \"GET\",\n      \"url\": \"https://ruby.dating/en/users/{username}\",\n      \"valid\": \"response.status == 200 and 'Looks for:' in responseContent\"\n    },\n    {\n      \"app\": \"RumbleChannel\",\n      \"id\": 460,\n      \"method\": \"GET\",\n      \"url\": \"https://rumble.com/c/{username}\",\n      \"valid\": \"response.status == 200 and 'href=https://rumble.com/c/' in responseContent\"\n    },\n    {\n      \"app\": \"RumbleUser\",\n      \"id\": 461,\n      \"method\": \"GET\",\n      \"url\": \"https://rumble.com/user/{username}\",\n      \"valid\": \"response.status == 200 and ' href=https://rumble.com/user/' in responseContent\"\n    },\n    {\n      \"app\": \"Salon24\",\n      \"id\": 462,\n      \"method\": \"GET\",\n      \"url\": \"https://www.salon24.pl/u/{username}/\",\n      \"valid\": \"response.status == 200 and 'Strona g\\u0142\\u00f3wna' in responseContent\"\n    },\n    {\n      \"app\": \"SaraCarterShow\",\n      \"id\": 463,\n      \"method\": \"GET\",\n      \"url\": \"https://saraacarter.com/author/{username}/\",\n      \"valid\": \"response.status == 200 and 'Home | Sara A. Carter' not in responseContent\"\n    },\n    {\n      \"app\": \"ScoutWiki\",\n      \"id\": 464,\n      \"method\": \"GET\",\n      \"url\": \"https://en.scoutwiki.org/User:{username}\",\n      \"valid\": \"response.status == 200 and 'NewPP limit report' in responseContent\"\n    },\n    {\n      \"app\": \"scratch\",\n      \"id\": 465,\n      \"method\": \"GET\",\n      \"url\": \"https://scratch.mit.edu/users/{username}/\",\n      \"valid\": \"response.status == 200 and 'on Scratch</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Seneporno\",\n      \"id\": 466,\n      \"method\": \"GET\",\n      \"url\": \"https://seneporno.com/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Dernier Login' in responseContent\"\n    },\n    {\n      \"app\": \"sentimente\",\n      \"id\": 467,\n      \"method\": \"GET\",\n      \"url\": \"https://www.sentimente.com/amp/{username}.html\",\n      \"valid\": \"response.status == 200 and 'Chat online with' in responseContent\"\n    },\n    {\n      \"app\": \"setlist fm\",\n      \"id\": 468,\n      \"method\": \"GET\",\n      \"url\": \"https://www.setlist.fm/user/{username}\",\n      \"valid\": \"response.status == 200 and 's setlist.fm | setlist.fm</title>' in responseContent\"\n    },\n    {\n      \"app\": \"SFD\",\n      \"id\": 469,\n      \"method\": \"GET\",\n      \"url\": \"https://www.sfd.pl/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Tematy u\\u017cytkownika' in responseContent\"\n    },\n    {\n      \"app\": \"Shanii Writes\",\n      \"id\": 470,\n      \"method\": \"GET\",\n      \"url\": \"https://forum.shanniiwrites.com/u/{username}/summary.json\",\n      \"valid\": \"response.status == 200 and 'topics' in responseContent\"\n    },\n    {\n      \"app\": \"Shesfreaky\",\n      \"id\": 471,\n      \"method\": \"GET\",\n      \"url\": \"https://www.shesfreaky.com/profile/{username}/\",\n      \"valid\": \"response.status == 200 and 's Profile - ShesFreaky</title>' in responseContent\"\n    },\n    {\n      \"app\": \"shopify\",\n      \"id\": 472,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.myshopify.com\",\n      \"valid\": \"response.status == 200 and 'home' in responseContent\"\n    },\n    {\n      \"app\": \"shutterstock\",\n      \"id\": 473,\n      \"method\": \"GET\",\n      \"url\": \"https://www.shutterstock.com/g/{username}\",\n      \"valid\": \"response.status == 200 and '| Shutterstock' in responseContent\"\n    },\n    {\n      \"app\": \"skeb\",\n      \"id\": 474,\n      \"method\": \"GET\",\n      \"url\": \"https://skeb.jp/@{username}\",\n      \"valid\": \"response.status == 200 and ') | Skeb' in responseContent\"\n    },\n    {\n      \"app\": \"Skypli\",\n      \"id\": 475,\n      \"method\": \"GET\",\n      \"url\": \"https://www.skypli.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Skype user profile - skypli.com' in responseContent\"\n    },\n    {\n      \"app\": \"Skyrock\",\n      \"id\": 476,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.skyrock.com/\",\n      \"valid\": \"response.status == 200 and '\\\\'s blog' in responseContent\"\n    },\n    {\n      \"app\": \"slant\",\n      \"id\": 477,\n      \"method\": \"GET\",\n      \"url\": \"https://www.slant.co/users/{username}\",\n      \"valid\": \"response.status == 200 and 's Profile - Slant' in responseContent\"\n    },\n    {\n      \"app\": \"slideshare\",\n      \"id\": 478,\n      \"method\": \"GET\",\n      \"url\": \"https://www.slideshare.net/{username}\",\n      \"valid\": \"response.status == 200 and 'photo user-photo' in responseContent\"\n    },\n    {\n      \"app\": \"slides\",\n      \"id\": 479,\n      \"method\": \"GET\",\n      \"url\": \"https://slides.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Presentations by' in responseContent\"\n    },\n    {\n      \"app\": \"SmashRun\",\n      \"id\": 480,\n      \"method\": \"GET\",\n      \"url\": \"https://smashrun.com/{username}/\",\n      \"valid\": \"response.status == 200 and 'Miles run overall' in responseContent\"\n    },\n    {\n      \"app\": \"smelsy\",\n      \"id\": 481,\n      \"method\": \"GET\",\n      \"url\": \"https://www.smelsy.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Smelsy -' in responseContent\"\n    },\n    {\n      \"app\": \"SmugMug\",\n      \"id\": 482,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.smugmug.com\",\n      \"valid\": \"response.status == 200 and 'schema.org/Person' in responseContent\"\n    },\n    {\n      \"app\": \"smule\",\n      \"id\": 483,\n      \"method\": \"GET\",\n      \"url\": \"https://www.smule.com/{username}/\",\n      \"valid\": \"response.status == 200 and 'followers' in responseContent\"\n    },\n    {\n      \"app\": \"soc citizen4 eu\",\n      \"id\": 484,\n      \"method\": \"GET\",\n      \"url\": \"https://soc.citizen4.eu/profile/{username}/profile\",\n      \"valid\": \"response.status == 200 and '@soc.citizen4.eu' in responseContent\"\n    },\n    {\n      \"app\": \"SoliKick\",\n      \"id\": 485,\n      \"method\": \"GET\",\n      \"url\": \"https://solikick.com/-{username}\",\n      \"valid\": \"response.status == 200 and 'page_guest_users-view' in responseContent\"\n    },\n    {\n      \"app\": \"SoundCloud\",\n      \"id\": 486,\n      \"method\": \"GET\",\n      \"url\": \"https://soundcloud.com/{username}\",\n      \"valid\": \"response.status == 200 and 'SoundCloud</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Soup\",\n      \"id\": 487,\n      \"method\": \"GET\",\n      \"url\": \"https://www.soup.io/author/{username}\",\n      \"valid\": \"response.status == 200 and 'Author at Soup.io' in responseContent\"\n    },\n    {\n      \"app\": \"SpankPay\",\n      \"id\": 488,\n      \"method\": \"GET\",\n      \"url\": \"https://pay-api.spankchain.com/profiles/{username}?allowBlank=\",\n      \"valid\": \"response.status == 200 and 'spankpayApiKey' in responseContent\"\n    },\n    {\n      \"app\": \"Speaker Deck\",\n      \"id\": 489,\n      \"method\": \"GET\",\n      \"url\": \"https://speakerdeck.com/{username}/\",\n      \"valid\": \"response.status == 200 and ') on Speaker Deck</title>' in responseContent\"\n    },\n    {\n      \"app\": \"SpiceWorks\",\n      \"id\": 490,\n      \"method\": \"GET\",\n      \"url\": \"https://community.spiceworks.com/people/{username}\",\n      \"valid\": \"response.status == 200 and 'Portfolio of IT Projects - Spiceworks' in responseContent\"\n    },\n    {\n      \"app\": \"sporcle\",\n      \"id\": 491,\n      \"method\": \"GET\",\n      \"url\": \"https://www.sporcle.com/user/{username}/people/\",\n      \"valid\": \"response.status == 200 and 's Sporcle Friends' in responseContent\"\n    },\n    {\n      \"app\": \"steemit\",\n      \"id\": 492,\n      \"method\": \"GET\",\n      \"url\": \"https://steemit.com/@{username}\",\n      \"valid\": \"response.status == 200 and 'blog' in responseContent\"\n    },\n    {\n      \"app\": \"StoryCorps\",\n      \"id\": 493,\n      \"method\": \"GET\",\n      \"url\": \"https://archive.storycorps.org/user/{username}/\",\n      \"valid\": \"response.status == 200 and 'archive author' in responseContent\"\n    },\n    {\n      \"app\": \"Stripchat\",\n      \"id\": 494,\n      \"method\": \"GET\",\n      \"url\": \"https://stripchat.com/{username}\",\n      \"valid\": \"response.status == 200 and 'I Do in My Shows:' in responseContent\"\n    },\n    {\n      \"app\": \"sukebei nyaa si\",\n      \"id\": 495,\n      \"method\": \"GET\",\n      \"url\": \"https://sukebei.nyaa.si/user/{username}\",\n      \"valid\": \"response.status == 200 and 's torrents' in responseContent\"\n    },\n    {\n      \"app\": \"Suzuri\",\n      \"id\": 496,\n      \"method\": \"GET\",\n      \"url\": \"https://suzuri.jp/{username}\",\n      \"valid\": \"response.status == 200 and 'Items' in responseContent\"\n    },\n    {\n      \"app\": \"Swalifnet\",\n      \"id\": 497,\n      \"method\": \"GET\",\n      \"url\": \"https://www.swalifnet.net/user/{username}\",\n      \"valid\": \"response.status == 200 and ' - ' in responseContent\"\n    },\n    {\n      \"app\": \"szmer info\",\n      \"id\": 498,\n      \"method\": \"GET\",\n      \"url\": \"https://szmer.info/u/{username}\",\n      \"valid\": \"response.status == 200 and 'Joined' in responseContent\"\n    },\n    {\n      \"app\": \"tabletoptournament\",\n      \"id\": 499,\n      \"method\": \"GET\",\n      \"url\": \"https://www.tabletoptournaments.net/eu/player/{username}\",\n      \"valid\": \"response.status == 200 and '- Player Profile | T\\u00b3 - TableTop Tournaments' in responseContent\"\n    },\n    {\n      \"app\": \"Tagged\",\n      \"id\": 500,\n      \"method\": \"GET\",\n      \"url\": \"https://secure.tagged.com/{username}\",\n      \"valid\": \"response.status == 200 and 's Profile</title>' in responseContent\"\n    },\n    {\n      \"app\": \"TamTam\",\n      \"id\": 501,\n      \"method\": \"GET\",\n      \"url\": \"https://tamtam.chat/{username}\",\n      \"valid\": \"response.status == 200 and 'deeplink=tamtam://chat/' in responseContent\"\n    },\n    {\n      \"app\": \"Tanuki pl\",\n      \"id\": 502,\n      \"method\": \"GET\",\n      \"url\": \"https://tanuki.pl/profil/{username}\",\n      \"valid\": \"response.status == 200 and 'Do\\u0142\\u0105czy\\u0142' in responseContent\"\n    },\n    {\n      \"app\": \"Taringa\",\n      \"id\": 503,\n      \"method\": \"GET\",\n      \"url\": \"https://www.taringa.net/{username}\",\n      \"valid\": \"response.status == 200 and ' en Taringa!</title>' in responseContent\"\n    },\n    {\n      \"app\": \"taskrabbit\",\n      \"id\": 504,\n      \"method\": \"GET\",\n      \"url\": \"https://www.taskrabbit.com/profile/{username}/about\",\n      \"valid\": \"response.status == 200 and '\\u2019s Profile' in responseContent\"\n    },\n    {\n      \"app\": \"Teamtreehouse\",\n      \"id\": 505,\n      \"method\": \"GET\",\n      \"url\": \"https://teamtreehouse.com/{username}\",\n      \"valid\": \"response.status == 200 and 'Member Since' in responseContent\"\n    },\n    {\n      \"app\": \"Tellonym\",\n      \"id\": 506,\n      \"method\": \"GET\",\n      \"url\": \"https://tellonym.me/{username}\",\n      \"valid\": \"response.status == 200 and 'on Tellonym' in responseContent\"\n    },\n    {\n      \"app\": \"TF2 Backpack Examiner\",\n      \"id\": 507,\n      \"method\": \"GET\",\n      \"url\": \"http://www.tf2items.com/id/{username}/\",\n      \"valid\": \"response.status == 200 and '<title>TF2 Backpack -' in responseContent\"\n    },\n    {\n      \"app\": \"tfl net pl\",\n      \"id\": 508,\n      \"method\": \"GET\",\n      \"url\": \"https://tfl.net.pl/@{username}\",\n      \"valid\": \"response.status == 200 and '@tfl.net.pl' in responseContent\"\n    },\n    {\n      \"app\": \"thegatewaypundit\",\n      \"id\": 509,\n      \"method\": \"GET\",\n      \"url\": \"https://www.thegatewaypundit.com/author/{username}/\",\n      \"valid\": \"response.status == 200 and 'summary' in responseContent\"\n    },\n    {\n      \"app\": \"theguardian\",\n      \"id\": 510,\n      \"method\": \"GET\",\n      \"url\": \"https://www.theguardian.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'https://www.theguardian.com/profile/' in responseContent\"\n    },\n    {\n      \"app\": \"themeforest\",\n      \"id\": 511,\n      \"method\": \"GET\",\n      \"url\": \"https://themeforest.net/user/{username}\",\n      \"valid\": \"response.status == 200 and 's profile on ThemeForest' in responseContent\"\n    },\n    {\n      \"app\": \"Thetattooforum\",\n      \"id\": 512,\n      \"method\": \"GET\",\n      \"url\": \"https://www.thetattooforum.com/members/{username}/\",\n      \"valid\": \"response.status == 200 and 'Insert This Gallery' in responseContent\"\n    },\n    {\n      \"app\": \"TotalWar\",\n      \"id\": 513,\n      \"method\": \"GET\",\n      \"url\": \"https://forums.totalwar.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Total War Forums' in responseContent\"\n    },\n    {\n      \"app\": \"TrackmaniaLadder\",\n      \"id\": 514,\n      \"method\": \"GET\",\n      \"url\": \"https://en.tm-ladder.com/{username}_rech.php\",\n      \"valid\": \"response.status == 200 and 'Login type :' in responseContent\"\n    },\n    {\n      \"app\": \"tradingview\",\n      \"id\": 515,\n      \"method\": \"GET\",\n      \"url\": \"https://www.tradingview.com/u/{username}/\",\n      \"valid\": \"response.status == 200 and '\\u2014 Trading Ideas &amp;' in responseContent\"\n    },\n    {\n      \"app\": \"trakt\",\n      \"id\": 516,\n      \"method\": \"GET\",\n      \"url\": \"https://trakt.tv/users/{username}\",\n      \"valid\": \"response.status == 200 and 's profile - Trakt' in responseContent\"\n    },\n    {\n      \"app\": \"tripadvisor\",\n      \"id\": 517,\n      \"method\": \"GET\",\n      \"url\": \"https://www.tripadvisor.com/Profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Contributions' in responseContent\"\n    },\n    {\n      \"app\": \"tumblr\",\n      \"id\": 518,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.tumblr.com\",\n      \"valid\": \"response.status == 200 and 'avatar' in responseContent\"\n    },\n    {\n      \"app\": \"Tunefind\",\n      \"id\": 519,\n      \"method\": \"GET\",\n      \"url\": \"https://www.tunefind.com/user/profile/{username}\",\n      \"valid\": \"response.status == 200 and 'Achievements' in responseContent\"\n    },\n    {\n      \"app\": \"Twitcasting\",\n      \"id\": 520,\n      \"method\": \"GET\",\n      \"url\": \"https://twitcasting.tv/{username}\",\n      \"valid\": \"response.status == 200 and 'Live History' in responseContent\"\n    },\n    {\n      \"app\": \"Twitch\",\n      \"id\": 521,\n      \"method\": \"GET\",\n      \"url\": \"https://twitchtracker.com/search?q={username}\",\n      \"valid\": \"response.status == 200 and 'Found Exact Match' in responseContent\"\n    },\n    {\n      \"app\": \"Twitter archived profile\",\n      \"id\": 522,\n      \"method\": \"GET\",\n      \"url\": \"http://archive.org/wayback/available?url=https://twitter.com/{username}\",\n      \"valid\": \"response.status == 200 and '\\\"archived_snapshots\\\": {\\\"closest\\\"' in responseContent\"\n    },\n    {\n      \"app\": \"Twitter archived tweets\",\n      \"id\": 523,\n      \"method\": \"GET\",\n      \"url\": \"http://archive.org/wayback/available?url=https://twitter.com/{username}/status/*\",\n      \"valid\": \"response.status == 200 and '\\\"archived_snapshots\\\": {\\\"closest\\\"' in responseContent\"\n    },\n    {\n      \"app\": \"twpro\",\n      \"id\": 524,\n      \"method\": \"GET\",\n      \"url\": \"https://twpro.jp/{username}\",\n      \"valid\": \"response.status == 200 and '\\u304a\\u3068\\u306a\\u308a\\u3055\\u3093' in responseContent\"\n    },\n    {\n      \"app\": \"Udemy\",\n      \"id\": 525,\n      \"method\": \"GET\",\n      \"url\": \"https://www.udemy.com/user/{username}/\",\n      \"valid\": \"response.status == 200 and 'profile' in soup.find('meta', attrs={'name':'type'})['content']\"\n    },\n    {\n      \"app\": \"uid\",\n      \"id\": 526,\n      \"method\": \"GET\",\n      \"url\": \"http://uid.me/{username}\",\n      \"valid\": \"response.status == 200 and '- uID.me' in responseContent\"\n    },\n    {\n      \"app\": \"Ultras Diary\",\n      \"id\": 527,\n      \"method\": \"GET\",\n      \"url\": \"http://ultrasdiary.pl/u/{username}/\",\n      \"valid\": \"response.status == 200 and 'Mecze wyjazdowe:' in responseContent\"\n    },\n    {\n      \"app\": \"ulub pl\",\n      \"id\": 528,\n      \"method\": \"GET\",\n      \"url\": \"http://ulub.pl/profil/{username}\",\n      \"valid\": \"response.status == 200 and 'Muzyka (' in responseContent\"\n    },\n    {\n      \"app\": \"unsplash\",\n      \"id\": 529,\n      \"method\": \"GET\",\n      \"url\": \"https://unsplash.com/@{username}\",\n      \"valid\": \"response.status == 200 and '| Unsplash Photo Community' in responseContent\"\n    },\n    {\n      \"app\": \"untappd\",\n      \"id\": 530,\n      \"method\": \"GET\",\n      \"url\": \"https://untappd.com/user/{username}/\",\n      \"valid\": \"response.status == 200 and 'on Untappd</title>' in responseContent\"\n    },\n    {\n      \"app\": \"USA Life\",\n      \"id\": 531,\n      \"method\": \"GET\",\n      \"url\": \"https://usa.life/{username}\",\n      \"valid\": \"response.status == 200 and 'Please log in to like, share and comment' in responseContent\"\n    },\n    {\n      \"app\": \"Vero\",\n      \"id\": 532,\n      \"method\": \"GET\",\n      \"url\": \"https://vero.co/{username}\",\n      \"valid\": \"response.status == 200 and 'on VERO\\u2122</title>' in responseContent\"\n    },\n    {\n      \"app\": \"vibilagare\",\n      \"id\": 533,\n      \"method\": \"GET\",\n      \"url\": \"https://www.vibilagare.se/users/{username}\",\n      \"valid\": \"response.status == 200 and 'Profil p\\u00e5 vibilagare.se' in responseContent\"\n    },\n    {\n      \"app\": \"viddler\",\n      \"id\": 534,\n      \"method\": \"GET\",\n      \"url\": \"https://www.viddler.com/channel/{username}/\",\n      \"valid\": \"response.status == 200 and 'profile-details' in responseContent\"\n    },\n    {\n      \"app\": \"VIP-blog\",\n      \"id\": 535,\n      \"method\": \"GET\",\n      \"url\": \"http://{username}.vip-blog.com\",\n      \"valid\": \"response.status == 200 and 'blog : ' in responseContent\"\n    },\n    {\n      \"app\": \"Virustotal\",\n      \"id\": 536,\n      \"method\": \"GET\",\n      \"url\": \"https://www.virustotal.com/gui/user/{username}\",\n      \"valid\": \"response.status == 200 and 'USER PROFILE' in responseContent\"\n    },\n    {\n      \"app\": \"Vivino\",\n      \"id\": 537,\n      \"method\": \"GET\",\n      \"url\": \"https://www.vivino.com/users/{username}\",\n      \"valid\": \"response.status == 200 and '<!-- User details -->' in responseContent\"\n    },\n    {\n      \"app\": \"vizjer pl\",\n      \"id\": 538,\n      \"method\": \"GET\",\n      \"url\": \"https://vizjer.pl/uzytkownik/{username}\",\n      \"valid\": \"response.status == 200 and 'Profil u\\u017cytkownika' in responseContent\"\n    },\n    {\n      \"app\": \"VK\",\n      \"id\": 539,\n      \"method\": \"GET\",\n      \"url\": \"https://vk.com/{username}\",\n      \"valid\": \"response.status == 200 and ' | VK</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Voice123\",\n      \"id\": 540,\n      \"method\": \"GET\",\n      \"url\": \"https://voice123.com/api/providers/search/{username}\",\n      \"valid\": \"response.status == 200 and 'user_id' in responseContent\"\n    },\n    {\n      \"app\": \"Voices com\",\n      \"id\": 541,\n      \"method\": \"GET\",\n      \"url\": \"https://www.voices.com/profile/{username}/\",\n      \"valid\": \"response.status == 200 and 'Last Online</h3>' in responseContent\"\n    },\n    {\n      \"app\": \"vsco\",\n      \"id\": 542,\n      \"method\": \"GET\",\n      \"url\": \"https://vsco.co/{username}/gallery\",\n      \"valid\": \"response.status == 200 and '| VSCO' in responseContent\"\n    },\n    {\n      \"app\": \"Wanelo\",\n      \"id\": 543,\n      \"method\": \"GET\",\n      \"url\": \"https://wanelo.co/{username}\",\n      \"valid\": \"response.status == 200 and 'on Wanelo</title>' in responseContent\"\n    },\n    {\n      \"app\": \"warriorforum\",\n      \"id\": 544,\n      \"method\": \"GET\",\n      \"url\": \"https://www.warriorforum.com/members/{username}.html\",\n      \"valid\": \"response.status == 200 and '| Warrior Forum' in responseContent\"\n    },\n    {\n      \"app\": \"watchmemore com\",\n      \"id\": 545,\n      \"method\": \"GET\",\n      \"url\": \"https://api.watchmemore.com/api3/profile/{username}/\",\n      \"valid\": \"response.status == 200 and 'displayName' in responseContent\"\n    },\n    {\n      \"app\": \"wattpad\",\n      \"id\": 546,\n      \"method\": \"GET\",\n      \"url\": \"https://www.wattpad.com/api/v3/users/{username}?fields=username%2Cname%2Cdescription%2Cavatar%2CbackgroundUrl%2CcreateDate%2Clocation%2Cfollowing%2CfollowingRequest%2CnumFollowing%2Cfollower%2CfollowerRequest%2CnumFollowers%2CnumLists%2CnumStoriesPublished%2CvotesReceived%2Cfacebook%2Ctwitter%2Cwebsite%2Csmashwords%2Chighlight_colour%2Chtml_enabled%2Cverified%2Cambassador%2Cwattpad_squad%2Cis_staff%2Cprograms(wattpad_stars)%2CisPrivate%2CisMuted%2CexternalId%2Cnotes\",\n      \"valid\": \"response.status == 200 and 'createDate' in responseContent\"\n    },\n    {\n      \"app\": \"Weasyl\",\n      \"id\": 547,\n      \"method\": \"GET\",\n      \"url\": \"https://www.weasyl.com/~{username}\",\n      \"valid\": \"response.status == 200 and '&#39;s profile \\u2014 Weasyl</title>' in responseContent\"\n    },\n    {\n      \"app\": \"wego\",\n      \"id\": 548,\n      \"method\": \"GET\",\n      \"url\": \"https://wego.social/{username}\",\n      \"valid\": \"response.status == 200 and 'Following</span>' in responseContent\"\n    },\n    {\n      \"app\": \"weheartit\",\n      \"id\": 549,\n      \"method\": \"GET\",\n      \"url\": \"https://weheartit.com/{username}\",\n      \"valid\": \"response.status == 200 and ' on We Heart It</title>' in responseContent\"\n    },\n    {\n      \"app\": \"weibo\",\n      \"id\": 550,\n      \"method\": \"GET\",\n      \"url\": \"https://tw.weibo.com/{username}\",\n      \"valid\": \"response.status == 200 and '\\u7c89\\u7d72' in responseContent\"\n    },\n    {\n      \"app\": \"Wikidot\",\n      \"id\": 551,\n      \"method\": \"GET\",\n      \"url\": \"http://www.wikidot.com/user:info/{username}\",\n      \"valid\": \"response.status == 200 and 'Wikidot.com:' in responseContent\"\n    },\n    {\n      \"app\": \"Wimkin-PublicProfile\",\n      \"id\": 552,\n      \"method\": \"GET\",\n      \"url\": \"https://wimkin.com/{username}\",\n      \"valid\": \"response.status == 200 and 'is on WIMKIN' in responseContent\"\n    },\n    {\n      \"app\": \"wishlistr\",\n      \"id\": 553,\n      \"method\": \"GET\",\n      \"url\": \"https://www.wishlistr.com/profile/{username}/\",\n      \"valid\": \"response.status == 200 and 's profile</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Wolni S\\u0142owianie\",\n      \"id\": 554,\n      \"method\": \"GET\",\n      \"url\": \"https://wolnislowianie.pl/{username}\",\n      \"valid\": \"response.status == 200 and 'O\\u015b czasu' in responseContent\"\n    },\n    {\n      \"app\": \"wordnik\",\n      \"id\": 555,\n      \"method\": \"GET\",\n      \"url\": \"https://www.wordnik.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 'Welcome,' in responseContent\"\n    },\n    {\n      \"app\": \"WordPress\",\n      \"id\": 556,\n      \"method\": \"GET\",\n      \"url\": \"https://profiles.wordpress.org/{username}/\",\n      \"valid\": \"response.status == 200 and 'user-member-since' in responseContent\"\n    },\n    {\n      \"app\": \"WordPress Support\",\n      \"id\": 557,\n      \"method\": \"GET\",\n      \"url\": \"https://wordpress.org/support/users/{username}/\",\n      \"valid\": \"response.status == 200 and 's Profile &#124; WordPress.org' in responseContent\"\n    },\n    {\n      \"app\": \"Wowhead\",\n      \"id\": 558,\n      \"method\": \"GET\",\n      \"url\": \"https://www.wowhead.com/user={username}\",\n      \"valid\": \"response.status == 200 and ' Profile - Wowhead' in responseContent\"\n    },\n    {\n      \"app\": \"Wykop\",\n      \"id\": 559,\n      \"method\": \"GET\",\n      \"url\": \"https://www.wykop.pl/ludzie/{username}/\",\n      \"valid\": \"response.status == 200 and 'Aktywno\\u015b\\u0107 u\\u017cytkownika' in responseContent\"\n    },\n    {\n      \"app\": \"Xanga\",\n      \"id\": 560,\n      \"method\": \"GET\",\n      \"url\": \"http://{username}.xanga.com/\",\n      \"valid\": \"response.status == 200 and 's Xanga Site | Just' in responseContent\"\n    },\n    {\n      \"app\": \"xHamster\",\n      \"id\": 561,\n      \"method\": \"GET\",\n      \"url\": \"https://xhamster.com/users/{username}\",\n      \"valid\": \"response.status == 200 and 's profile | xHamster</title>' in responseContent\"\n    },\n    {\n      \"app\": \"Xing\",\n      \"id\": 562,\n      \"method\": \"GET\",\n      \"url\": \"https://www.xing.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and ' | XING</title>' in responseContent\"\n    },\n    {\n      \"app\": \"XVIDEOS-models\",\n      \"id\": 563,\n      \"method\": \"GET\",\n      \"url\": \"https://www.xvideos.com/models/{username}\",\n      \"valid\": \"response.status == 200 and 'Total video views' in responseContent\"\n    },\n    {\n      \"app\": \"XVIDEOS-profiles\",\n      \"id\": 564,\n      \"method\": \"GET\",\n      \"url\": \"https://www.xvideos.com/profiles/{username}\",\n      \"valid\": \"response.status == 200 and 'page - XVIDEOS.COM' in responseContent\"\n    },\n    {\n      \"app\": \"Yahoo! JAPAN Auction\",\n      \"id\": 565,\n      \"method\": \"GET\",\n      \"url\": \"https://auctions.yahoo.co.jp/follow/list/{username}\",\n      \"valid\": \"response.status == 200 and '\\u51fa\\u54c1\\u8005' in responseContent\"\n    },\n    {\n      \"app\": \"Yazawaj\",\n      \"id\": 566,\n      \"method\": \"GET\",\n      \"url\": \"https://www.yazawaj.com/profile/{username}\",\n      \"valid\": \"response.status == 200 and '\\u0623\\u0646\\u0627' in responseContent\"\n    },\n    {\n      \"app\": \"Yelp\",\n      \"id\": 567,\n      \"method\": \"GET\",\n      \"url\": \"https://www.yelp.com/user_details?userid={username}\",\n      \"valid\": \"response.status == 200 and '\\\\'s Reviews |' in responseContent\"\n    },\n    {\n      \"app\": \"zatrybi pl\",\n      \"id\": 568,\n      \"method\": \"GET\",\n      \"url\": \"https://zatrybi.pl/user/{username}\",\n      \"valid\": \"response.status == 200 and 'Zarejestrowany od:' in responseContent\"\n    },\n    {\n      \"app\": \"Zbiornik\",\n      \"id\": 569,\n      \"method\": \"GET\",\n      \"url\": \"https://mini.zbiornik.com/{username}\",\n      \"valid\": \"response.status == 200 and 'INFO' in responseContent\"\n    },\n    {\n      \"app\": \"zhihu\",\n      \"id\": 570,\n      \"method\": \"GET\",\n      \"url\": \"https://www.zhihu.com/people/{username}\",\n      \"valid\": \"response.status == 200 and 'zhihu:voteupCount' in responseContent\"\n    },\n    {\n      \"app\": \"Zillow\",\n      \"id\": 571,\n      \"method\": \"GET\",\n      \"url\": \"https://www.zillow.com/profile/{username}/\",\n      \"valid\": \"response.status == 200 and '- Real Estate Agent' in responseContent\"\n    },\n    {\n      \"app\": \"zmarsa com\",\n      \"id\": 572,\n      \"method\": \"GET\",\n      \"url\": \"https://zmarsa.com/uzytkownik/{username}/glowna/\",\n      \"valid\": \"response.status == 200 and 'Galeria u\\u017cytkownika' in responseContent\"\n    },\n    {\n      \"app\": \"Zomato\",\n      \"id\": 573,\n      \"method\": \"GET\",\n      \"url\": \"https://www.zomato.com/{username}/foodjourney\",\n      \"valid\": \"response.status == 200 and '| Zomato' in responseContent\"\n    },\n    {\n      \"app\": \"zoomitir\",\n      \"id\": 574,\n      \"method\": \"GET\",\n      \"url\": \"https://www.zoomit.ir/user/{username}/\",\n      \"valid\": \"response.status == 200 and '\\u067e\\u0631\\u0648\\u0641\\u0627\\u06cc\\u0644 - \\u0632\\u0648\\u0645\\u06cc\\u062a' in responseContent\"\n    },\n    {\n      \"app\": \"Zepeto\",\n      \"id\": 575,\n      \"method\": \"GET\",\n      \"url\": \"https://user.zepeto.me/{username}\",\n      \"valid\": \"response.status == 200 and 'https://static-zepeto.pstatic.net/users/' in responseContent\"\n    },\n    {\n      \"app\": \"YouPic\",\n      \"id\": 576,\n      \"method\": \"GET\",\n      \"url\": \"https://youpic.com/{username}/\",\n      \"valid\": \"response.status == 200 and 'PORTFOLIO' in responseContent\"\n    },\n    {\n      \"app\": \"VIEWBUG\",\n      \"id\": 577,\n      \"method\": \"GET\",\n      \"url\": \"https://www.viewbug.com/member/{username}\",\n      \"valid\": \"response.status == 200 and 'Member since ' in responseContent\"\n    },\n    {\n      \"app\": \"Art Limited\",\n      \"id\": 578,\n      \"method\": \"GET\",\n      \"url\": \"https://www.artlimited.net/{username}\",\n      \"valid\": \"response.status == 200 and ' image view' in responseContent\"\n    },\n    {\n      \"app\": \"35photo.pro\",\n      \"id\": 579,\n      \"method\": \"GET\",\n      \"url\": \"https://35photo.pro/{username}/\",\n      \"valid\": \"response.status == 200 and 'view photo' in responseContent\"\n    },\n    {\n      \"app\": \"Purple Port\",\n      \"id\": 580,\n      \"method\": \"GET\",\n      \"url\": \"https://purpleport.com/portfolio/{username}/\",\n      \"valid\": \"response.status == 200 and 'Joined ' in responseContent\"\n    },\n    {\n      \"app\": \"Pixieset\",\n      \"id\": 581,\n      \"method\": \"GET\",\n      \"url\": \"https://{username}.mypixieset.com/\",\n      \"valid\": \"response.status == 200 and 'cannot be found' not in responseContent\"\n    },\n    {\n      \"app\": \"Volo\",\n      \"id\": 582,\n      \"method\": \"GET\",\n      \"url\": \"https://volo.host/{username}\",\n      \"valid\": \"response.status == 200\",\n      \"metadata\": [\n        {\n          \"key\": \"Name\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta', property='og:title')['content']\"\n        },\n        {\n          \"key\": \"Bio\",\n          \"type\": \"generic-data\",\n          \"value\": \"soup.find('meta', property='og:description')['content']\"\n        },\n        {\n          \"key\": \"picture\",\n          \"type\": \"image\",\n          \"value\": \"soup.find('meta', property='og:image')['content']\"\n        }\n      ]\n    }\n  ]\n}\n"
  },
  {
    "path": "resources/wordlist.txt",
    "content": "images\nindex.php\ncss\njs\nwp-content\nwp-content/mysql.sql\nrobots.txt\nassets\nwp-admin\nwp-includes\nimg\nfonts\nlicense.txt\nwp-login.php\nxmlrpc.php\nwp-load.php\nwp-blog-header.php\nwp-trackback.php\nwp-mail.php\nwp-links-opml.php\nvendor\nwp-cron.php\nwp-comments-post.php\nwp-activate.php\nwp-settings.php\nwp-signup.php\nwp-config-sample.php\n.htaccess\nwp-config.php\n.git\nweb.config\nadmin\nuploads\ntemplates\nsitemap.xml\napp\ncgi-bin\nsystem\nthemes\ncomposer.json\ncache\nincludes\nREADME.md\nstatic\nmix-manifest.json\nfiles\nplugins\n.DS_Store\nmedia\napplication\nmanifest.json\nconfig.php\n.well-known\ncomposer.lock\nupload\nscripts\nlib\npublic\nconfig\npdf\ntest\ndata\nmodules\nstorage\n.idea\nnews\nblog\nerror_log\nresources\n.gitignore\ndocs\nvideo\napi\ncatalog\n_notes\nsrc\nlibrary\nscss\nBingSiteAuth.xml\nstyles\nvideos\npackage.json\ninc\nabout\nads.txt\ntest.php\ninstall\nfont\nbootstrap\nnode_modules\nbrowserconfig.xml\ntools\ntmp\ndownload\nbuild\nads\n404.php\nphp\nimage\napi.php\nphp.ini\ninfo.php\ncontent\nckeditor\nfooter.php\ndownloads\ncrossdomain.xml\ncommon\ninclude\nicons\nheader.php\ncontrollers\nmobile\nemail\ntests\nclasses\nbundles\nen\ndist\nsass\nsvg\npackage-lock.json\nkeywords\ncontact\nNginx-1.12_vhost.conf\nlinks.txt\nsupport\nlibs\najax\nsearch\njobs\nwp-snapshots\ntemplate\ncomponents\ncms\nOneSignalSDKWorker.js\ntemp\nloader.php\nbin\nOneSignalSDKUpdaterWorker.js\nmojo-package.sh\nphpinfo.php\ndatabase\nlogs\nsitemap.php\nmaps\ndemo\nstylesheets\nstyle\nsites\nsitemap\nservices\nlogin.php\nhtml\ncontact.php\nsitemaps\nlocal\ngames\naudio\nWEB-INF\nwebfonts\ngulpfile.js\nroutes\njavascript\npages\nnewsletter\nhome\nhelp\nbfb\nviews\nvar\nsearch.php\n_wpeprivate\nTemplates\npackages\nlanguage\nicon\nfrontend\nfavicon\nevents\ndocuments\nweb\ntranslations\nprivacy\ndev\nconsole\napps\nnbproject\nlibraries\ndesign\nckfinder\nbanner\nqa-theme\nqa-tests\nqa-src\nqa-plugin\nqa-lang\nqa-include\nqa-external-example\nqa-content\nqa-cache\nphpunit.xml\nlogout.php\njavascripts\ninit.php\ntheme\nsports\nphotos\nmails\nlog\nlocalization\ncore\nbitrix\nbanners\nbackup\nbackend\nsite\nquotes.txt\nmisc\nlogo\nlang\nengine\ncareers\nabout.php\nScripts\n.vscode\nwp\nstats\nspeedtest\nlogos\nless\ndashboard\napp_dev.php\nsite.webmanifest\nrss\nradio\nfavicons\napp.php\nwebservice\nmail\nlp\njson\nforms\ndb\ncoupons\nasset\nartisan\narchive\ntranslate\nswal\npdfconverter\noverride\nlegal\nglobal\nfront\nfont-awesome\nelFinder-2.1.51\nyss\nwebpack.mix.js\ntransit\ntrailertab\ntextfrompc\nsuccesspages\nsmartSearch\nsingleImage.php\nshop\nsearchtab\nsearchselect\nsearchsafe\nsearchprivacy\nsearchplus\nsearcheasy\nsearchAnonymous\nproducts\npokemon\npackagetracker\noverlay\nonlineforms\nnewsprompt\nmusicsearch\nmoviesearch\nlocalweather\nlanding\ninstall-failed-template\ngamessearch\nforbessearch\nforbesnews\nflirtywallpapers\ndogs\ndailysearch\ncli\ncheapflights\ncelebjunky\nbrowserappreviews\nastrology\naspnet_client\nadministrator\nTV\nRecipeSearch\nRecipe\nLICENSE.txt\n404\n.platform\nwordpress\nupdate\nsounds\nindex.htm\nflash\nes\ncron\narticles\n.elasticbeanstalk\nstore\nscript\nprivacy.php\npartials\nm\nlogs-files\nde\ncaptcha\nCore\nwp-pass.php\nserver.php\nred\npacks\nnewsletters\nindex_arab.php\nindex2_arab.php\nimages.inc.php\nemails\ndoc\ncompany\nblogs\narchivos\nLICENSE\nCONTRIBUTING.md\nAdapter\nwp-register.php\nsw.js\nplayer\nmaintenance\nindex2.php\ni\ngallery\nflags\ncalendar\navatars\nPHPMailer\n.ebextensions\nregister.php\nreadme.rst\nreadme.md\npub\nonline\nlogin\nfeatures\nexport\nerrors\nThumbs.db\nxml\nwp-rss2.php\nwp-rss.php\nwp-rdf.php\nwp-feed.php\nwp-commentsrss2.php\nwp-atom.php\nwidgets\nwebsite\nvendors\nservice-worker.js\nindex_dillema_eng.php\nindex-test.php\nfr\nfont-awesome-4.7.0\nfaq.php\nfancybox\ncss_dillema\ncomposer.phar\ncomposer\nbranding\najax.php\nterms\nprivacy-policy.php\nold\nnews.php\nnew\nmail.php\nfaq\ncron.php\ncontacts\nbower_components\nblocks\narchitecture.md\nImages\nCONTRIBUTORS.md\nwidget\nrss.php\nprotected\npricing\nprepros-6.config\npayment\nimgs\nerror\nconf\nResources\nFront\n.gitattributes\nwebposition\nupload.js\nuntitled folder\nstores.xml\nmembers\nmarketing\nlayouts\nlanguages\nit\nfunctions.php\nforums\nfilpond.js\nfeed\nevents.js.map\nes.js\ncontributing.md\nanalytics\nadmin.php\nSource\nPackages\nConfiguration\n8A659ECD4028395F9B6BE794F6C59ECF.txt\nwordfence-waf.php\nwelcome\nview\nurlrewrite.php\ntpl\ntinymce\nskin\nshortpixel.txt\nsettings-popup\nservice\ns\nresearch\npromo\nprivacy-policy\npreview\npress\npartner\nmain\njquery.js\nimagenes\nhumans.txt\nhtaccess.txt\ngeocity\nfunctions\nfeeds\neditor-build\nblog.php\n_css\nViewerJS\n.ftpquota\nvqmod\ntraining\ntr\nshared\nsetup\nportfolio\nphoto\nnginx.conf\nhtaccess\nhr\nforum\ndownload.php\ndocumentation\ndb.php\ncookie\nconfig.rb\ncommunity\napp-ads.txt\nadmincp\nad\n__MACOSX\n2017\n.editorconfig\nusers_upload\nuser_guide\ntracking\ntesting\ntags\nsources\nsource\nsocial\nrobots.php\npt\nprofiles\nprivate-bip.ogg\nprivate-bip.mp3\npreeti\npdfs\npartners\npage.php\npage\nminify\nmedia-file\niafb.json\nholiday\nfile\nfacebook\nerror.php\ndocker-compose.yml\ncourses\nclass\nchat\ncaptcha.php\nadmin_css\nabout-us.php\nSpryAssets\nFiles\nDockerfile\n2016\n1\n.settings\nwww\ntestimonials\ntest.txt\nsdk\nsales\nreports\nprojects\npersonal\nmarketplace\njquery\nhome.php\ncustom\ncrm\ncontact-us.php\nart\nadminer.php\nadmin_assets\nMETA-INF\n.env\nwiki\nweather\nvideo.php\nupload.php\ntv\ntext.php\ntemplates_c\nstaging\nsoftware\nsitemap.xml.gz\nscript.js\nrss_cookie.txt\npublic_html\nprofile\npodcast\nphpthumb\nmodels\nmin\nmanage\nindex\nincl\nhosting.htaccess\nhelpers\nfrontend_css\nform\nfeedback.php\nfeedback\nfb\ncsv\ncpresources\nclient\ncampaign\nbitbucket-pipelines.yml\nbg\nback\nauth\napple-app-site-association\namp\n_js\nLibrary\nConnections\nCHANGELOG.md\n.apdisk\nuser\ntsconfig.json\ntravel\nthank-you.php\nterms.php\nssl\nschedule\nreviews\nreadme.txt\npromos\npro\nprivate\nplugin\npay\npanel\norder\nnl\nmusic\nmeta\nlicencia.txt\nimport\nimg.php\nhomepage\nguides\nguide\ngenerator\ngame\nfeed.php\nespanol\ne\ndefault\ncontroller\ncheckout\napplications\naffiliates\naccount\nabout-us\nREADME.txt\nyarn.lock\nv3\nuk\ntwitter\ntokens\nthird_party\nteam\nsubscribe\nstatics\nspecial\nsitemap.txt\nsfa\nsafety\nro\nres\nr\nprofile.php\npost\npodcasts\nphpmyadmin\nparking\nmanager\nlocales\nlive\nlearn\nlayout\njetpack-onboarding\ninstaller\nindex.html___jb_bak___\nimage.php\nico\nhead.php\nframework\nform.php\nfontawesome\nfi\nfavicon.gif\nfacilities\nexternal\nemergency\ndraft\ndevelopment\ndebug.log\ncategory.php\ncart.php\ncareers.php\ncareer\nbower.json\nbeta\naws\nactivity\naa\n_template\n_cache\nPublic\n2018\n2014\n.sass-cache\n.github\n.TemporaryItems\nzip\nwp-app.php\nuserfiles\nupdates\nupdate.php\nthumbnails\nterms-of-use\nteams\nswf\nslider\nslick\nsl\nskins\nsina\nsignup.php\nsignup\nru\nrss.xml\nreport.php\nregistration\nregistrar\npurchase\npromotion\nproduct\npopup\nplus28\nmwp_db\nmod\nmenu.php\nmember\nmap.php\nlocations\nloading.gif\nlandings\niphone\nios\ninstall.php\ninfo\nimporting\nimgcache\nhistory\ngps\nfirebase-messaging-sw.js\netc\ndompdf\ndemo.php\ncovid\ncompany.php\nclass.phpmailer.php\nchrome\nar\napply\napk\napc.php\naffiliate\nadm\naccessibility\na\n_redirects\nDocuments\nApplication.cfm\n360\n2020\n2015\n2\n.project\nwow\nwhitepapers\nwc-logs\nvid\nv\nupsell-a2.php\nupsell-a1.php\nupgrade-a2.php\nupgrade-a1.php\nuninstall_templates\nua\ntour\nthankyou\ntags.php\nswfobject.js\nsurvey\nsql\nshowcase\nsecure\nsamples\nsample\ns_code.js\nreseller\nreport\nregister\nredirect.php\nrecaptchalib.php\npython\npublisher\nproject\npl\npics\nphpserver\nphpmailer\nphpMyAdmin\npeople\npatterns\nparents\npage-data\nopensearch.xml\nnewsletter.php\nn\nmp3\nmonitor\nmodule\nmodal\nmap\nmain.php\nma\nlinks\nl\nits\ninternational\nintegrations\nindex1.php\nindex.asp\nhighslide\ngrow\ngoogle\ngenerated\nfpdf\nfarbtastic.js\nfaqs.php\neyeblaster\next\nembed\ndocker\ndisclaimer.php\ndirectory\ndeveloper\ndatabase.php\ncss.php\ncrons\nconnect\nconfiguration.php\ncompliance\ncn\ncm\nclientscript\nclients\ncheck.php\ncba\ncaches\nc\nbrowser\nbooks\nbook\nbackups\nb2b\navatar\nauth.php\nattachments\nasset-manifest.json\napp.js\namazon\nads.php\n_vti_pvt\n_includes\nPIE.htc\nLiveSearchSiteAuth.xml\nDocumentation\n2019\n.svn\n.htpasswd\n.env.example\nwebservices\nwebmail\nwebalizer\nweb-hosting.php\nvps-hosting.php\nvote\nv1\nusers\nuser-uploads\nupdate_pack\ntransfer\ntop.php\ntips\nthumbs\ntest.xml\nterms-of-service.php\ntechnology\nteam.php\nsummer\nstudents\nstory.php\nstart\nstaff\nsss.php\nspirit\nspacer.gif\nsolutions\nsms\nsmall\nsk\nsitemap1.xml\nshell\nsession.php\nserver\nsendmessage.php\nse\nsandbox\nrules_files\nrouter.php\nreview\nrefer.php\nredirect\nquiz\npy\nprototype\nproofs\nproduction\npricing.php\npresentations\nplayers\npictures\npic\npaypal\nother\norder.php\nnotfound.php\nnotes\nmexico\nmenu\nmaterial\nmailing\nlt\nlog.txt\nlicenses\nlayerslider\nlangs\nko\njquery.min.js\njob\ninsurance\ninstallation\nindex3.php\nindex copy.php\nimages2\nie\nhu\nhttpdocs\nhelper\nhelpdesk\nhelp.php\nhandler\ngraphics\ngfx\ngame-servers.php\nfinance\nfimages\nfencing\nfaculty\nexamples\nevent\ness\nelements\ndomain.php\ndomain\ndiversity\ndesktop_app\ndeportes\ndeploy.php\nd\ncz\ncronjobs\ncorporate\ncopyright\ncontest\ncontactus.php\ncontact.htm\ncontact-us\nconfig.codekit\nconferences\nconference\nconcrete\ncomments\ncoming-soon\ncheckout.php\ncfc\ncdn\ncaribbean\ncaptions\ncanada\ncalendar.php\ncal\nbbs\nbackgrounds\nauthor.php\narticle.php\nappasset\naplicaciones\nanimations\nandroid\nalerts\nakamai\naddons\n_vti_cnf\n_scripts\n_private\n_mmServerScripts\n_layouts\n_inc\n_assets\nSozcu_V3\nNews\nLICENSE_AFL.txt\nAssets\nApplications\n.htpasswds\nwp-config-local.php\nwp-cli.yml\nwebcam\nwallpapers\nvsa\nverify\nvb\nutil\nusc_cp.php\nusage\nurl.txt\nuploader\nunsubscribe.php\nui\ntst.ln\ntpc\ntourplanner\ntos.php\ntos\ntoday\nthumbnail\nthumb\ntext.txt\ntesting.php\ntestimonials.php\ntemp.php\ntailwind.config.js\ntag\nt.php\nsync\nsubdomains\nstyleguide\nstyle.scss\nstudyabroad\nstaticmap\nst\nsound\nslopelegend\nsitemgr\nsitemap.htm\nsingle.php\nsignin.php\nsignaturepics\nsignature\nsid\nshare\nsession\nservices.php\nsend.php\nschool\nscholarships\nsc\nroute.planner\nrollingStone\nrobots.txt.dist\nrobot.txt\nroadbook\nrevolution\nresume\nresponsive.css.php\nreg.php\nredesign\nrecovery\nrecaptcha.php\nrec\nreact\nre\nquotes\nquestions\npsych\npsu-edu-assets\nproxy.php\nprovost\nprint.php\nprice\npr\nportal.php\nportal\npool\npolicy\npolicies\nplayer.swf\npicture_library\nphp.php\nphp.ini.sample\nphone\nparts\nparticipacion\norders\nopt\nopenhouse\nop\nonboarding\nohlala\noffers.php\noauth\nnoticiasDeLaManana-2011-05-11.xml\nnode\nnginx.htaccess\nnginx.conf.sample\nnba\nmulticultural\nmoodle\nmondo\nmodel\nmobile-app\nmlp\nmilitary\nmerchant\nmath\nmasterplan\nmaster.css.php\nmaru\nmarketing.php\nmanual\nmanifest\nmanagement\nmaintenance.php\nmain.js\nmailings\nmailer\nmagazine\nlogin2.php\nlocale\nlocal-phpinfo.php\nlnvideos\nliving\nlinks.php\nleadership\nlanacion\nkey.php\nkes\njwplayer\njsonp.php\njs2\nje\niso\nipad\nip\ninvoice.php\ninstaller.php\ninit\ninformation\nindex_files\nindex.cfm\ninclude.php\niletisim.php\nic\nhtml2pdf\nhosting\nhooks\nhola\nheaders\ngr\ngo.php\ngmc\ngif\ngh\ngeo\nfriends\nfree\nframeworks\nfit\nfilemanager\nextres\nenvironment\nentretiempos\nenglish\neid\nedit.php\nebay\ndrugs\ndoubleclick\ndomain-search-result.php\ndl\ndevelopers\ndesktop\ndefault.php\ndashboard.php\ncustomprofilepics\ncustomavatars\ncubs\ncss1\ncsp\ncoupon\ncounseling\ncookies\ncontact_us.php\nconnectors\nconfirm.php\nconfig.codekit3\ncomplement.css.php\ncommon.php\ncomments.php\ncomment\ncloud_theme\nclick.php\nci_sessions\ncharts\nchannel\nchangelog.txt\nccc\ncategory\ncategories\ncast\ncanchallena\nbuttons\nbusiness\nbuilder\nbrand\nborrar_video\nborder-radius.htc\nbk\nbioinformatics\nbig\nbackEnd\nb06871f281fee6b241d60582ae9369b9.ttf\nawards\nattractions\nathletics\nassessment\narchivo\narcade\nappspec.yml\nantibot\nanexos\nalumni\nalpregio\nalpcms\naffiliate-program.php\nadvising\nadvertising\nadmissions\nadmin-panel\naboutus.php\nabc.php\n_src\n_sitespect\n_resources\n_partials\n_img\n_db_backups\n_akamai\n__utils\n_\nWidgets.LN\nVista\nSusana\nServicios.LN\nRollingStone.Net\nRecetas\nREADME\nPersonajes\nPDF\nNew Folder With Items\nNetBoot\nMobile_Detect.php\nMasterLN\nLogos\nLN7\nLN\nLICENSE.md\nKioscoLN\nJardin\nHTML\nGuia.LN\nFourth\nContactoOH.asp\nClub.LN\nCOPYING.txt\nBrando\n674f50d287a8c48dc19ba404d20fe713.eot\n503.php\n500.php\n3\n2021\n2013\n20110919_LN6yLNNET_bkp.zip\n0.js\n.user.ini\n.tmb\n.styleci.yml\n.smileys\n.qidb\n.buildpath\nzohoverify\nzohoForm.php\nyii.bat\nxmas\nxgame\nwptest4.xml\nwp-scrap\nwp-json\nwp-config-bak.php\nwp-config-backup.php\nwomen\nwidget.php\nwhite_paper_consumers_driving_the_digital_uptake.pdf\nwebpack.prod.js\nwebpack.dev.js\nwebpack.common.js\nwebmdportal61\nwebmd_health_check_do_not_delete.gif\nwebmd_aka_test\nwebmd-interviews-obama.ics\nwebmd-app-updates\nweb.config.txt\nwbmd\nvr\nvpn\nvidplayer\nvideoimages\nvideo_itunes\nvideo_for_iPhone\nvideo_config\nvgoalhi\nvet\nver.php\nv2\nutils\nutilities\nusr\nusfs\nuseruploads\nuploads.ini\nupgrade\nupdate.sh\nupc\nunsubscribe\nultimate\ntt_init.php\ntt_ads.php\ntt.txt\ntrust\ntrunk\ntranslate.php\ntransactions\ntrackad.gif\ntoyota\ntopics\ntool\ntms\ntimeline\nthemes_admin\nthanks.php\ntestpage.php\ntestmail.php\ntest2.xml\ntest2\ntest1.php\ntest.htm\nterms-of-service\ntemplate.php\ntemplate.N4BPage.php\ntemplate.MobilePage.php\ntemplate.Content_SearchPage.php\ntemplate.Content_HomePage.php\ntemplate.Content_FullPage.php\ntemplate.Content_BasicPage.php\ntemplate.BasicPage.php\ntema\nt3-assets\nsys\nsustainability\nsuspended.page\nsupport.php\nsuccess.php\nsubscribe.php\nsubs_files\nstylesheet\nstructure\nstrategicplan\nstatistics\nstandard.php\nstage\nsspv.xml\nsport\nsparkle\nsocs\nsocialmedia\nsoccer\nsnippets\nslideshow_fp\nslides\nsliders\nsitemap_files.xml\nsitemap2.xml\nsite_specific\nsignatures\nsidebar.php\nshowthread.php\nshizhan\nsetup32.exe\nsettings\nservicios\nserviceWorker.php\nsensor\nselect2\nsecurity\nsearchresults.htm\nsdc\nscuk\nscript.SiteLoadLive.php\nscreenshots\nscreens\nscholars\nsbm\nrumors\nrs-plugin\nroot\nrole_banner.mp4\nrobots.txt.20120205\nrobots.txt.20090320\nrn\nrewards\nreviews.php\nretail\nreseller-hosting.php\nrequests.php\nrepository\nreg\nreferral\nredirect_random_video.php\nredirect_random_album.php\nredirect_cs.php\nreadme\nrd.php\nrandom_image.php\nr.php\nqa\npromotions\nproduct_images\nprod\nprocesssiterequest.php\nprocess\nprivacy.htm\npresident\npregnancy-app-updates\nposts\npostform.php\nportraits\npolicy.php\npolice\npma\nplesk-stat\nplayoffs2009\nplayer.php\nplay.php\nplatform\nplaceholders\npixel\npingconnection.php\nping.php\npimg\npi\nphp_errorlog\npg\npeak\npb\npaypalprocess.php\npayments.php\npayments\npayment.php\npay.php\npassport\nparty\npain-app-updates\npackage.json.sample\npac\npa\np\nowl-carousel\norgs\noptoutcollector.php\nold_robots.txt\nofflineVersion.php\noffline.php\nnor\nnews.htm\nnew_photo\nnew_events\nnewThreeNav\nnbaguess\nmysql.php\nmysql\nmy_test\nmuse_manifest.xml\nmsgimg.php\nmsg1.php\nmsg.php\nmsg-xilver.php\nmsds\nmsVisionTest.php\nmovie\nmonitors\nmods\nmodcp\nmobility\nmobile-app-updates\nmm_menu.js\nmigrations\nmigration\nmemcached\nmelbet\nmedscape_holiday\nmedscape\nmedpulse\nmedicinenet\nmedias\nmaster\nmarketing.txt\nmaintenance_pages\nmac\nlv\nlu\nlost+found\nlogout\nllv\nlinks.htm\nlightbox\nlife\nlicensing\nliberty\nlearnmore\nlbg\nlaw\nlandingpage\nlander\nkr\nkog\nkg\nkatalog\njscripts\njp\njoggers\njadu\ninvoice\nintranet\ninternalaudit\ninsights\ninit.bat\ninfiniti\nindex_inc.php\nindex_.php\nindex.jsp\nindex.html.old\nin\nimg2021\nimages2010\nimagecache\nillustrations\nics\nice\ni18n\ni.php\nhybrid\nhuputv\nhupubridgedoc\nhtdocs\nhou\nhlc\nhistory.php\nhi\nhealthcheck.php\nhealthcheck\nhd\nhao123\ngroups\ngrid\ngraduation\ngrad\ngoldenkey\ngoddess\ngoalhi\ngo\nget_image.php\nget_file.php\ngeneric\ngenerate_sitemap\ngenerate_robots.cfm\ngeneral\ngamespace\ng\nfrom.php\nframe\nfotos\nformUploads\nford\nfood\nflyers\nflipbook\nfileLibrary\nfileAccess.php\nfile.php\nfeed.rss\nfeatures2010\nfbcanvas\nfairtrade\nf1\neyewonder\nextra\nextensions\nexports\nexportcookies.php\nexpmatch\nexec\nexample\nevergreen\nestaticos\nenvironments\nenv\nentrepreneurship\nengine1\nenews\nems\nemployment\nemail.php\nelection_widget\nee\nedu\neditorial\neditor\nedit_pc\nedaa\nec-landing\nec\nebook\ndyprize\ndu\ndtmcms\ndraft2013\ndraft2011\ndraft2010\ndownloader\ndocumentos\ndocument\ndo_not_delete\ndisplay.ShoppingCartQuickDisplay.php\ndiscover\ndisclaimer\ndirections\ndh\ndevtest\ndeploy.sh\ndepartments.php\ndepartments\nden\ndedicated-servers.php\ndecom\ndatastore\ndatabases\ndata_sample\ncy\ncustomgroupicons\ncu\ncrossdomain.xml.20100125.pl\ncrossdomain.xml.20100125-2.pl\ncrossdomain.xml.20090917\ncrossdomain.xml.20090820.pl\ncron_jobs\ncredit\ncpstyles\ncourse\ncount\ncorporate_nl\ncookiewarningtest\ncookieselector.php\ncookielistdata.php\ncontactus\nconsult\nconstants.php\nconnection\nconfirmation.php\nconfig.js\ncomscore-pv.json\ncomputing\ncompany_test\ncommon-js\ncolombia\ncollections\ncmsAPI\ncloud-hosting.php\nclear.gif\nclass.smtp.php\ncheck18\nchatbot-client\nchangecookie.php\nch-rm\nch-it\nch-fr\nch-de\ncgi\ncg\ncfsearch.cfm\nces\ncerts\ncelebrate-modern-life\nce\ncatalogue\ncase-studies\ncasa\ncart\ncareer.php\ncampaigns\ncallback.php\ncall\ncacert.pem\nc2dm\nbudget\nbrochure\nbridge\nbob.php\nboard\nbo\nbmw\nblueberry\nblank.gif\nbjs\nbif\nbi_consumer.js.mjw.20150421.textClipping\nbi_consumer.js\nbi_common_20091022\nbi_common_20090915\nbi_common_20090911\nbi_common_20090813\nbi_common.2015-04-21.sc\nbi_common.20140527\nbi_common.20100125.pl\nbi_common\nbermuda\nbe-nl\nbe-fr\nbc\nbayern\nbat\nbackoffice\nbaby-app-updates\nb1\nb\nautoload.php\naudit\nattachment.php\nat\nasia2009\nartsci\narchives\narchive.php\napns\napiprocess.php\nanswers3-api-test-data.xml\nannouncements\nanalyticstracking.php\nanalytics.txt\namp.php\namazon.php\namazon-associates-link-builder\nallergy-app-updates\nai-cache\nadvertiser\nads_view.php\nads_realtime_view.php\nads_realtime.php\nads_fifa.php\nadminlte\nadmin_files\nadclick.php\nad_view.php\nad_preview.php\nad_antd_pro\nactions\naccommodation\nacceptable-usage-policy.php\nacademics\nab\na450d92cb6be01b3b3669c18bfca7901\n_test\n_skins\n_schedule\n_main\n_images\n_global\n_functions\n_fonts\n_error\n_core\n_cfc\nYOCPlugin-installers.zip\nVideo\nTest.pdf\nTest-ICS-Event-nostart.ics\nTest\nService\nReadme.txt\nQBserver.php\nQBhelp.php\nPayPal\nMusic\nGruntfile.js.sample\nGruntfile.js\nGemfile.lock\nGemfile\nDB\nCopy of bi_common\nCSS\nCOOP\nCNAME\nArticles\nArchive.zip\nApp_Data\nApp\nAPI\n403.php\n4\n3rdpartylicenses.txt\n2025\n2022\n2012\n12\n11\n1.php\n1.js\n0507\n001.php\n000.php\n.top.menu.php\n.section.php\n.quarantine\n.localized\n.hgignore\n.hg\n.gitlab-ci.yml\n.access.php\n"
  },
  {
    "path": "setup.bat",
    "content": ":: Copyright (c) 2024-2025 Vili and contributors.\n@echo off\ncls\necho.\necho H4X-Tools Setup Script\necho.\necho -- Made in Finland, by Vili.\necho.\n\n:: Install dependencies\necho Installing dependencies...\npython -m pip install -r requirements.txt\nif %errorlevel% neq 0 (\n    echo Error: Failed to install dependencies.\n    pause\n    exit /b 1\n)\necho Done.\n\n:: Build executable\necho Building H4X-Tools to a single executable...\npython -m PyInstaller h4xtools.py --add-data \"resources\\*;resources\" --onefile -F --clean\nif %errorlevel% neq 0 (\n    echo Error: Failed to build executable.\n    pause\n    exit /b 1\n)\necho Done.\n\n:: Open dist folder\necho Your H4X-Tools executable is located in the 'dist' folder. You can now move it to your desired location.\necho OR you can start H4X-Tools with python by typing 'python h4xtools.py' in the terminal.\necho.\nset /p \"input=Open 'dist' folder now? (y/N) -> \"\nif /i \"%input%\"==\"y\" (\n    echo Opening up 'dist'.\n    start dist\\.\n) else (\n    echo Exiting...\n)\npause\nexit /b 0\n"
  },
  {
    "path": "setup.sh",
    "content": "#!/usr/bin/env bash\n\n# Copyright (c) 2024-2025 Vili and contributors.\n\nclear\n\necho \"H4X-Tools Setup Script\"\necho\necho \"-- Made in Finland, by Vili.\"\necho\necho \"You may need to install 'python-devel' packages.\"\n\n# Create and activate virtual environment\necho \"Creating virtual environment...\"\npython3 -m venv .venv\nsource .venv/bin/activate\n\n# Install dependencies\necho \"Installing dependencies...\"\nif command -v pip3 >/dev/null 2>&1; then\n    pip3 install --upgrade pip\n    pip3 install -r requirements.txt\nelse\n    echo \"python3-pip not installed, failed to install dependencies.\"\n    exit 1\nfi\n\n# Build H4X-Tools to a single executable\necho \"Building H4X-Tools to a single executable...\"\nif command -v pyinstaller >/dev/null 2>&1; then\n    pyinstaller h4xtools.py --add-data \"resources/*:resources\" --onefile -F --clean\n    chmod +x dist/h4xtools\n    if [ -d \"$HOME/.local/bin\" ]; then\n        mv dist/h4xtools \"$HOME/.local/bin\"\n    else\n        mkdir \"$HOME/.local/bin\"\n        mv dist/h4xtools \"$HOME/.local/bin\"\n    fi\n    rm h4xtools.spec\n    rm -r build\n    rm -r dist\n    echo \"Done! Type h4xtools in your terminal to start!\"\n    read -r -p \"Do you want to start H4XTools now? (y/N) \" answer\n    if [[ \"$answer\" =~ ^[yY]$ ]]; then\n        h4xtools\n    fi\nelse\n    echo \"pyinstaller not installed or not in PATH!\"\n    exit 1\nfi\n"
  },
  {
    "path": "update.bat",
    "content": ":: Copyright (c) 2024-2025 Vili and contributors.\n@echo off\ncls\necho.\necho H4X-Tools Update Script\necho.\necho -- Made in Finland, by Vili.\necho.\necho Make sure you run this in the root project directory!\necho.\n\nset /p \"input=Do you want to continue and update H4X-Tools? (y/N) \"\n\nif /i \"%input%\"==\"y\" (\n    echo Updating...\n    git fetch\n    git pull\n    echo Updating complete. Rebuilding setup...\n    start setup.bat\n) else (\n    echo Exiting...\n)\nexit\n"
  },
  {
    "path": "update.sh",
    "content": "#!/usr/bin/env bash\n\n# Copyright (c) 2024-2025 Vili and contributors.\n\nclear\n\necho \"H4X-Tools Update Script\"\necho\necho \"-- Made in Finland, by Vili.\"\necho\necho \"Make sure to run this in the root project directory!\"\n\nread -r -p \"Do you want to update H4XTools? (y/N) \" answer\nif [[ $answer == \"y\" ||  $answer == \"Y\" || $answer == \"yes\" || $answer == \"Yes\" ]]; then\n    git fetch\n    git pull\n    echo \"Update complete. Run setup.sh to apply changes.\"\n    read -r -p \"Do it now? (y/N) \" answer\n    if [[ \"$answer\" =~ ^[yY]$ ]]; then\n        bash setup.sh\n    fi\nelse\n    echo \"Update cancelled.\"\nfi\n"
  },
  {
    "path": "utils/__init__.py",
    "content": ""
  },
  {
    "path": "utils/bluetooth_scanner.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport os\nimport subprocess\nimport time\n\nfrom colorama import Style\n\nfrom helper import printer, timer\n\n\n@timer.timer(require_input=True)\ndef scan_nearby_bluetooth(duration: int) -> None:\n    \"\"\"\n    Performs a basic scan for nearby Bluetooth devices.\n\n    - Requires 'bluetoothctl' on Linux\n\n    :param duration: amount of seconds to scan for\n    \"\"\"\n    match os.name:\n        case \"nt\":\n            # TODO\n            printer.warning(\"Sorry, this feature isn't ready yet..!\")\n        case \"posix\":\n            _parse_output(output=_scan_linux(duration=duration), platform=\"linux\")\n        case _:\n            printer.error(\"Unsupported platform..!\")\n\n\ndef _scan_linux(duration: int) -> str:\n    \"\"\"\n    Performs a bluetoothctl scan on Linux systems.\n\n    :param duration: How many seconds to scan for.\n    \"\"\"\n    printer.info(\n        f\"Linux system detected... Performing {Style.BRIGHT}bluetoothctl{Style.RESET_ALL} scan...\"\n    )\n\n    output = \"\"\n\n    try:\n        process = subprocess.Popen(\n            [\"bluetoothctl\"],\n            stdin=subprocess.PIPE,\n            stdout=subprocess.PIPE,\n            stderr=subprocess.PIPE,\n            text=True,\n            bufsize=1,\n        )\n\n        if process.stdin is None:\n            printer.error(\"Failed to open bluetoothctl stdin.\")\n            return output\n\n        process.stdin.write(\"scan on\\n\")\n        process.stdin.flush()\n\n        time.sleep(duration)\n\n        process.stdin.write(\"scan off\\n\")\n        process.stdin.write(\"quit\\n\")\n        process.stdin.flush()\n\n        output, _ = process.communicate()\n\n    except subprocess.CalledProcessError as e:\n        printer.error(f\"Error : {e.returncode} - {e.stderr}\")\n        printer.error(\n            f\"Is your system using {Style.BRIGHT}bluetoothctl{Style.RESET_ALL}?\"\n        )\n\n    return output\n\n\ndef _scan_windows() -> None:\n    \"\"\"TODO\"\"\"\n    return\n\n\ndef _parse_output(output: str, platform: str) -> None:\n    match platform:\n        case \"windows\":\n            # Parse Windows output\n            devices = []\n            printer.error(\"Sorry, this feature isn't ready yet..!\")\n\n        case \"linux\":\n            # Parse Linux output\n            devices = []\n            clean_output = printer.ansi_escape(output)\n\n            for line in clean_output.splitlines():\n                printer.debug(line)\n                if \"[NEW] Device\" in line:\n                    parts = line.split()\n                    printer.debug(parts)\n                    # parts[0] = [NEW]\n                    # parts[1] = Device\n                    # parts[2] = MAC address\n                    # parts[3:] = device name (may contain spaces)\n                    if len(parts) < 3:\n                        continue\n                    mac = parts[2]\n                    name = \" \".join(parts[3:]).strip() if len(parts) > 3 else \"No Name\"\n\n                    devices.append({\"mac\": mac, \"name\": name, \"raw\": line})\n\n            printer.noprefix(\"\")\n            printer.section(\"Nearby Bluetooth Devices\")\n\n            if not devices:\n                printer.warning(\"No devices found.\")\n            else:\n                for device in devices:\n                    printer.success(f\"Name : {device['name']}\")\n                    printer.success(f\"MAC  : {device['mac']}\")\n                    printer.noprefix(\"\")\n        case _:\n            printer.error(\"idk how u got here.\")\n"
  },
  {
    "path": "utils/dirbuster.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport asyncio\n\nimport aiohttp\nfrom colorama import Style\n\nfrom helper import printer, randomuser, timer, url_helper\n\n\n@timer.timer(require_input=True)\ndef bust(domain: str) -> None:\n    \"\"\"\n    Scans the given url for valid paths.\n\n    :param domain: url to scan\n    \"\"\"\n    printer.info(\n        f\"Scanning for valid URLs for {Style.BRIGHT}{domain}{Style.RESET_ALL}...\"\n    )\n    printer.warning(\"This may take a while...\")\n\n    valid_urls = _scan_urls(domain)\n\n    printer.success(\n        f\"Scan Completed..! There were {Style.BRIGHT}{len(valid_urls)}{Style.RESET_ALL} valid URLs!\"\n    )\n\n\ndef _get_wordlist() -> set[str] | None:\n    \"\"\"\n    Reads the wordlist from the local resources and returns a set of paths.\n\n    :return: set of path strings, or None if an error occurs.\n    \"\"\"\n    content = url_helper.read_local_content(\"resources/wordlist.txt\")\n    if not isinstance(content, str):\n        return None\n    return {line.strip() for line in content.splitlines() if line.strip()}\n\n\nasync def _fetch_url(\n    session: aiohttp.ClientSession, domain: str, path: str, url_set: set\n) -> None:\n    \"\"\"\n    Fetches a URL and records it if it returns HTTP 200.\n\n    :param session: aiohttp session\n    :param domain: target domain\n    :param path: path to check\n    :param url_set: set to collect valid URLs into\n    \"\"\"\n    url = f\"https://{domain}/{path}\"\n    headers = {\"User-Agent\": str(randomuser.GetUser())}\n    try:\n        async with session.get(url, headers=headers) as response:\n            if response.status == 200:\n                url_set.add(url)\n                printer.success(\n                    f\"{len(url_set)} Valid URL(s) found : {Style.BRIGHT}{url}{Style.RESET_ALL}\"\n                )\n    except Exception:\n        pass\n\n\nasync def _scan_async(domain: str, paths: set[str]) -> set[str]:\n    \"\"\"\n    Scans the domain asynchronously for all paths.\n\n    :param domain: target domain\n    :param paths: set of paths to scan\n    :return: set of valid URLs found\n    \"\"\"\n    url_set: set[str] = set()\n    async with aiohttp.ClientSession() as session:\n        tasks = [_fetch_url(session, domain, path, url_set) for path in paths]\n        await asyncio.gather(*tasks, return_exceptions=True)\n    return url_set\n\n\ndef _scan_urls(domain: str) -> set[str]:\n    \"\"\"\n    Loads the wordlist and runs the async scan.\n\n    :param domain: target domain\n    :return: set of valid URLs found\n    \"\"\"\n    paths = _get_wordlist()\n    printer.debug(f\"Domain: {domain}, paths loaded: {len(paths) if paths else 0}\")\n    if paths is None:\n        printer.error(\"Failed to load wordlist..!\")\n        return set()\n\n    try:\n        return asyncio.run(_scan_async(domain, paths))\n    except KeyboardInterrupt:\n        printer.error(\"Cancelled..!\")\n        return set()\n    except RuntimeError as e:\n        printer.error(f\"Ran into a RuntimeError: {e}\")\n        return set()\n"
  },
  {
    "path": "utils/email_search.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport re\nimport subprocess\nfrom dataclasses import dataclass, field\n\nfrom colorama import Style\n\nfrom helper import printer, timer\n\n# Basic RFC-5321-ish pattern — catches obvious typos without being overly\n# strict.  Proper validation would require actually sending a test message.\n_EMAIL_RE = re.compile(r\"^[^@\\s]+@[^@\\s]+\\.[^@\\s]+$\")\n\n# Matches holehe's footer stat line, e.g. \"118 websites checked in 4.32 seconds\"\n_STATS_RE = re.compile(r\"^(\\d+)\\s+websites?\\s+checked\\s+in\\s+([\\d.]+)\\s+seconds?\", re.I)\n\n# Matches a result line like:\n#   [+] adobe.com recovery@other.com / +1234567890 / FullName John Doe\n#   [-] twitter.com\n#   [x] netflix.com\n_RESULT_RE = re.compile(r\"^\\[(\\+|-|x)\\]\\s+(\\S+)(.*)?$\")\n\n\n@dataclass\nclass HoleheSite:\n    \"\"\"Represents a single holehe result entry.\"\"\"\n\n    domain: str\n    found: bool\n    rate_limited: bool\n    # Extra data holehe may recover: recovery email, phone number, full name, etc.\n    extra: str = field(default=\"\")\n\n\n@timer.timer(require_input=True)\ndef search(email: str) -> None:\n    \"\"\"\n    Searches for registered accounts linked to an email address using holehe.\n\n    Validates the email format, runs holehe with clean output flags, then\n    displays a structured summary of every site where the address was found,\n    including any extra data holehe could recover (recovery email, phone\n    number, full name, account creation date).\n\n    Thanks to Holehe — https://github.com/megadose/holehe\n\n    :param email: The email address to search for.\n    \"\"\"\n    email = email.strip().lower()\n\n    if not _validate_email(email):\n        return\n\n    printer.info(\n        f\"Searching for accounts linked to {Style.BRIGHT}{email}{Style.RESET_ALL}...\"\n    )\n\n    _run_holehe(email)\n\n\n# Internal helpers\n\n\ndef _validate_email(email: str) -> bool:\n    \"\"\"\n    Performs basic email format validation.\n\n    :param email: The email address to check.\n    :return: ``True`` if the format looks valid, ``False`` otherwise.\n    \"\"\"\n    if not _EMAIL_RE.match(email):\n        printer.error(f\"'{email}' does not look like a valid email address.\")\n        printer.info(\"Expected format: user@example.com\")\n        return False\n    return True\n\n\ndef _run_holehe(email: str) -> None:\n    \"\"\"\n    Invokes the holehe CLI and prints the parsed results.\n\n    Flags used:\n\n    * ``--only-used``  — suppress not-found (``[-]``) lines; show only hits\n    * ``--no-color``   — plain-text output for reliable parsing\n    * ``--no-clear``   — do not wipe the terminal before printing results\n\n    :param email: The validated email address.\n    \"\"\"\n    try:\n        result = subprocess.run(\n            [\n                \"holehe\",\n                email,\n                \"--only-used\",\n                \"--no-color\",\n                \"--no-clear\",\n            ],\n            capture_output=True,\n            text=True,\n            timeout=120,\n        )\n\n        combined = result.stdout + result.stderr\n\n        if not combined.strip():\n            printer.warning(\"holehe returned no output.\")\n            return\n\n        sites, stats = _parse_holehe_output(combined)\n        _print_results(sites, stats)\n        printer.info(\"Credits to megadose (Palenath) for holehe.\")\n\n    except FileNotFoundError:\n        printer.error(\n            f\"{Style.BRIGHT}holehe{Style.RESET_ALL} was not found. \"\n            f\"Install it with: {Style.BRIGHT}pip install holehe{Style.RESET_ALL}\"\n        )\n    except subprocess.TimeoutExpired:\n        printer.error(\n            \"holehe timed out after 120 seconds. \"\n            \"Try again or check your internet connection.\"\n        )\n    except subprocess.CalledProcessError as exc:\n        printer.error(f\"holehe exited with a non-zero status: {exc}\")\n    except Exception as exc:\n        printer.error(f\"Unexpected error while running holehe: {exc}\")\n\n\ndef _parse_holehe_output(output: str) -> tuple[list[HoleheSite], str]:\n    \"\"\"\n    Parses holehe's plain-text output into structured :class:`HoleheSite`\n    objects and an optional stats string.\n\n    Holehe result-line conventions (with ``--only-used``, ``[-]`` lines are\n    suppressed by holehe itself, but we guard against them regardless):\n\n    * ``[+] domain.com [extra…]`` — email **is** registered; may carry\n      a recovery email, phone number, full name, or creation date inline.\n    * ``[-] domain.com``          — email is *not* registered (filtered out).\n    * ``[x] domain.com``          — rate limited; result unknown.\n\n    Footer stat line example::\n\n        118 websites checked in 4.32 seconds\n\n    :param output: Raw captured stdout from holehe.\n    :return: Tuple of ``(sites, stats_line)`` where *sites* is sorted by\n             domain and *stats_line* is the footer string (empty if absent).\n    \"\"\"\n    # Strip any residual ANSI escape codes that slip through despite --no-color.\n    ansi_re = re.compile(r\"\\x1B(?:[@-Z\\\\-_]|\\[[0-?]*[ -/]*[@-~])\")\n\n    sites: list[HoleheSite] = []\n    stats = \"\"\n\n    for line in output.splitlines():\n        clean = ansi_re.sub(\"\", line).strip()\n        if not clean:\n            continue\n\n        # Footer stats line\n        if _STATS_RE.match(clean):\n            stats = clean\n            continue\n\n        match = _RESULT_RE.match(clean)\n        if not match:\n            continue\n\n        symbol, domain, extra_raw = match.group(1), match.group(2), match.group(3) or \"\"\n\n        # Holehe prints a legend line at the bottom:\n        #   [+] Email used, [-] Email not used, [x] Rate limit\n        # _RESULT_RE matches it with domain=\"Email\" (no dot).  Real domains\n        # always contain at least one dot, so we use that to skip it.\n        if \".\" not in domain:\n            continue\n\n        extra = extra_raw.strip().lstrip(\"/\").strip()\n\n        if symbol == \"+\":\n            sites.append(\n                HoleheSite(domain=domain, found=True, rate_limited=False, extra=extra)\n            )\n        elif symbol == \"x\":\n            sites.append(\n                HoleheSite(domain=domain, found=False, rate_limited=True, extra=extra)\n            )\n        # \"-\" (not found) is intentionally ignored — holehe suppresses these\n        # with --only-used, but skip them explicitly for robustness.\n\n    sites.sort(key=lambda s: s.domain)\n    return sites, stats\n\n\ndef _print_results(sites: list[HoleheSite], stats: str) -> None:\n    \"\"\"\n    Displays the parsed holehe results with counts and extra recovered data.\n\n    :param sites: Parsed list of :class:`HoleheSite` objects.\n    :param stats: Footer stats string from holehe (may be empty).\n    \"\"\"\n    found = [s for s in sites if s.found]\n    rate_limited = [s for s in sites if s.rate_limited]\n\n    printer.noprefix(\"\")\n    printer.section(\"Holehe Results\")\n\n    if found:\n        printer.info(\n            f\"Registered on {Style.BRIGHT}{len(found)}{Style.RESET_ALL} site(s):\"\n        )\n        for site in found:\n            # Build the display line; append any extra data holehe recovered.\n            line = f\"  {Style.BRIGHT}{site.domain}{Style.RESET_ALL}\"\n            if site.extra:\n                line += f\"  ({_format_extra(site.extra)})\"\n            printer.success(line)\n    else:\n        printer.warning(\"Not found on any sites.\")\n\n    if rate_limited:\n        printer.noprefix(\"\")\n        printer.warning(\n            f\"Rate limited on {Style.BRIGHT}{len(rate_limited)}{Style.RESET_ALL} \"\n            \"site(s) — could not determine registration status:\"\n        )\n        for site in rate_limited:\n            printer.warning(f\"  [x] {site.domain}\")\n\n    if stats:\n        printer.noprefix(\"\")\n        printer.info(stats)\n\n\ndef _format_extra(raw: str) -> str:\n    \"\"\"\n    Converts holehe's inline extra string into a human-readable label.\n\n    Holehe concatenates extra fields with `` / `` separators, e.g.::\n\n        recovery@other.com / +1234567890 / FullName John Doe\n\n    This function splits on those separators and returns a tidy comma-joined\n    string with labelled fields where possible.\n\n    :param raw: The raw extra string from the holehe output line.\n    :return: Formatted string, e.g.\n             ``\"recovery: recovery@other.com, phone: +1234567890, name: John Doe\"``.\n    \"\"\"\n    # Holehe joins fields with \" / \" (space-slash-space).  Splitting on bare\n    # \"/\" would break URLs like \"https://gravatar.com/username\", so we use\n    # the full \" / \" separator instead.\n    parts = [p.strip() for p in raw.split(\" / \") if p.strip()]\n    labelled: list[str] = []\n\n    for part in parts:\n        if part.startswith(\"FullName \"):\n            labelled.append(f\"name: {part[len('FullName ') :]}\")\n        elif part.startswith(\"Date, time of the creation \"):\n            labelled.append(f\"created: {part[len('Date, time of the creation ') :]}\")\n        elif re.match(r\"^\\+?\\d[\\d\\s\\-().]{6,}$\", part):\n            labelled.append(f\"phone: {part}\")\n        elif \"@\" in part:\n            labelled.append(f\"recovery: {part}\")\n        else:\n            labelled.append(part)\n\n    return \", \".join(labelled)\n"
  },
  {
    "path": "utils/fake_info_generator.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport time\n\nfrom faker import Faker\n\nfrom helper import printer, timer\n\n\n@timer.timer(require_input=True)\ndef generate() -> None:\n    \"\"\"\n    Generates a fake identity with contact, payment, and location details.\n\n    Thanks to Faker, https://pypi.org/project/Faker/\n    \"\"\"\n    fake = Faker()\n    printer.info(\"Generating fake information...\")\n    time.sleep(1)\n\n    printer.noprefix(\"\")\n    printer.section(\"Identity\")\n    printer.success(f\"Name             : {fake.name()}\")\n    printer.success(f\"Job              : {fake.job()}\")\n    printer.success(f\"Company          : {fake.company()}\")\n\n    printer.noprefix(\"\")\n    printer.section(\"Contact\")\n    printer.success(f\"Email            : {fake.email()}\")\n    printer.success(f\"Phone Number     : {fake.phone_number()}\")\n    printer.success(f\"Address          : {fake.address()}\")\n\n    printer.noprefix(\"\")\n    printer.section(\"Payment\")\n    printer.success(f\"Card Number      : {fake.credit_card_number()}\")\n    printer.success(f\"Card Type        : {fake.credit_card_provider()}\")\n    printer.success(f\"Expiry Date      : {fake.credit_card_expire()}\")\n    printer.success(f\"Security Code    : {fake.credit_card_security_code()}\")\n    printer.success(f\"IBAN             : {fake.iban()}\")\n    printer.success(f\"BBAN             : {fake.bban()}\")\n\n    printer.noprefix(\"\")\n    printer.section(\"Location\")\n    printer.success(f\"Country          : {fake.country()}\")\n    printer.success(f\"City             : {fake.city()}\")\n"
  },
  {
    "path": "utils/ig_scrape.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport csv\nimport json\nfrom dataclasses import asdict, dataclass, field\nfrom datetime import datetime\nfrom pathlib import Path\n\nimport phonenumbers\nfrom colorama import Style\nfrom ensta import Guest\nfrom ensta.lib.Exceptions import APIError, NetworkError, RateLimitedError\nfrom toutatis.core import advanced_lookup, getInfo\n\nfrom helper import printer, timer\n\n_KEY_WIDTH = 24\n_SAVE_DIR = Path(\"scraped_data\")\n\n\n@dataclass\nclass Post:\n    \"\"\"A single Instagram post.\"\"\"\n\n    index: int = 0\n    url: str = \"\"\n    caption: str = \"\"\n\n\n@dataclass\nclass IGProfile:\n    \"\"\"Aggregated Instagram profile data from all available sources.\"\"\"\n\n    # Basic identity\n    username: str = \"\"\n    user_id: str = \"\"\n    full_name: str = \"\"\n    biography: str = \"\"\n    website: str = \"\"\n    profile_pic_url: str = \"\"\n\n    # Counts\n    followers: int | str = \"N/A\"\n    following: int | str = \"N/A\"\n    posts: int | str = \"N/A\"\n    igtv_videos: int | str = \"N/A\"\n\n    # Status flags\n    is_private: bool | str = \"N/A\"\n    is_verified: bool | str = \"N/A\"\n    is_business: bool | str = \"N/A\"\n    is_whatsapp_linked: bool | str = \"N/A\"\n    is_memorialized: bool | str = \"N/A\"\n    is_new_to_instagram: bool | str = \"N/A\"\n\n    # Contact — populated by toutatis when available\n    public_email: str = \"\"\n    public_phone: str = \"\"\n    obfuscated_email: str = \"\"\n    obfuscated_phone: str = \"\"\n\n    # Recent posts — populated by ensta guest track\n    recent_posts: list[Post] = field(default_factory=list)\n\n    # Meta\n    session_used: bool = False\n    scraped_at: str = field(default_factory=lambda: datetime.now().isoformat())\n\n\ndef _fetch_guest(username: str, profile: IGProfile) -> None:\n    \"\"\"\n    Populate *profile* using the ensta Guest API (no authentication needed).\n\n    Fills basic profile fields and recent posts.\n\n    :param username: Instagram username to look up.\n    :param profile:  :class:`IGProfile` to populate in-place.\n    \"\"\"\n    try:\n        api = Guest()\n        data = api.profile(username)\n        if data is None or not hasattr(data, \"raw\") or data.raw is None:\n            printer.error(\"Ensta returned no data for this account.\")\n            return\n        raw: dict = data.raw\n\n        profile.username = raw.get(\"username\", username)\n        profile.user_id = str(raw.get(\"id\", \"\"))\n        profile.full_name = raw.get(\"full_name\", \"\")\n        profile.biography = raw.get(\"biography\", \"\")\n        profile.website = raw.get(\"external_url\", \"\")\n        profile.profile_pic_url = raw.get(\"profile_pic_url_hd\", \"\")\n        profile.followers = raw.get(\"edge_followed_by\", {}).get(\"count\", \"N/A\")\n        profile.following = raw.get(\"edge_follow\", {}).get(\"count\", \"N/A\")\n        profile.posts = raw.get(\"edge_owner_to_timeline_media\", {}).get(\"count\", \"N/A\")\n        profile.is_private = raw.get(\"is_private\", \"N/A\")\n        profile.is_verified = raw.get(\"is_verified\", \"N/A\")\n\n        # Recent posts\n        edges = raw.get(\"edge_owner_to_timeline_media\", {}).get(\"edges\", [])\n        for idx, edge in enumerate(edges, start=1):\n            node = edge.get(\"node\", {})\n            shortcode = node.get(\"shortcode\", \"\")\n            caption_edges = node.get(\"edge_media_to_caption\", {}).get(\"edges\", [])\n            caption = caption_edges[0][\"node\"][\"text\"] if caption_edges else \"\"\n            profile.recent_posts.append(\n                Post(\n                    index=idx,\n                    url=(\n                        f\"https://www.instagram.com/p/{shortcode}/\" if shortcode else \"\"\n                    ),\n                    caption=caption,\n                )\n            )\n\n    except NetworkError as exc:\n        printer.error(f\"Network error during guest fetch: {exc}\")\n    except RateLimitedError:\n        printer.error(\"Ensta rate limited. Try again later or provide a session ID.\")\n    except APIError as exc:\n        printer.error(f\"Ensta API error: {exc}\")\n    except AttributeError:\n        printer.error(\n            \"Ensta returned no data. The account may not exist, be private, \"\n            \"or Instagram may be blocking requests. Try providing a session ID.\"\n        )\n\n\ndef _fetch_authenticated(username: str, session_id: str, profile: IGProfile) -> None:\n    \"\"\"\n    Populate *profile* using the Toutatis authenticated API.\n\n    Fills all basic fields plus business flag, WhatsApp/memorial/new-user\n    flags, IGTV count, and public contact details (email / phone).\n\n    :param username:   Instagram username.\n    :param session_id: Instagram ``sessionid`` cookie value.\n    :param profile:    :class:`IGProfile` to populate in-place.\n    \"\"\"\n    result = getInfo(username, session_id)\n\n    if result.get(\"error\"):\n        printer.error(f\"Toutatis getInfo error: {result['error']}\")\n        printer.warning(\"Falling back to guest track for basic profile data.\")\n        _fetch_guest(username, profile)\n        return\n\n    info: dict = result.get(\"user\", {})\n    profile.session_used = True\n\n    profile.username = info.get(\"username\", username)\n    profile.user_id = str(info.get(\"userID\", info.get(\"pk\", \"\")))\n    profile.full_name = info.get(\"full_name\", \"\")\n    profile.biography = info.get(\"biography\", \"\")\n    profile.website = info.get(\"external_url\", \"\")\n\n    pic_info = info.get(\"hd_profile_pic_url_info\") or {}\n    profile.profile_pic_url = pic_info.get(\"url\", info.get(\"profile_pic_url\", \"\"))\n\n    profile.followers = info.get(\"follower_count\", \"N/A\")\n    profile.following = info.get(\"following_count\", \"N/A\")\n    profile.posts = info.get(\"media_count\", \"N/A\")\n    profile.igtv_videos = info.get(\"total_igtv_videos\", \"N/A\")\n\n    profile.is_private = info.get(\"is_private\", \"N/A\")\n    profile.is_verified = info.get(\"is_verified\", \"N/A\")\n    profile.is_business = info.get(\"is_business\", \"N/A\")\n    profile.is_whatsapp_linked = info.get(\"is_whatsapp_linked\", \"N/A\")\n    profile.is_memorialized = info.get(\"is_memorialized\", \"N/A\")\n    profile.is_new_to_instagram = info.get(\"is_new_to_instagram\", \"N/A\")\n\n    # Public email\n    if info.get(\"public_email\"):\n        profile.public_email = info[\"public_email\"]\n\n    # Public phone — format with phonenumbers library if possible\n    raw_phone = str(info.get(\"public_phone_number\", \"\")).strip()\n    country_code = str(info.get(\"public_phone_country_code\", \"\")).strip()\n    if raw_phone and raw_phone not in {\"0\", \"\"}:\n        full = f\"+{country_code} {raw_phone}\" if country_code else raw_phone\n        try:\n            pn = phonenumbers.parse(full)\n            profile.public_phone = phonenumbers.format_number(\n                pn, phonenumbers.PhoneNumberFormat.INTERNATIONAL\n            )\n        except phonenumbers.NumberParseException:\n            profile.public_phone = full\n\n\ndef _fetch_advanced_lookup(username: str, profile: IGProfile) -> None:\n    \"\"\"\n    Run the Toutatis ``advanced_lookup`` to surface obfuscated contact info.\n\n    This endpoint uses Instagram's account-recovery flow and requires no\n    session cookie, but may be rate-limited.\n\n    :param username: Instagram username.\n    :param profile:  :class:`IGProfile` to populate in-place.\n    \"\"\"\n    printer.info(\"Running Toutatis advanced lookup (obfuscated contact)...\")\n    result = advanced_lookup(username)\n\n    if result.get(\"error\") == \"rate limit\":\n        printer.warning(\n            \"Advanced lookup rate limited. Obfuscated contact info unavailable.\"\n        )\n        return\n\n    user_data: dict = result.get(\"user\") or {}\n\n    if user_data.get(\"message\") == \"No users found\":\n        printer.warning(\"Advanced lookup: no users found for this username.\")\n        return\n\n    msg = user_data.get(\"message\")\n    if msg:\n        printer.debug(f\"Advanced lookup message: {msg}\")\n\n    obf_email = user_data.get(\"obfuscated_email\", \"\")\n    obf_phone = str(user_data.get(\"obfuscated_phone\", \"\")).strip()\n\n    if obf_email:\n        profile.obfuscated_email = obf_email\n    if obf_phone and obf_phone not in {\"None\", \"0\", \"\"}:\n        profile.obfuscated_phone = obf_phone\n\n\ndef _print_profile(profile: IGProfile) -> None:\n    \"\"\"Render the full :class:`IGProfile` to the terminal.\"\"\"\n    printer.noprefix(\"\")\n    printer.section(\"Instagram Profile\")\n\n    rows = [\n        (\"Username\", profile.username),\n        (\"User ID\", profile.user_id),\n        (\"Full Name\", profile.full_name),\n        (\"Biography\", profile.biography),\n        (\"Website\", profile.website),\n        (\"Followers\", profile.followers),\n        (\"Following\", profile.following),\n        (\"Total Posts\", profile.posts),\n        (\"IGTV Videos\", profile.igtv_videos),\n        (\"Private\", profile.is_private),\n        (\"Verified\", profile.is_verified),\n        (\"Business Account\", profile.is_business),\n        (\"WhatsApp Linked\", profile.is_whatsapp_linked),\n        (\"Memorial Account\", profile.is_memorialized),\n        (\"New User\", profile.is_new_to_instagram),\n        (\"Profile Picture\", profile.profile_pic_url),\n    ]\n    for label, value in rows:\n        if value not in (\"\", \"N/A\", False, None):\n            printer.success(f\"{label:<{_KEY_WIDTH}} : {value}\")\n        elif value == \"N/A\":\n            printer.success(f\"{label:<{_KEY_WIDTH}} : N/A\")\n\n    # Contact section\n    has_contact = any(\n        [\n            profile.public_email,\n            profile.public_phone,\n            profile.obfuscated_email,\n            profile.obfuscated_phone,\n        ]\n    )\n    if has_contact:\n        printer.noprefix(\"\")\n        printer.section(\"Contact Information\")\n        if profile.public_email:\n            printer.success(f\"{'Public Email':<{_KEY_WIDTH}} : {profile.public_email}\")\n        if profile.public_phone:\n            printer.success(f\"{'Public Phone':<{_KEY_WIDTH}} : {profile.public_phone}\")\n        if profile.obfuscated_email:\n            printer.success(\n                f\"{'Obfuscated Email':<{_KEY_WIDTH}} : {profile.obfuscated_email}\"\n            )\n        if profile.obfuscated_phone:\n            printer.success(\n                f\"{'Obfuscated Phone':<{_KEY_WIDTH}} : {profile.obfuscated_phone}\"\n            )\n\n    # Recent posts (guest track only)\n    if profile.recent_posts:\n        printer.noprefix(\"\")\n        printer.section(\"Recent Posts\")\n        for post in profile.recent_posts:\n            printer.success(f\"Post {post.index}\")\n            if post.url:\n                printer.success(f\"  {'URL':<{_KEY_WIDTH - 2}} : {post.url}\")\n            if post.caption:\n                cap = (\n                    post.caption\n                    if len(post.caption) <= 120\n                    else post.caption[:117] + \"...\"\n                )\n                printer.success(f\"  {'Caption':<{_KEY_WIDTH - 2}} : {cap}\")\n            printer.noprefix(\"\")\n    elif not profile.session_used:\n        printer.warning(\"No recent posts found (account may be private).\")\n\n    printer.noprefix(\"\")\n    source = (\n        \"Toutatis + ensta  (authenticated)\"\n        if profile.session_used\n        else \"ensta guest + Toutatis advanced lookup\"\n    )\n    printer.info(f\"{'Data source':<{_KEY_WIDTH}} : {source}\")\n    printer.info(f\"{'Scraped at':<{_KEY_WIDTH}} : {profile.scraped_at}\")\n\n\ndef _ask_export() -> str | None:\n    \"\"\"\n    Prompt the user for an optional export format.\n\n    :return: ``'txt'``, ``'csv'``, or ``'json'``; ``None`` if declined.\n    \"\"\"\n    answer = printer.user_input(\"Save results to file? (y/N) : \").strip().lower()\n    if answer not in {\"y\", \"yes\"}:\n        return None\n\n    printer.noprefix(\"\")\n    printer.section(\"Export Format\")\n    printer.info(\"  1 : TXT  (plain text report)\")\n    printer.info(\"  2 : CSV  (spreadsheet-friendly)\")\n    printer.info(\"  3 : JSON (full structured data)\")\n\n    fmt_map = {\"1\": \"txt\", \"2\": \"csv\", \"3\": \"json\", \"\": \"txt\"}\n    choice = printer.user_input(\"Choose format (1/2/3) [default: 1] : \").strip()\n    return fmt_map.get(choice, \"txt\")\n\n\ndef _export(profile: IGProfile, fmt: str) -> None:\n    \"\"\"\n    Write *profile* to ``scraped_data/`` in the requested format.\n\n    :param profile: The populated :class:`IGProfile` to export.\n    :param fmt:     ``'txt'``, ``'csv'``, or ``'json'``.\n    \"\"\"\n    _SAVE_DIR.mkdir(exist_ok=True)\n    timestamp = datetime.now().strftime(\"%Y%m%d_%H%M%S\")\n    slug = profile.username or \"unknown\"\n\n    try:\n        match fmt.lower():\n            case \"txt\":\n                filepath = _SAVE_DIR / f\"igscrape_{slug}_{timestamp}.txt\"\n                with filepath.open(\"w\", encoding=\"utf-8\") as fh:\n                    fh.write(\"Instagram Scrape Report\\n\")\n                    fh.write(f\"Target     : {profile.username}\\n\")\n                    fh.write(\n                        f\"Date       : {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}\\n\"\n                    )\n                    fh.write(\n                        f\"Auth       : {'Yes (session)' if profile.session_used else 'No (guest)'}\\n\"\n                    )\n                    fh.write(\"=\" * 80 + \"\\n\\n\")\n\n                    fh.write(\"PROFILE\\n\" + \"-\" * 40 + \"\\n\")\n                    fields = [\n                        (\"Username\", profile.username),\n                        (\"User ID\", profile.user_id),\n                        (\"Full Name\", profile.full_name),\n                        (\"Biography\", profile.biography),\n                        (\"Website\", profile.website),\n                        (\"Followers\", profile.followers),\n                        (\"Following\", profile.following),\n                        (\"Total Posts\", profile.posts),\n                        (\"IGTV Videos\", profile.igtv_videos),\n                        (\"Private\", profile.is_private),\n                        (\"Verified\", profile.is_verified),\n                        (\"Business\", profile.is_business),\n                        (\"WhatsApp Linked\", profile.is_whatsapp_linked),\n                        (\"Memorial\", profile.is_memorialized),\n                        (\"New User\", profile.is_new_to_instagram),\n                        (\"Profile Pic\", profile.profile_pic_url),\n                    ]\n                    for label, val in fields:\n                        fh.write(f\"  {label:<22}: {val}\\n\")\n\n                    fh.write(\"\\nCONTACT\\n\" + \"-\" * 40 + \"\\n\")\n                    fh.write(f\"  Public Email    : {profile.public_email or 'N/A'}\\n\")\n                    fh.write(f\"  Public Phone    : {profile.public_phone or 'N/A'}\\n\")\n                    fh.write(\n                        f\"  Obfusc. Email   : {profile.obfuscated_email or 'N/A'}\\n\"\n                    )\n                    fh.write(\n                        f\"  Obfusc. Phone   : {profile.obfuscated_phone or 'N/A'}\\n\"\n                    )\n\n                    if profile.recent_posts:\n                        fh.write(\"\\nRECENT POSTS\\n\" + \"-\" * 40 + \"\\n\")\n                        for post in profile.recent_posts:\n                            fh.write(f\"  [{post.index}] {post.url}\\n\")\n                            if post.caption:\n                                fh.write(f\"      {post.caption}\\n\")\n                            fh.write(\"\\n\")\n\n                printer.success(\n                    f\"Results saved → {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n\n            case \"csv\":\n                filepath = _SAVE_DIR / f\"igscrape_{slug}_{timestamp}.csv\"\n                with filepath.open(\"w\", newline=\"\", encoding=\"utf-8\") as fh:\n                    writer = csv.writer(fh)\n                    # Profile sheet\n                    writer.writerow([\"field\", \"value\"])\n                    for label, val in [\n                        (\"username\", profile.username),\n                        (\"user_id\", profile.user_id),\n                        (\"full_name\", profile.full_name),\n                        (\"biography\", profile.biography),\n                        (\"website\", profile.website),\n                        (\"followers\", profile.followers),\n                        (\"following\", profile.following),\n                        (\"posts\", profile.posts),\n                        (\"igtv_videos\", profile.igtv_videos),\n                        (\"is_private\", profile.is_private),\n                        (\"is_verified\", profile.is_verified),\n                        (\"is_business\", profile.is_business),\n                        (\"is_whatsapp_linked\", profile.is_whatsapp_linked),\n                        (\"is_memorialized\", profile.is_memorialized),\n                        (\"is_new_to_instagram\", profile.is_new_to_instagram),\n                        (\"profile_pic_url\", profile.profile_pic_url),\n                        (\"public_email\", profile.public_email),\n                        (\"public_phone\", profile.public_phone),\n                        (\"obfuscated_email\", profile.obfuscated_email),\n                        (\"obfuscated_phone\", profile.obfuscated_phone),\n                        (\"session_used\", profile.session_used),\n                        (\"scraped_at\", profile.scraped_at),\n                    ]:\n                        writer.writerow([label, val])\n\n                    if profile.recent_posts:\n                        writer.writerow([])\n                        writer.writerow([\"post_index\", \"post_url\", \"caption\"])\n                        for post in profile.recent_posts:\n                            writer.writerow([post.index, post.url, post.caption])\n\n                printer.success(\n                    f\"Results saved → {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n\n            case \"json\":\n                filepath = _SAVE_DIR / f\"igscrape_{slug}_{timestamp}.json\"\n                payload = asdict(profile)\n                # recent_posts are dataclasses too — asdict handles them\n                with filepath.open(\"w\", encoding=\"utf-8\") as fh:\n                    json.dump(payload, fh, indent=2, ensure_ascii=False)\n                printer.success(\n                    f\"Results saved → {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n\n            case _:\n                printer.error(f\"Unknown format '{fmt}'. Use 'txt', 'csv', or 'json'.\")\n\n    except OSError as exc:\n        printer.error(f\"Could not write file: {exc}\")\n\n\n@timer.timer(require_input=True)\ndef scrape(target: str) -> None:\n    \"\"\"\n    Scrapes and aggregates data from an Instagram account.\n\n    Two tracks are supported:\n\n    **Guest track** (no session ID)\n      Uses the ``ensta`` Guest API for public profile data and recent posts,\n      then runs the Toutatis ``advanced_lookup`` to surface any obfuscated\n      e-mail address or phone number tied to the account's recovery flow.\n\n    **Authenticated track** (session ID provided)\n      Uses Toutatis ``getInfo()`` via Instagram's private mobile API for a\n      richer profile including business flags, IGTV count, WhatsApp link\n      status, and publicly listed contact details, then also runs\n      ``advanced_lookup`` for the obfuscated recovery contacts.\n\n    The session ID is your Instagram ``sessionid`` cookie value.  To find it:\n    open Instagram in a browser → DevTools → Application → Cookies →\n    copy the value of the ``sessionid`` cookie.\n\n    Results can be exported to ``scraped_data/`` as TXT, CSV, or JSON.\n\n    :param target: The Instagram username to investigate.\n    \"\"\"\n    target = target.strip().lstrip(\"@\")\n    if not target:\n        printer.error(\"Username cannot be empty.\")\n        return\n\n    printer.noprefix(\"\")\n    printer.info(\n        \"Provide your Instagram session ID for richer data (phone, email, \"\n        \"business flags, etc.).\"\n    )\n    printer.info(\n        \"Find it in your browser: DevTools → Application → Cookies → sessionid\"\n    )\n    session_id = printer.user_input(\n        \"Session ID (leave blank for guest mode) : \"\n    ).strip()\n\n    profile = IGProfile()\n\n    if session_id:\n        printer.info(\n            f\"Authenticated scrape for {Style.BRIGHT}{target}{Style.RESET_ALL}...\"\n        )\n        _fetch_authenticated(target, session_id, profile)\n    else:\n        printer.info(\n            f\"Guest scrape for {Style.BRIGHT}{target}{Style.RESET_ALL} \"\n            \"(no session ID — limited data)...\"\n        )\n        _fetch_guest(target, profile)\n\n    # Advanced lookup runs regardless of authentication track.\n    _fetch_advanced_lookup(target, profile)\n\n    if not profile.username:\n        printer.error(\"No data could be retrieved for this account.\")\n        return\n\n    _print_profile(profile)\n\n    printer.noprefix(\"\")\n    fmt = _ask_export()\n    if fmt:\n        _export(profile, fmt)\n"
  },
  {
    "path": "utils/ip_lookup.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport socket\nimport time\n\nimport requests\nfrom colorama import Style\n\nfrom helper import printer, randomuser, timer\n\n# Human-readable labels for the keys returned by ipinfo.io.\n# Any key not listed here falls back to key.replace(\"_\", \" \").title().\n_IP_KEY_LABELS: dict[str, str] = {\n    \"ip\": \"IP Address\",\n    \"hostname\": \"Hostname\",\n    \"city\": \"City\",\n    \"region\": \"Region\",\n    \"country\": \"Country\",\n    \"loc\": \"Coordinates\",\n    \"org\": \"Organization\",\n    \"postal\": \"Postal Code\",\n    \"timezone\": \"Timezone\",\n}\n\n# Pad all labels to this width so values line up neatly.\n_KEY_WIDTH = max(len(v) for v in _IP_KEY_LABELS.values()) + 2  # = 14\n\n\n@timer.timer(require_input=True)\ndef lookup(ip_address: str) -> None:\n    \"\"\"\n    Gets information about a given IP address or hostname using ipinfo.io.\n\n    :param ip_address: The IP address or hostname to look up.\n    \"\"\"\n    try:\n        ip_address = socket.gethostbyname(ip_address)\n        url = f\"https://ipinfo.io/{ip_address}/json\"\n        headers = {\"User-Agent\": str(randomuser.GetUser())}\n        response = requests.get(url, headers=headers)\n        response.raise_for_status()\n        values = response.json()\n\n        printer.info(f\"Looking up {Style.BRIGHT}{ip_address}{Style.RESET_ALL}...\")\n        time.sleep(1)\n\n        printer.noprefix(\"\")\n        printer.section(\"IP Lookup Results\")\n\n        for key, value in values.items():\n            if key == \"readme\":\n                continue\n            label = _IP_KEY_LABELS.get(key, key.replace(\"_\", \" \").title())\n            display_value = value if value else \"N/A\"\n            printer.success(f\"{label:<{_KEY_WIDTH}} : {display_value}\")\n\n        if \"loc\" in values:\n            printer.noprefix(\"\")\n            printer.info(\n                \"OpenStreetMap : \"\n                f\"https://www.openstreetmap.org/search?query={values['loc']}\"\n            )\n\n    except requests.exceptions.RequestException as e:\n        printer.error(f\"Request error : {e}\")\n    except socket.gaierror as e:\n        printer.error(f\"Could not resolve host : {e}\")\n    except Exception as e:\n        printer.error(f\"Error : {e}\")\n"
  },
  {
    "path": "utils/leak_search.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport csv\nimport json\nimport time\nfrom dataclasses import asdict, dataclass, field\nfrom datetime import datetime\nfrom pathlib import Path\n\nimport requests\nfrom colorama import Style\n\nfrom helper import printer, randomuser, timer\n\n_REQUEST_TIMEOUT: int = 20\n_MAX_RETRIES: int = 3\n_RETRY_BASE_DELAY: float = 3.0\n\n_SAVE_DIR = Path(\"scraped_data\")\n\n# How many stealer/credential entries to display inline (user-selectable).\n_ENTRY_LIMIT_CHOICES: dict[str, int | None] = {\n    \"1\": 3,\n    \"2\": 5,\n    \"3\": 10,\n    \"4\": None,  # all\n}\n_DEFAULT_ENTRY_LIMIT: int = 5\n\n# Hudson Rock Cavalier API base URL.\n_HR_BASE = \"https://cavalier.hudsonrock.com/api/json/v2/osint-tools\"\n\n# ProxyNova COMB (Collection Of Many Breaches) — email-only, no key needed.\n_PN_BASE = \"https://api.proxynova.com/comb\"\n\n\n@dataclass\nclass StealerEntry:\n    \"\"\"One stealer-log record returned by Hudson Rock.\"\"\"\n\n    date_compromised: str = \"\"\n    stealer_family: str = \"\"\n    computer_name: str = \"\"\n    operating_system: str = \"\"\n    ip: str = \"\"\n    malware_path: str = \"\"\n    antiviruses: list[str] = field(default_factory=list)\n    top_logins: list[str] = field(default_factory=list)\n    top_passwords: list[str] = field(default_factory=list)\n    total_corporate_services: int = 0\n    total_user_services: int = 0\n\n\n@dataclass\nclass CombEntry:\n    \"\"\"One credential line from the ProxyNova COMB dataset.\"\"\"\n\n    line: str = \"\"  # raw \"email:password\" string\n    email: str = \"\"\n    password: str = \"\"\n\n\n@dataclass\nclass LeakReport:\n    \"\"\"Aggregated results from all sources for a single target.\"\"\"\n\n    target: str = \"\"\n    target_type: str = \"\"  # \"email\" | \"domain\" | \"username\"\n    # Hudson Rock\n    hr_message: str = \"\"\n    hr_stealers: list[StealerEntry] = field(default_factory=list)\n    hr_employees: list[dict] = field(default_factory=list)\n    hr_users: list[dict] = field(default_factory=list)\n    # ProxyNova COMB (email only)\n    comb_count: int = 0\n    comb_entries: list[CombEntry] = field(default_factory=list)\n\n\n# Target-type detection\n\n\ndef _detect_type(target: str) -> str:\n    \"\"\"\n    Infer whether *target* is an e-mail address, a domain, or a username.\n\n    :param target: Raw input string from the user.\n    :return: ``'email'``, ``'domain'``, or ``'username'``.\n    \"\"\"\n    if \"@\" in target:\n        return \"email\"\n    if \".\" in target and \" \" not in target:\n        return \"domain\"\n    return \"username\"\n\n\n# HTTP helpers\n\n\ndef _get(url: str, params: dict | None = None) -> dict | None:\n    \"\"\"\n    Performs a GET request with retry / exponential back-off.\n\n    :param url:    Full URL to request.\n    :param params: Optional query-string parameters.\n    :return:       Parsed JSON dict, or ``None`` on permanent failure.\n    \"\"\"\n    headers = {\"User-Agent\": str(randomuser.GetUser())}\n    for attempt in range(1, _MAX_RETRIES + 1):\n        try:\n            resp = requests.get(\n                url,\n                params=params,\n                headers=headers,\n                timeout=_REQUEST_TIMEOUT,\n            )\n            resp.raise_for_status()\n            return resp.json()\n        except requests.exceptions.Timeout:\n            delay = _RETRY_BASE_DELAY * (2 ** (attempt - 1))\n            printer.warning(\n                f\"Request timed out (attempt {attempt}/{_MAX_RETRIES}). \"\n                f\"Retrying in {delay:.0f}s...\"\n            )\n            time.sleep(delay)\n        except requests.exceptions.HTTPError as exc:\n            if exc.response is not None and exc.response.status_code == 404:\n                return {}  # target not found — not an error\n            printer.error(f\"HTTP error: {exc}\")\n            return None\n        except requests.exceptions.RequestException as exc:\n            printer.error(f\"Request failed: {exc}\")\n            return None\n    printer.error(f\"All {_MAX_RETRIES} attempts failed for {url}.\")\n    return None\n\n\n# Hudson Rock fetching & parsing\n\n\ndef _fetch_hudson_rock(\n    target: str, target_type: str\n) -> tuple[str, list[StealerEntry], list[dict], list[dict]]:\n    \"\"\"\n    Query the Hudson Rock Cavalier API for *target*.\n\n    :param target:      The value to search for.\n    :param target_type: ``'email'``, ``'domain'``, or ``'username'``.\n    :return: Tuple of (message, stealers, employees, users).\n    \"\"\"\n    endpoint_map = {\n        \"email\": f\"{_HR_BASE}/search-by-email\",\n        \"domain\": f\"{_HR_BASE}/search-by-domain\",\n        \"username\": f\"{_HR_BASE}/search-by-username\",\n    }\n    param_map = {\n        \"email\": \"email\",\n        \"domain\": \"domain\",\n        \"username\": \"username\",\n    }\n\n    url = endpoint_map[target_type]\n    param_key = param_map[target_type]\n\n    printer.debug(f\"Hudson Rock → {url}?{param_key}={target}\")\n    data = _get(url, params={param_key: target})\n\n    if data is None:\n        return \"\", [], [], []\n    if not data:\n        return \"\", [], [], []\n\n    message: str = data.get(\"message\", \"\")\n\n    # Top-level stealers (email / username responses)\n    raw_stealers: list[dict] = data.get(\"stealers\", []) or []\n    stealers = [_parse_stealer(s) for s in raw_stealers]\n\n    # Prepare employees/users lists (domain responses use different shapes)\n    employees: list[dict] = []\n    users: list[dict] = []\n\n    # If domain-style response contains a \"data\" dict\n    data_block = data.get(\"data\")\n    if isinstance(data_block, dict):\n        # Prefer detailed all_urls entries which include type/occurrence\n        all_urls = data_block.get(\"all_urls\") or []\n        if isinstance(all_urls, list) and all_urls:\n            for item in all_urls:\n                try:\n                    t = (item.get(\"type\") or \"\").lower()\n                    rec = {\n                        \"url\": item.get(\"url\", \"\") or \"\",\n                        \"occurrence\": int(item.get(\"occurrence\") or 0),\n                    }\n                except Exception:\n                    # Skip malformed items\n                    continue\n                if t == \"employee\":\n                    employees.append(rec)\n                elif t == \"user\":\n                    users.append(rec)\n            # Return early; employees/users built from all_urls\n            return message, stealers, employees, users\n\n        # Fallback to employees_urls / clients_urls arrays with occurrence/type\n        emp_urls = data_block.get(\"employees_urls\") or []\n        if isinstance(emp_urls, list):\n            for item in emp_urls:\n                if not isinstance(item, dict):\n                    continue\n                employees.append(\n                    {\n                        \"url\": item.get(\"url\", \"\") or \"\",\n                        \"occurrence\": int(item.get(\"occurrence\") or 0),\n                    }\n                )\n        client_urls = data_block.get(\"clients_urls\") or []\n        if isinstance(client_urls, list):\n            for item in client_urls:\n                if not isinstance(item, dict):\n                    continue\n                users.append(\n                    {\n                        \"url\": item.get(\"url\", \"\") or \"\",\n                        \"occurrence\": int(item.get(\"occurrence\") or 0),\n                    }\n                )\n        # Continue to return at end after attempting data_block parsing\n\n    # If there's a stats block with URL lists (less detailed)\n    stats = data.get(\"stats\")\n    if (not employees and not users) and isinstance(stats, dict):\n        emp_stats_urls = stats.get(\"employees_urls\") or []\n        if isinstance(emp_stats_urls, list):\n            for u in emp_stats_urls:\n                if isinstance(u, str):\n                    employees.append({\"url\": u, \"occurrence\": None})\n        client_stats_urls = stats.get(\"clients_urls\") or []\n        if isinstance(client_stats_urls, list):\n            for u in client_stats_urls:\n                if isinstance(u, str):\n                    users.append({\"url\": u, \"occurrence\": None})\n\n    # Some responses include top-level numeric counts for employees/users - don't treat as lists\n    # Normalize any accidental scalar values from data.get(\"employees\") / data.get(\"users\")\n    raw_employees = data.get(\"employees\")\n    raw_users = data.get(\"users\")\n    if isinstance(raw_employees, list) and not employees:\n        for item in raw_employees:\n            if isinstance(item, dict):\n                employees.append(\n                    {\n                        \"url\": item.get(\"url\", \"\") or \"\",\n                        \"occurrence\": int(item.get(\"occurrence\") or 0),\n                    }\n                )\n    if isinstance(raw_users, list) and not users:\n        for item in raw_users:\n            if isinstance(item, dict):\n                users.append(\n                    {\n                        \"url\": item.get(\"url\", \"\") or \"\",\n                        \"occurrence\": int(item.get(\"occurrence\") or 0),\n                    }\n                )\n\n    # Final normalization: ensure lists are lists (empty if nothing found)\n    if employees is None:\n        employees = []\n    if users is None:\n        users = []\n\n    return message, stealers, employees, users\n\n\ndef _parse_stealer(raw: dict) -> StealerEntry:\n    \"\"\"\n    Convert a raw Hudson Rock stealer dict into a typed :class:`StealerEntry`.\n\n    :param raw: Dict from the API response.\n    :return:    Populated :class:`StealerEntry`.\n    \"\"\"\n    return StealerEntry(\n        date_compromised=raw.get(\"date_compromised\", \"\"),\n        stealer_family=raw.get(\"stealer_family\", \"Unknown\"),\n        computer_name=raw.get(\"computer_name\", \"\"),\n        operating_system=raw.get(\"operating_system\", \"\"),\n        ip=raw.get(\"ip\", \"\"),\n        malware_path=raw.get(\"malware_path\", \"\"),\n        antiviruses=raw.get(\"antiviruses\") or [],\n        top_logins=raw.get(\"top_logins\") or [],\n        top_passwords=raw.get(\"top_passwords\") or [],\n        total_corporate_services=int(raw.get(\"total_corporate_services\") or 0),\n        total_user_services=int(raw.get(\"total_user_services\") or 0),\n    )\n\n\n# ProxyNova COMB fetching & parsing\n\n\ndef _fetch_comb(email: str) -> tuple[int, list[CombEntry]]:\n    \"\"\"\n    Query the ProxyNova COMB dataset for *email*.\n\n    :param email: E-mail address to look up.\n    :return: Tuple of (total_count, list of :class:`CombEntry`).\n    \"\"\"\n    printer.debug(f\"ProxyNova COMB → {_PN_BASE}?query={email}\")\n    data = _get(_PN_BASE, params={\"query\": email})\n\n    if not data:\n        return 0, []\n\n    count: int = int(data.get(\"count\") or 0)\n    lines: list[str] = data.get(\"lines\") or []\n\n    entries: list[CombEntry] = []\n    for line in lines:\n        line = line.strip()\n        if not line:\n            continue\n        # Format is  email:password  — split on the first colon only.\n        if \":\" in line:\n            _, _, password = line.partition(\":\")\n        else:\n            password = \"\"\n        entries.append(CombEntry(line=line, email=email, password=password))\n\n    return count, entries\n\n\n# Display helpers\n\n\ndef _print_stealer(entry: StealerEntry, index: int, total: int) -> None:\n    \"\"\"Print one :class:`StealerEntry` in a clearly labelled block.\"\"\"\n    printer.info(f\"  Entry {index}/{total}\")\n    printer.success(f\"    {'Date compromised':<22} : {entry.date_compromised or 'N/A'}\")\n    printer.success(f\"    {'Stealer family':<22} : {entry.stealer_family or 'N/A'}\")\n    printer.success(f\"    {'Computer':<22} : {entry.computer_name or 'N/A'}\")\n    printer.success(f\"    {'OS':<22} : {entry.operating_system or 'N/A'}\")\n    printer.success(f\"    {'IP address':<22} : {entry.ip or 'N/A'}\")\n    printer.success(f\"    {'Malware path':<22} : {entry.malware_path or 'N/A'}\")\n    if entry.antiviruses:\n        printer.success(f\"    {'Antiviruses':<22} : {', '.join(entry.antiviruses)}\")\n    printer.success(\n        f\"    {'Corporate services':<22} : {entry.total_corporate_services}\"\n    )\n    printer.success(f\"    {'User services':<22} : {entry.total_user_services}\")\n    if entry.top_logins:\n        printer.success(f\"    {'Top logins':<22} : {', '.join(entry.top_logins)}\")\n    if entry.top_passwords:\n        printer.success(f\"    {'Top passwords':<22} : {', '.join(entry.top_passwords)}\")\n    printer.noprefix(\"\")\n\n\ndef _print_comb_entry(entry: CombEntry, index: int) -> None:\n    \"\"\"Print one :class:`CombEntry`.\"\"\"\n    printer.success(f\"  [{index:>3}] {entry.line}\")\n\n\n# Export\n\n\ndef _ask_export() -> str | None:\n    \"\"\"\n    Ask the user whether and in what format to export the report.\n\n    :return: ``'txt'``, ``'csv'``, or ``'json'``; ``None`` if declined.\n    \"\"\"\n    answer = printer.user_input(\"Save results to file? (y/N) : \").strip().lower()\n    if answer not in {\"y\", \"yes\"}:\n        return None\n\n    printer.noprefix(\"\")\n    printer.section(\"Export Format\")\n    printer.info(\"  1 : TXT  (plain text report)\")\n    printer.info(\"  2 : CSV  (spreadsheet-friendly)\")\n    printer.info(\"  3 : JSON (full structured data)\")\n\n    fmt_map = {\"1\": \"txt\", \"2\": \"csv\", \"3\": \"json\", \"\": \"txt\"}\n    choice = printer.user_input(\"Choose format (1/2/3) [default: 1] : \").strip()\n    return fmt_map.get(choice, \"txt\")\n\n\ndef _export(report: LeakReport, fmt: str) -> None:\n    \"\"\"\n    Write *report* to ``scraped_data/`` in the requested format.\n\n    :param report: The fully populated :class:`LeakReport`.\n    :param fmt:    ``'txt'``, ``'csv'``, or ``'json'``.\n    \"\"\"\n    _SAVE_DIR.mkdir(exist_ok=True)\n\n    slug = \"\".join(c if c.isalnum() or c in \"-_.\" else \"_\" for c in report.target)[:50]\n    timestamp = datetime.now().strftime(\"%Y%m%d_%H%M%S\")\n\n    try:\n        match fmt.lower():\n            case \"txt\":\n                filepath = _SAVE_DIR / f\"leaksearch_{slug}_{timestamp}.txt\"\n                with filepath.open(\"w\", encoding=\"utf-8\") as fh:\n                    fh.write(\"Leak Search Report\\n\")\n                    fh.write(f\"Target      : {report.target}\\n\")\n                    fh.write(f\"Type        : {report.target_type}\\n\")\n                    fh.write(\n                        f\"Date        : {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}\\n\"\n                    )\n                    fh.write(\"=\" * 80 + \"\\n\\n\")\n\n                    # Hudson Rock stealers\n                    fh.write(\n                        f\"HUDSON ROCK  ({len(report.hr_stealers)} stealer record(s))\\n\"\n                    )\n                    fh.write(\"-\" * 80 + \"\\n\")\n                    if report.hr_message:\n                        fh.write(f\"Note: {report.hr_message}\\n\\n\")\n                    for i, s in enumerate(report.hr_stealers, 1):\n                        fh.write(f\"  [{i}] Date       : {s.date_compromised}\\n\")\n                        fh.write(f\"      Stealer    : {s.stealer_family}\\n\")\n                        fh.write(f\"      Computer   : {s.computer_name}\\n\")\n                        fh.write(f\"      OS         : {s.operating_system}\\n\")\n                        fh.write(f\"      IP         : {s.ip}\\n\")\n                        fh.write(f\"      Corp svcs  : {s.total_corporate_services}\\n\")\n                        fh.write(f\"      User svcs  : {s.total_user_services}\\n\")\n                        fh.write(f\"      Logins     : {', '.join(s.top_logins)}\\n\")\n                        fh.write(f\"      Passwords  : {', '.join(s.top_passwords)}\\n\")\n                        fh.write(\"\\n\")\n\n                    # Domain-mode employees / users\n                    if report.hr_employees or report.hr_users:\n                        fh.write(\n                            f\"\\nEMPLOYEES ({len(report.hr_employees)}) / \"\n                            f\"USERS ({len(report.hr_users)})\\n\"\n                        )\n                        fh.write(\"-\" * 80 + \"\\n\")\n                        for rec in report.hr_employees + report.hr_users:\n                            for k, v in rec.items():\n                                fh.write(f\"  {k:<20}: {v}\\n\")\n                            fh.write(\"\\n\")\n\n                    # COMB\n                    if report.comb_count:\n                        fh.write(\n                            f\"\\nPROXYNOVA COMB  ({report.comb_count} total hit(s))\\n\"\n                        )\n                        fh.write(\"-\" * 80 + \"\\n\")\n                        for entry in report.comb_entries:\n                            fh.write(f\"  {entry.line}\\n\")\n\n                printer.success(\n                    f\"Results saved → {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n\n            case \"csv\":\n                filepath = _SAVE_DIR / f\"leaksearch_{slug}_{timestamp}.csv\"\n                with filepath.open(\"w\", newline=\"\", encoding=\"utf-8\") as fh:\n                    writer = csv.writer(fh)\n                    # Hudson Rock stealers sheet\n                    writer.writerow(\n                        [\n                            \"source\",\n                            \"date_compromised\",\n                            \"stealer_family\",\n                            \"computer_name\",\n                            \"operating_system\",\n                            \"ip\",\n                            \"total_corporate_services\",\n                            \"total_user_services\",\n                            \"top_logins\",\n                            \"top_passwords\",\n                        ]\n                    )\n                    for s in report.hr_stealers:\n                        writer.writerow(\n                            [\n                                \"hudson_rock\",\n                                s.date_compromised,\n                                s.stealer_family,\n                                s.computer_name,\n                                s.operating_system,\n                                s.ip,\n                                s.total_corporate_services,\n                                s.total_user_services,\n                                \" | \".join(s.top_logins),\n                                \" | \".join(s.top_passwords),\n                            ]\n                        )\n                    # COMB entries\n                    if report.comb_entries:\n                        writer.writerow([])\n                        writer.writerow([\"source\", \"line\", \"email\", \"password\"])\n                        for entry in report.comb_entries:\n                            writer.writerow(\n                                [\n                                    \"proxynova_comb\",\n                                    entry.line,\n                                    entry.email,\n                                    entry.password,\n                                ]\n                            )\n                printer.success(\n                    f\"Results saved → {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n\n            case \"json\":\n                filepath = _SAVE_DIR / f\"leaksearch_{slug}_{timestamp}.json\"\n                payload = {\n                    \"target\": report.target,\n                    \"target_type\": report.target_type,\n                    \"timestamp\": datetime.now().isoformat(),\n                    \"hudson_rock\": {\n                        \"total_stealers\": len(report.hr_stealers),\n                        \"message\": report.hr_message,\n                        \"stealers\": [asdict(s) for s in report.hr_stealers],\n                        \"employees\": report.hr_employees,\n                        \"users\": report.hr_users,\n                    },\n                    \"proxynova_comb\": {\n                        \"total_hits\": report.comb_count,\n                        \"entries\": [asdict(e) for e in report.comb_entries],\n                    },\n                }\n                with filepath.open(\"w\", encoding=\"utf-8\") as fh:\n                    json.dump(payload, fh, indent=2, ensure_ascii=False)\n                printer.success(\n                    f\"Results saved → {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n\n            case _:\n                printer.error(f\"Unknown format '{fmt}'. Use 'txt', 'csv', or 'json'.\")\n\n    except OSError as exc:\n        printer.error(f\"Could not write file: {exc}\")\n\n\ndef _ask_entry_limit() -> int | None:\n    \"\"\"\n    Ask how many stealer/credential entries to display inline.\n\n    :return: An integer cap, or ``None`` meaning \"show all\".\n    \"\"\"\n    printer.noprefix(\"\")\n    printer.info(\"Entries to display per source:\")\n    for key, val in _ENTRY_LIMIT_CHOICES.items():\n        label = str(val) if val is not None else \"all\"\n        printer.noprefix(f\"   {Style.BRIGHT}[{key}]{Style.RESET_ALL} {label}\")\n    choice = printer.user_input(\"Select [1-4] (default = 2) : \").strip()\n    return _ENTRY_LIMIT_CHOICES.get(choice, _DEFAULT_ENTRY_LIMIT)\n\n\n@timer.timer(require_input=True)\ndef lookup(target: str) -> None:\n    \"\"\"\n    Multi-source leak and breach search for an e-mail, domain, or username.\n\n    Sources\n    -------\n    - **Hudson Rock Cavalier API** — stealer-log intelligence (email, domain,\n      username). Returns infected-computer records including date of compromise,\n      stealer family, machine details, and masked credential samples.\n    - **ProxyNova COMB** — Collection Of Many Breaches credential dataset\n      (email only). Returns the total number of hits and raw credential lines.\n\n    :param target: E-mail address, domain name, or username to investigate.\n    \"\"\"\n    target = target.strip()\n    if not target:\n        printer.error(\"Target cannot be empty.\")\n        return\n\n    target_type = _detect_type(target)\n    printer.info(\n        f\"Target {Style.BRIGHT}{target}{Style.RESET_ALL} \"\n        f\"detected as {Style.BRIGHT}{target_type}{Style.RESET_ALL}.\"\n    )\n\n    entry_limit = _ask_entry_limit()\n    report = LeakReport(target=target, target_type=target_type)\n\n    # Source 1 — Hudson Rock                                               #\n    printer.noprefix(\"\")\n    printer.section(\"Hudson Rock Cavalier\")\n    printer.info(f\"Querying Hudson Rock for {Style.BRIGHT}{target}{Style.RESET_ALL}...\")\n\n    hr_msg, hr_stealers, hr_employees, hr_users = _fetch_hudson_rock(\n        target, target_type\n    )\n    report.hr_message = hr_msg\n    report.hr_stealers = hr_stealers\n    report.hr_employees = hr_employees or []\n    report.hr_users = hr_users or []\n\n    if hr_stealers:\n        printer.success(\n            f\"Found {Style.BRIGHT}{len(hr_stealers)}{Style.RESET_ALL} \"\n            \"stealer record(s).\"\n        )\n        if hr_msg:\n            printer.warning(hr_msg)\n        printer.noprefix(\"\")\n\n        display_count = (\n            len(hr_stealers)\n            if entry_limit is None\n            else min(entry_limit, len(hr_stealers))\n        )\n        for i, entry in enumerate(hr_stealers[:display_count], 1):\n            _print_stealer(entry, i, len(hr_stealers))\n\n        if display_count < len(hr_stealers):\n            printer.info(\n                f\"  ... {len(hr_stealers) - display_count} more record(s) not shown \"\n                \"(export to see all).\"\n            )\n\n    elif hr_employees or hr_users:\n        printer.debug(hr_employees, hr_users)\n\n        emp_count = len(hr_employees) if isinstance(hr_employees, (list, tuple)) else 0\n        user_count = len(hr_users) if isinstance(hr_users, (list, tuple)) else 0\n\n        printer.success(\n            f\"Found {Style.BRIGHT}{emp_count}{Style.RESET_ALL} \"\n            f\"employee(s) and \"\n            f\"{Style.BRIGHT}{user_count}{Style.RESET_ALL} user(s).\"\n        )\n        printer.noprefix(\"\")\n\n        all_records = hr_employees + hr_users\n        display_count = (\n            len(all_records)\n            if entry_limit is None\n            else min(entry_limit, len(all_records))\n        )\n        for rec in all_records[:display_count]:\n            for k, v in rec.items():\n                label = k.replace(\"_\", \" \").title()\n                printer.success(f\"    {label:<22} : {v}\")\n            printer.noprefix(\"\")\n\n        if display_count < len(all_records):\n            printer.info(\n                f\"  ... {len(all_records) - display_count} more record(s) not shown \"\n                \"(export to see all).\"\n            )\n\n    else:\n        printer.warning(\"No stealer records found on Hudson Rock for this target.\")\n\n    # Source 2 — ProxyNova COMB (email only)                              #\n    if target_type == \"email\":\n        printer.noprefix(\"\")\n        printer.section(\"ProxyNova COMB Dataset\")\n        printer.info(\n            f\"Querying COMB dataset for {Style.BRIGHT}{target}{Style.RESET_ALL}...\"\n        )\n\n        comb_count, comb_entries = _fetch_comb(target)\n        report.comb_count = comb_count\n        report.comb_entries = comb_entries\n\n        if comb_count:\n            printer.success(\n                f\"Found {Style.BRIGHT}{comb_count:,}{Style.RESET_ALL} \"\n                \"credential hit(s) in the COMB dataset.\"\n            )\n            printer.noprefix(\"\")\n\n            display_count = (\n                len(comb_entries)\n                if entry_limit is None\n                else min(entry_limit, len(comb_entries))\n            )\n            for i, entry in enumerate(comb_entries[:display_count], 1):\n                _print_comb_entry(entry, i)\n\n            if display_count < len(comb_entries):\n                printer.info(\n                    f\"  ... {len(comb_entries) - display_count} more line(s) not shown \"\n                    \"(export to see all).\"\n                )\n        else:\n            printer.warning(\"No entries found in the COMB dataset for this address.\")\n\n    # Summary                                                              #\n    printer.noprefix(\"\")\n    printer.section(\"Summary\")\n    printer.info(f\"Target        : {Style.BRIGHT}{target}{Style.RESET_ALL}\")\n    printer.info(f\"Type          : {target_type}\")\n    printer.info(\n        f\"HR stealers   : {Style.BRIGHT}{len(report.hr_stealers)}{Style.RESET_ALL}\"\n    )\n    if target_type == \"domain\":\n        printer.info(\n            f\"HR employees  : {Style.BRIGHT}{len(report.hr_employees)}{Style.RESET_ALL}\"\n        )\n        printer.info(\n            f\"HR users      : {Style.BRIGHT}{len(report.hr_users)}{Style.RESET_ALL}\"\n        )\n    if target_type == \"email\":\n        printer.info(\n            f\"COMB hits     : {Style.BRIGHT}{report.comb_count:,}{Style.RESET_ALL}\"\n        )\n\n    # Export                                                               #\n    printer.noprefix(\"\")\n    fmt = _ask_export()\n    if fmt:\n        _export(report, fmt)\n    printer.noprefix(\"\")\n    printer.info(\n        f\"Hudson Rock raw data : \"\n        f\"{Style.BRIGHT}https://cavalier.hudsonrock.com{Style.RESET_ALL}\"\n    )\n"
  },
  {
    "path": "utils/local_users.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport getpass\nimport os\nimport subprocess\nimport time\n\nfrom helper import printer, timer\n\n\n@timer.timer(require_input=True)\ndef scan_for_local_users() -> None:\n    \"\"\"\n    Scans for local accounts on the system and their information.\n    \"\"\"\n\n    match os.name:\n        case \"nt\":\n            import psutil\n\n            printer.info(\"Windows system detected.\")\n\n            try:\n                user_info_list = []\n\n                for user in psutil.users():\n                    username = user.name\n                    terminal = user.terminal\n                    host = user.host\n                    started = time.strftime(\n                        \"%m/%d/%Y %H:%M:%S\", time.localtime(user.started)\n                    )\n                    pid = user.pid\n\n                    # Get additional information using subprocess\n                    user_sid = (\n                        subprocess.check_output(\n                            [\"wmic\", \"useraccount\", \"get\", \"sid\", \"/value\"]\n                        )\n                        .decode(\"utf-8\")\n                        .strip()\n                    )\n                    user_domain = (\n                        subprocess.check_output(\n                            [\"wmic\", \"useraccount\", \"get\", \"domain\", \"/value\"]\n                        )\n                        .decode(\"utf-8\")\n                        .strip()\n                    )\n\n                    user_info_list.append(\n                        {\n                            \"Username\": username,\n                            \"Terminal\": terminal,\n                            \"Host\": host,\n                            \"Started\": started,\n                            \"PID\": pid,\n                            \"SID\": user_sid,\n                            \"Domain\": user_domain,\n                        }\n                    )\n\n                printer.noprefix(\"\")\n                printer.section(\"Local User Accounts\")\n\n                for user_info in user_info_list:\n                    printer.success(f\"Username : {user_info['Username']}\")\n                    printer.success(f\"Terminal : {user_info['Terminal']}\")\n                    printer.success(f\"Host     : {user_info['Host']}\")\n                    printer.success(f\"Started  : {user_info['Started']}\")\n                    printer.success(f\"PID      : {user_info['PID']}\")\n                    printer.success(f\"SID      : {user_info['SID']}\")\n                    printer.success(f\"Domain   : {user_info['Domain']}\")\n                    printer.noprefix(\"\")\n\n            except Exception as e:\n                printer.error(f\"Error retrieving account information: {e}\")\n\n        case \"posix\":\n            import grp\n            import pwd\n\n            printer.info(\"Linux system detected.\")\n\n            try:\n                user_info_list = []\n\n                login_name = getpass.getuser()\n\n                for user in pwd.getpwall():\n                    username = user.pw_name\n                    uid = user.pw_uid\n                    gid = user.pw_gid\n                    full_name = user.pw_gecos\n                    home_dir = user.pw_dir\n                    shell = user.pw_shell\n\n                    # Get additional information using grp\n                    group_name = grp.getgrgid(gid)[0]\n\n                    user_info_list.append(\n                        {\n                            \"Username\": username,\n                            \"UID\": uid,\n                            \"GID\": gid,\n                            \"Full Name\": full_name,\n                            \"Home Directory\": home_dir,\n                            \"Shell\": shell,\n                            \"Group Name\": group_name,\n                            \"Login Name\": login_name,\n                        }\n                    )\n\n                printer.noprefix(\"\")\n                printer.section(\"Local User Accounts\")\n\n                for user_info in user_info_list:\n                    printer.success(f\"Username       : {user_info['Username']}\")\n                    printer.success(f\"UID            : {user_info['UID']}\")\n                    printer.success(f\"GID            : {user_info['GID']}\")\n                    printer.success(f\"Full Name      : {user_info['Full Name']}\")\n                    printer.success(f\"Home Directory : {user_info['Home Directory']}\")\n                    printer.success(f\"Shell          : {user_info['Shell']}\")\n                    printer.success(f\"Group Name     : {user_info['Group Name']}\")\n                    printer.success(f\"Login Name     : {user_info['Login Name']}\")\n                    printer.noprefix(\"\")\n\n            except Exception as e:\n                printer.error(f\"Error retrieving account information: {e}\")\n\n        case _:\n            printer.error(\"Unsupported platform..!\")\n"
  },
  {
    "path": "utils/phonenumber_lookup.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport re\nimport subprocess\nimport time\n\nimport phonenumbers as p\nfrom colorama import Style\nfrom phonenumbers import (\n    PhoneNumberType,\n    carrier,\n    geocoder,\n    number_type,\n    timezone,\n)\n\nfrom helper import printer, timer\n\n# Human-readable labels for every PhoneNumberType variant.\n# PhoneNumberType is an int alias, not a true enum, so the dict is keyed by int.\n_NUMBER_TYPE_LABELS: dict[int, str] = {\n    PhoneNumberType.MOBILE: \"Mobile\",\n    PhoneNumberType.FIXED_LINE: \"Fixed Line\",\n    PhoneNumberType.FIXED_LINE_OR_MOBILE: \"Fixed Line or Mobile\",\n    PhoneNumberType.TOLL_FREE: \"Toll-Free\",\n    PhoneNumberType.PREMIUM_RATE: \"Premium Rate\",\n    PhoneNumberType.SHARED_COST: \"Shared Cost\",\n    PhoneNumberType.VOIP: \"VoIP\",\n    PhoneNumberType.PERSONAL_NUMBER: \"Personal Number\",\n    PhoneNumberType.PAGER: \"Pager\",\n    PhoneNumberType.UAN: \"UAN\",\n    PhoneNumberType.VOICEMAIL: \"Voicemail\",\n    PhoneNumberType.UNKNOWN: \"Unknown\",\n}\n\n\n@timer.timer(require_input=True)\ndef lookup(phone_number: str) -> None:\n    \"\"\"\n    Looks up a phone number and checks it across OSINT sources.\n\n    Performs structured number analysis via the phonenumbers library (formats,\n    validity, type, carrier, region, time zones) and then uses *ignorant* to\n    check whether the number is registered on known social-media platforms.\n\n    :param phone_number: The phone number to look up.  Should include the\n                         country code, e.g. ``+358501234567``.\n    \"\"\"\n    phone_number = phone_number.strip()\n\n    try:\n        ph_no = p.parse(phone_number)\n    except p.NumberParseException as exc:\n        printer.error(f\"Could not parse phone number: {exc}\")\n        printer.info(\n            \"Make sure to include the country code, e.g. \"\n            f\"{Style.BRIGHT}+358501234567{Style.RESET_ALL}\"\n        )\n        return\n\n    if not p.is_valid_number(ph_no):\n        printer.warning(\n            \"The number does not appear to be valid. Results may be inaccurate.\"\n        )\n\n    _print_number_info(ph_no)\n    _run_ignorant(ph_no)\n\n\n# Internal helpers\n\n\ndef _print_number_info(ph_no: p.PhoneNumber) -> None:\n    \"\"\"\n    Prints a structured breakdown of the parsed phone number.\n\n    :param ph_no: Parsed :class:`phonenumbers.PhoneNumber` object.\n    \"\"\"\n    e164 = p.format_number(ph_no, p.PhoneNumberFormat.E164)\n    intl = p.format_number(ph_no, p.PhoneNumberFormat.INTERNATIONAL)\n    national = p.format_number(ph_no, p.PhoneNumberFormat.NATIONAL)\n\n    country = p.region_code_for_country_code(ph_no.country_code or 0)\n    no_carrier = carrier.name_for_number(ph_no, \"en\")\n    no_valid = p.is_valid_number(ph_no)\n    no_possible = p.is_possible_number(ph_no)\n    time_zones = timezone.time_zones_for_number(ph_no)\n    region = geocoder.description_for_number(ph_no, \"en\")\n    n_type = _NUMBER_TYPE_LABELS.get(int(number_type(ph_no)), \"Unknown\")\n\n    printer.info(f\"Looking up {Style.BRIGHT}{intl}{Style.RESET_ALL}...\")\n    time.sleep(0.5)\n\n    printer.noprefix(\"\")\n    printer.section(\"Phone Number Details\")\n    printer.success(f\"E.164 Format     : {Style.BRIGHT}{e164}{Style.RESET_ALL}\")\n    printer.success(f\"International    : {intl}\")\n    printer.success(f\"National         : {national}\")\n    printer.success(f\"Country Code     : +{ph_no.country_code}\")\n    printer.success(f\"Country          : {country or 'Unknown'}\")\n    printer.success(f\"Region           : {region or 'Unknown'}\")\n    printer.success(f\"Number Type      : {n_type}\")\n    printer.success(f\"Carrier          : {no_carrier or 'Unknown'}\")\n    printer.success(f\"Valid            : {no_valid}\")\n    printer.success(f\"Possible         : {no_possible}\")\n    printer.success(\n        \"Time Zone(s)     : \" + (\", \".join(time_zones) if time_zones else \"Unknown\")\n    )\n\n\ndef _run_ignorant(ph_no: p.PhoneNumber) -> None:\n    \"\"\"\n    Runs *ignorant* against the phone number and prints social-media results.\n\n    ``ignorant`` expects the country code and the national significant number\n    as two separate positional arguments, e.g.::\n\n        ignorant +1 3455685544\n\n    We derive both values directly from the already-parsed\n    :class:`~phonenumbers.PhoneNumber` object so no extra string manipulation\n    is needed.\n\n    :param ph_no: Parsed :class:`phonenumbers.PhoneNumber` object.\n    \"\"\"\n    printer.noprefix(\"\")\n    printer.section(\"Social Media Presence (via ignorant)\")\n\n    country_code_arg = f\"+{ph_no.country_code}\"\n    national_number_arg = str(ph_no.national_number)\n\n    try:\n        result = subprocess.run(\n            [\n                \"ignorant\",\n                \"--only-used\",  # show only platforms where number IS registered\n                \"--no-color\",  # plain text output for reliable parsing\n                \"--no-clear\",  # do not wipe the terminal\n                country_code_arg,\n                national_number_arg,\n            ],\n            capture_output=True,\n            text=True,\n            timeout=60,\n        )\n\n        combined = result.stdout + result.stderr\n\n        if not combined.strip():\n            printer.warning(\"ignorant returned no output.\")\n            return\n\n        found = _parse_ignorant_output(combined)\n\n        if not found:\n            printer.warning(\"No social media platforms found for this number.\")\n        else:\n            printer.info(f\"Found on {len(found)} platform(s):\")\n            for site in found:\n                printer.success(f\"  [+] {site}\")\n\n        printer.info(\"Credits to megadose (Palenath) for ignorant.\")\n\n    except FileNotFoundError:\n        printer.error(\n            f\"{Style.BRIGHT}ignorant{Style.RESET_ALL} was not found. \"\n            f\"Install it with: {Style.BRIGHT}pip install ignorant{Style.RESET_ALL}\"\n        )\n    except subprocess.TimeoutExpired:\n        printer.error(\"ignorant timed out. The request took too long.\")\n    except subprocess.CalledProcessError as exc:\n        printer.error(f\"ignorant exited with a non-zero status: {exc}\")\n    except Exception as exc:\n        printer.error(f\"Unexpected error while running ignorant: {exc}\")\n\n\ndef _parse_ignorant_output(output: str) -> list[str]:\n    \"\"\"\n    Extracts platform names from ignorant's ``[+] platform.com`` output lines.\n\n    ignorant uses the same ``[+] / [-] / [x]`` convention as holehe.  Because\n    we always pass ``--only-used``, only ``[+]`` lines appear in the output,\n    but we guard against all three variants just in case.\n\n    :param output: Raw stdout (and stderr) captured from the ignorant process.\n    :return: Sorted list of platform names/domains that reported a match.\n    \"\"\"\n    # Strip any residual ANSI escape codes that might slip through.\n    ansi_re = re.compile(r\"\\x1B(?:[@-Z\\\\-_]|\\[[0-?]*[ -/]*[@-~])\")\n    found: list[str] = []\n\n    for line in output.splitlines():\n        clean = ansi_re.sub(\"\", line).strip()\n        # Capture the first token (domain) separately from the rest of the line.\n        # ignorant prints a legend line at the bottom:\n        #   [+] Phone number used, [-] Phone number not used, [x] Rate limit\n        # Real platform domains (snapchat.com, instagram.com) always contain a\n        # dot; the legend word \"Phone\" does not, so we use that to skip it.\n        match = re.match(r\"^\\[(\\+)\\]\\s*(\\S+)\", clean)\n        if match:\n            domain = match.group(2).strip()\n            if \".\" not in domain:\n                continue\n            found.append(domain)\n\n    return sorted(found)\n"
  },
  {
    "path": "utils/port_scanner.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport socket\nfrom concurrent.futures import ThreadPoolExecutor, as_completed\n\nfrom colorama import Style\n\nfrom helper import printer, timer\n\n\n@timer.timer(require_input=True)\ndef scan(ip: str, port_range: int) -> None:\n    \"\"\"\n    Scans for open ports in a given IP address.\n\n    :param ip: IP address.\n    :param port_range: The range of ports to scan.\n    \"\"\"\n    # Initialise fresh state for each run so consecutive scans don't accumulate.\n    open_ports: list[int] = []\n    failed_ports: list[int] = []\n\n    try:\n        printer.info(\n            f\"Scanning for open ports for {Style.BRIGHT}{ip}{Style.RESET_ALL} \"\n            f\"with the port range of {Style.BRIGHT}1-{port_range}{Style.RESET_ALL}...\"\n        )\n        if port_range > 1000:\n            printer.warning(\"This may take a while...\")\n\n        printer.noprefix(\"\")\n        printer.section(\"Port Scan Results\")\n\n        _scan_ports(ip, port_range, open_ports, failed_ports)\n\n        printer.noprefix(\"\")\n        if not open_ports:\n            printer.error(\n                f\"No open ports found for {Style.BRIGHT}{ip}{Style.RESET_ALL}..!\"\n            )\n        else:\n            printer.success(\n                f\"Found {len(open_ports)} open port(s) out of \"\n                f\"{len(open_ports) + len(failed_ports)} scanned on {ip}.\"\n            )\n    except KeyboardInterrupt:\n        printer.error(\"Cancelled..!\")\n    except RecursionError:\n        printer.error(\"Unexpected recursion error.\")\n\n\ndef _scan_ports(\n    ip: str,\n    port_range: int,\n    open_ports: list[int],\n    failed_ports: list[int],\n) -> None:\n    \"\"\"\n    Scans for open ports in a given IP address using a thread pool.\n\n    :param ip: IP address.\n    :param port_range: The range of ports to scan.\n    :param open_ports: List to collect open port numbers into.\n    :param failed_ports: List to collect closed/timed-out port numbers into.\n    \"\"\"\n    with ThreadPoolExecutor(max_workers=50) as executor:\n        futures = {\n            executor.submit(_connect_to_port, ip, port, open_ports, failed_ports): port\n            for port in range(1, port_range + 1)\n        }\n        for future in as_completed(futures):\n            # Exceptions are already handled inside connect_to_port; just drain results.\n            future.result()\n\n\ndef _connect_to_port(\n    ip: str,\n    port: int,\n    open_ports: list[int],\n    failed_ports: list[int],\n) -> None:\n    \"\"\"\n    Attempts to connect to a single port on the given IP address.\n    Records the result in open_ports or failed_ports accordingly.\n\n    :param ip: IP address.\n    :param port: Port number.\n    :param open_ports: Shared list to append open ports to.\n    :param failed_ports: Shared list to append failed ports to.\n    \"\"\"\n    try:\n        with socket.socket() as sock:\n            sock.settimeout(0.5)\n            sock.connect((ip, port))\n            open_ports.append(port)\n            printer.success(f\"Port {Style.BRIGHT}{port}{Style.RESET_ALL}/TCP is open\")\n    except socket.timeout:\n        failed_ports.append(port)\n    except ConnectionRefusedError:\n        failed_ports.append(port)\n    except socket.error as e:\n        printer.error(\n            f\"An error occurred while scanning port {Style.BRIGHT}{port}{Style.RESET_ALL} \"\n            f\"for {Style.BRIGHT}{ip}{Style.RESET_ALL} : {e}\"\n        )\n"
  },
  {
    "path": "utils/search_username.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport asyncio\nimport json\nfrom datetime import datetime\n\nimport aiohttp\nfrom colorama import Style\n\nfrom helper import printer, randomuser, timer, url_helper\n\n\n@timer.timer(require_input=True)\ndef search(username: str) -> None:\n    \"\"\"\n    Performs a search for the given username.\n\n    :param username: The username to search for.\n    \"\"\"\n    try:\n        _check_user_from_data(username)\n    except KeyboardInterrupt:\n        printer.error(\"Cancelled..!\")\n\n\ndef _check_user_from_data(username: str) -> dict:\n    \"\"\"\n    Scans for the given username across many different sites.\n\n    :param username: The username to scan for.\n    :return: A dict summarising the search parameters and matched sites.\n    \"\"\"\n    # Read the data file once and reuse it throughout this call.\n    data = url_helper.read_local_content(\"resources/data.json\")\n    if not isinstance(data, dict):\n        printer.error(\"Failed to load site data.\")\n        return {}\n\n    sites = data.get(\"sites\", [])\n    printer.info(\n        f\"Searching for {Style.BRIGHT}{username}{Style.RESET_ALL} \"\n        f\"across {len(sites)} different websites...\"\n    )\n\n    printer.noprefix(\"\")\n    printer.section(\"Username Search Results\")\n\n    results = asyncio.run(_make_requests(username, sites))\n\n    now = datetime.now().strftime(\"%m/%d/%Y %H:%M:%S\")\n    user_json = {\n        \"search-params\": {\n            \"username\": username,\n            \"sites-number\": len(sites),\n            \"date\": now,\n        },\n        \"sites\": results,\n    }\n\n    return user_json\n\n\nasync def _make_requests(username: str, sites: list) -> list:\n    \"\"\"\n    Makes the requests to all sites and returns a list of matched results.\n\n    :param username: The username to scan for.\n    :param sites: List of site configuration dicts from data.json.\n    :return: List of site dicts where the username was found.\n    \"\"\"\n    async with aiohttp.ClientSession(\n        timeout=aiohttp.ClientTimeout(total=20)\n    ) as session:\n        tasks = [\n            asyncio.ensure_future(_make_request(session, content, username))\n            for content in sites\n        ]\n        results = await asyncio.gather(*tasks)\n\n    # Filter out None entries (sites where the username was not found).\n    return [r for r in results if r is not None]\n\n\nasync def _make_request(\n    session: aiohttp.ClientSession, content: dict, username: str\n) -> dict | None:\n    \"\"\"\n    Makes a single request to one site and returns the site dict on a match.\n\n    :param session: The shared aiohttp client session.\n    :param content: Site configuration dict from data.json.\n    :param username: The username to check.\n    :return: The site dict if the username was found, otherwise None.\n    \"\"\"\n    url = content[\"url\"].format(username=username)\n    json_body = None\n    headers = {\"User-Agent\": str(randomuser.GetUser())}\n\n    if \"headers\" in content:\n        # NOTE: eval() is used here because the data.json format stores headers\n        # as Python expression strings. Treat data.json as trusted input only.\n        headers.update(eval(content[\"headers\"]))  # noqa: S307\n\n    if \"json\" in content:\n        json_body = json.loads(content[\"json\"].format(username=username))\n\n    try:\n        async with session.request(\n            content[\"method\"],\n            url,\n            json=json_body,\n            proxy=None,\n            headers=headers,\n            ssl=False,\n        ) as response:\n            # `valid` is a Python expression string evaluated against the response.\n            # NOTE: same caveat as above — data.json must be trusted.\n            if eval(content[\"valid\"]):  # noqa: S307\n                printer.success(\n                    f\"{Style.BRIGHT}{content['app']:<20}{Style.RESET_ALL} : {url}\"\n                )\n                return content\n    except Exception:\n        pass\n\n    return None\n"
  },
  {
    "path": "utils/web_reconnaissance.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport csv\nimport json\nimport time\nfrom dataclasses import asdict, dataclass, field\nfrom datetime import datetime\nfrom pathlib import Path\n\nfrom colorama import Style\nfrom ddgs import DDGS\nfrom ddgs.exceptions import DDGSException, RatelimitException, TimeoutException\n\nfrom helper import printer, timer\n\n# How many results to fetch per query (user-selectable).\n_RESULTS_CHOICES: dict[str, int] = {\"1\": 5, \"2\": 10, \"3\": 20, \"4\": 50}\n_DEFAULT_MAX_RESULTS: int = 5\n\n# Seconds to wait between consecutive queries to reduce rate-limit risk.\n_INTER_QUERY_DELAY: float = 1.5\n\n# Retry policy for transient errors.\n_MAX_RETRIES: int = 3\n_RETRY_BASE_DELAY: float = 4.0  # doubles on each attempt (exponential back-off)\n\n# Directory where optional JSON exports are written.\n_SAVE_DIR = Path(\"scraped_data\")\n\n\n@dataclass\nclass SearchResult:\n    \"\"\"One search result entry.\"\"\"\n\n    title: str\n    url: str\n    snippet: str = field(default=\"\")\n    label: str = field(default=\"\")  # dork label / section the result came from\n\n\n# Dork template tables\n#\n# Each entry is a (label, query_template) tuple.\n# Templates reference placeholders that match keys in the *target* dict:\n#   {query}    – raw query (General mode)\n#   {name}     – full person name\n#   {email}    – full e-mail address\n#   {domain}   – domain name (auto-derived from e-mail when needed)\n#   {username} – account handle / screen name\n#   {phone}    – phone number string\n#   {target}   – generic placeholder (Custom Dork mode)\n\n_PERSON_DORKS: list[tuple[str, str]] = [\n    (\"General mention\", '\"{name}\"'),\n    (\"LinkedIn\", '\"{name}\" site:linkedin.com'),\n    (\"Twitter / X\", '\"{name}\" (site:twitter.com OR site:x.com)'),\n    (\"Facebook\", '\"{name}\" site:facebook.com'),\n    (\"Instagram\", '\"{name}\" site:instagram.com'),\n    (\"GitHub\", '\"{name}\" site:github.com'),\n    (\"YouTube\", '\"{name}\" site:youtube.com'),\n    (\"Email links\", '\"{name}\" email'),\n    (\"Phone links\", '\"{name}\" phone'),\n    (\"News / press\", '\"{name}\" (news OR article OR press OR interview)'),\n    (\"Public records\", '\"{name}\" (address OR court OR record OR lawsuit)'),\n    (\"Images\", '\"{name}\" photos OR pictures'),\n]\n\n_EMAIL_DORKS: list[tuple[str, str]] = [\n    (\"Direct mention\", '\"{email}\"'),\n    (\n        \"Paste sites\",\n        '\"{email}\" (site:pastebin.com OR site:ghostbin.com OR site:hastebin.com)',\n    ),\n    (\"GitHub\", '\"{email}\" site:github.com'),\n    (\"LinkedIn\", '\"{email}\" site:linkedin.com'),\n    (\"Forum / community\", '\"{email}\" (forum OR community OR post OR thread)'),\n    (\"Same-domain users\", '\"@{domain}\" email'),\n    (\"Data breach mention\", '\"{email}\" (breach OR leaked OR dump OR exposed)'),\n    (\"Registration traces\", '\"{email}\" (register OR account OR signup OR profile)'),\n]\n\n_DOMAIN_DORKS: list[tuple[str, str]] = [\n    (\"All indexed pages\", \"site:{domain}\"),\n    (\"Subdomains\", \"site:*.{domain}\"),\n    (\"PDF documents\", \"site:{domain} filetype:pdf\"),\n    (\n        \"Office documents\",\n        \"site:{domain} (filetype:doc OR filetype:docx OR filetype:xls OR filetype:xlsx OR filetype:ppt)\",\n    ),\n    (\n        \"Config / secrets\",\n        \"site:{domain} (filetype:env OR filetype:conf OR filetype:bak OR filetype:cfg OR filetype:ini)\",\n    ),\n    (\n        \"Admin panels\",\n        \"site:{domain} (inurl:admin OR inurl:administrator OR inurl:wp-admin OR inurl:cpanel OR inurl:panel)\",\n    ),\n    (\n        \"Login pages\",\n        \"site:{domain} (inurl:login OR inurl:signin OR inurl:auth OR inurl:sso)\",\n    ),\n    (\n        \"API endpoints\",\n        \"site:{domain} (inurl:api OR inurl:v1 OR inurl:v2 OR inurl:graphql OR inurl:swagger)\",\n    ),\n    (\"Email addresses\", '\"@{domain}\"'),\n    (\"Related / linked\", \"related:{domain}\"),\n    (\"Mentions elsewhere\", '\"{domain}\" -site:{domain}'),\n    (\n        \"Tech stack hints\",\n        \"site:{domain} (intext:powered OR intext:built OR intext:wordpress OR intext:shopify)\",\n    ),\n]\n\n_USERNAME_DORKS: list[tuple[str, str]] = [\n    (\"General mention\", '\"{username}\"'),\n    (\"GitHub\", 'site:github.com \"{username}\"'),\n    (\"Reddit\", 'site:reddit.com/user \"{username}\"'),\n    (\"Twitter / X\", '(site:twitter.com OR site:x.com) \"{username}\"'),\n    (\"Instagram\", 'site:instagram.com \"{username}\"'),\n    (\"LinkedIn\", 'site:linkedin.com/in \"{username}\"'),\n    (\"Steam\", 'site:steamcommunity.com \"{username}\"'),\n    (\"Hacker News\", 'site:news.ycombinator.com \"{username}\"'),\n    (\n        \"Dev platforms\",\n        '(site:dev.to OR site:medium.com OR site:hashnode.dev) \"{username}\"',\n    ),\n    (\"TikTok\", 'site:tiktok.com \"{username}\"'),\n    (\"Twitch\", 'site:twitch.tv \"{username}\"'),\n    (\"YouTube\", 'site:youtube.com \"{username}\"'),\n]\n\n_PHONE_DORKS: list[tuple[str, str]] = [\n    (\"Direct mention\", '\"{phone}\"'),\n    (\"Name / owner\", '\"{phone}\" (name OR owner OR who OR person)'),\n    (\"TrueCaller\", '\"{phone}\" site:truecaller.com'),\n    (\n        \"Social profiles\",\n        '\"{phone}\" (site:linkedin.com OR site:facebook.com OR site:twitter.com)',\n    ),\n    (\n        \"Business listing\",\n        '\"{phone}\" (company OR business OR office OR ltd OR llc OR inc)',\n    ),\n    (\n        \"Public directory\",\n        '\"{phone}\" (directory OR whitepages OR yellowpages OR phonebook)',\n    ),\n    (\"Address links\", '\"{phone}\" (address OR location OR street)'),\n    (\"Paste / leak sites\", \"'{phone}' (site:pastebin.com OR site:ghostbin.com)\"),\n]\n\n# Custom Dork mode uses a single user-supplied template; no pre-built list.\n_CUSTOM_DORKS: list[tuple[str, str]] = []\n\n\n# Mode registry\n#\n# Key → (display_name, dork_list, input_specs)\n# input_specs: list of (prompt_label, target_dict_key) pairs.\n# The first spec is the \"primary\" value shown in progress messages.\n\n_ModeSpec = tuple[str, list[tuple[str, str]], list[tuple[str, str]]]\n\n_MODES: dict[str, _ModeSpec] = {\n    \"1\": (\"General\", [], [(\"Search query\", \"query\")]),\n    \"2\": (\"Person\", _PERSON_DORKS, [(\"Full name (e.g. Jane Doe)\", \"name\")]),\n    \"3\": (\"Email\", _EMAIL_DORKS, [(\"Email address\", \"email\")]),\n    \"4\": (\"Domain\", _DOMAIN_DORKS, [(\"Domain (e.g. example.com)\", \"domain\")]),\n    \"5\": (\"Username\", _USERNAME_DORKS, [(\"Username / handle\", \"username\")]),\n    \"6\": (\n        \"Phone Number\",\n        _PHONE_DORKS,\n        [(\"Phone number (e.g. +13125550123)\", \"phone\")],\n    ),\n    \"7\": (\n        \"Custom Dork\",\n        _CUSTOM_DORKS,\n        [\n            (\n                \"Dork template  (use {target} as placeholder, e.g. site:{target} filetype:pdf)\",\n                \"template\",\n            ),\n            (\"Target value\", \"target\"),\n        ],\n    ),\n}\n\n\n@timer.timer(require_input=True)\ndef websearch() -> None:\n    \"\"\"\n    Interactive multi-mode OSINT web search powered by the ddgs library.\n\n    Presents a mode-selection menu, collects relevant inputs, builds\n    tailored dork queries, fetches results with automatic retry/back-off,\n    deduplicates by URL, and displays grouped output.  Optionally exports\n    all collected results to a timestamped JSON file in ``scraped_data/``.\n\n    Modes\n    -----\n    1  General       – free-form query, single search\n    2  Person        – 12 dorks for a person's online presence\n    3  Email         – 8 dorks for e-mail OSINT\n    4  Domain        – 12 dorks for website / domain reconnaissance\n    5  Username      – 12 dorks across major platforms\n    6  Phone Number  – 8 dorks for phone-number attribution\n    7  Custom Dork   – user-defined dork template with a ``{target}`` slot\n    \"\"\"\n    _print_mode_menu()\n\n    mode_key = printer.user_input(\"Select mode [1-7] : \").strip()\n    if mode_key not in _MODES:\n        printer.error(\"Invalid mode. Please choose a number from 1 to 7.\")\n        return\n\n    mode_name, dorks, input_spec = _MODES[mode_key]\n    target: dict[str, str] = {}\n\n    for label, key in input_spec:\n        value = printer.user_input(f\"{label} : \").strip()\n        if not value:\n            printer.error(\"Input cannot be empty.\")\n            return\n        target[key] = value\n\n    # Auto-derive domain from e-mail address.\n    if \"email\" in target and \"@\" in target[\"email\"] and \"domain\" not in target:\n        target[\"domain\"] = target[\"email\"].split(\"@\", 1)[1]\n\n    # For Custom Dork mode, build a one-element dork list on the fly.\n    if mode_key == \"7\":\n        template = target.pop(\"template\")\n        dorks = [(\"Custom Dork\", template)]\n\n    primary_value = list(target.values())[0]\n    max_results = _ask_max_results()\n    save_fmt = _ask_save_results()\n\n    printer.noprefix(\"\")\n\n    # General mode — single query\n    if mode_key == \"1\":\n        query = target[\"query\"]\n        printer.info(f\"Searching for {Style.BRIGHT}{query}{Style.RESET_ALL}...\")\n        printer.noprefix(\"\")\n        printer.section(\"Web Search Results\")\n\n        results = _fetch_results(query, max_results, label=\"General\")\n        if not results:\n            printer.warning(f\"No results found for '{query}'.\")\n            return\n\n        for result in results:\n            _print_result(result)\n\n        printer.info(f\"Total results : {Style.BRIGHT}{len(results)}{Style.RESET_ALL}\")\n\n        if save_fmt:\n            _save_results(results, mode_name, primary_value, save_fmt)\n        return\n\n    # OSINT dork modes (2–7)\n    printer.info(\n        f\"Running {Style.BRIGHT}{mode_name}{Style.RESET_ALL} search for \"\n        f\"{Style.BRIGHT}{primary_value}{Style.RESET_ALL}...\"\n    )\n    printer.warning(\n        f\"Executing {len(dorks)} quer{'y' if len(dorks) == 1 else 'ies'} \"\n        f\"({max_results} results each) with a {_INTER_QUERY_DELAY}s delay between \"\n        \"each to reduce rate-limiting.\"\n    )\n    printer.noprefix(\"\")\n    printer.section(f\"{mode_name} Search Results\")\n\n    seen_urls: set[str] = set()\n    all_results: list[SearchResult] = []\n    total_unique = 0\n\n    for idx, (label, template) in enumerate(dorks, start=1):\n        query = _build_query(template, target)\n        printer.debug(f\"[{idx}/{len(dorks)}] {label} → {query}\")\n\n        results = _fetch_results(query, max_results, label=label)\n        new_results = [r for r in results if r.url not in seen_urls]\n\n        if new_results:\n            printer.noprefix(\"\")\n            printer.info(\n                f\"── [{idx}/{len(dorks)}] {label}  \"\n                f\"({Style.BRIGHT}{len(new_results)}{Style.RESET_ALL} new)\"\n            )\n            for result in new_results:\n                seen_urls.add(result.url)\n                all_results.append(result)\n                total_unique += 1\n                _print_result(result)\n        else:\n            printer.debug(f\"No new results for [{label}]\")\n\n        if idx < len(dorks):\n            time.sleep(_INTER_QUERY_DELAY)\n\n    printer.noprefix(\"\")\n    printer.info(\n        f\"Search complete — \"\n        f\"{Style.BRIGHT}{total_unique}{Style.RESET_ALL} unique result(s) across \"\n        f\"{len(dorks)} quer{'y' if len(dorks) == 1 else 'ies'}.\"\n    )\n\n    if save_fmt and all_results:\n        _save_results(all_results, mode_name, primary_value, save_fmt)\n    elif save_fmt:\n        printer.warning(\"No results to save.\")\n\n\n# Internal helpers\n\n\ndef _print_mode_menu() -> None:\n    \"\"\"Displays the mode selection menu.\"\"\"\n    printer.noprefix(\"\")\n    printer.section(\"Web Reconnaissance — Select Mode\")\n    for key, (name, dorks, _) in _MODES.items():\n        hint = f\"({len(dorks)} dork queries)\" if dorks else \"(free-form)\"\n        printer.noprefix(f\"   {Style.BRIGHT}[{key}]{Style.RESET_ALL} {name:<18} {hint}\")\n    printer.noprefix(\"\")\n\n\ndef _ask_max_results() -> int:\n    \"\"\"Prompt the user to choose how many results to fetch per query.\"\"\"\n    printer.noprefix(\"\")\n    printer.info(\"Results per query:\")\n    for key, val in _RESULTS_CHOICES.items():\n        printer.noprefix(f\"   {Style.BRIGHT}[{key}]{Style.RESET_ALL} {val}\")\n    choice = printer.user_input(\"Select [1-4] (default = 1) : \").strip()\n    selected = _RESULTS_CHOICES.get(choice, _DEFAULT_MAX_RESULTS)\n    printer.debug(f\"Max results per query set to {selected}.\")\n    return selected\n\n\ndef _ask_save_results() -> str | None:\n    \"\"\"\n    Ask whether to export collected results and, if so, in which format.\n\n    :return: ``'txt'``, ``'csv'``, or ``'json'`` if the user wants to save;\n             ``None`` if they decline.\n    \"\"\"\n    answer = printer.user_input(\"Save results to file? (y/N) : \").strip().lower()\n    if answer not in {\"y\", \"yes\"}:\n        return None\n\n    printer.noprefix(\"\")\n    printer.section(\"Export Format\")\n    printer.info(\"  1 : TXT  (plain text)\")\n    printer.info(\"  2 : CSV  (comma-separated values)\")\n    printer.info(\"  3 : JSON (structured data)\")\n\n    format_map = {\"1\": \"txt\", \"2\": \"csv\", \"3\": \"json\", \"\": \"txt\"}\n    choice = printer.user_input(\"Choose format (1/2/3) [default: 1] : \").strip()\n    return format_map.get(choice, \"txt\")\n\n\ndef _save_results(\n    results: list[SearchResult],\n    mode_name: str,\n    target: str,\n    fmt: str = \"json\",\n) -> None:\n    \"\"\"\n    Export *results* to a timestamped file under ``scraped_data/``.\n\n    :param results:   List of :class:`SearchResult` objects to persist.\n    :param mode_name: Human-readable mode name used in the filename.\n    :param target:    Primary target value used in the filename.\n    :param fmt:       Export format — ``'txt'``, ``'csv'``, or ``'json'``.\n    \"\"\"\n    _SAVE_DIR.mkdir(exist_ok=True)\n\n    slug = (\n        mode_name.lower().replace(\" \", \"_\")\n        + \"_\"\n        + \"\".join(c if c.isalnum() or c in \"-_.\" else \"_\" for c in target)[:40]\n    )\n    timestamp = datetime.now().strftime(\"%Y%m%d_%H%M%S\")\n\n    try:\n        match fmt.lower():\n            case \"txt\":\n                filepath = _SAVE_DIR / f\"websearch_{slug}_{timestamp}.txt\"\n                with filepath.open(\"w\", encoding=\"utf-8\") as fh:\n                    fh.write(f\"Deep Web Search — {mode_name}\\n\")\n                    fh.write(f\"Target  : {target}\\n\")\n                    fh.write(\n                        f\"Date    : {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}\\n\"\n                    )\n                    fh.write(f\"Results : {len(results)}\\n\")\n                    fh.write(\"-\" * 80 + \"\\n\\n\")\n                    for r in results:\n                        fh.write(f\"[{r.label}]\\n\" if r.label else \"\")\n                        fh.write(f\"Title   : {r.title}\\n\")\n                        fh.write(f\"URL     : {r.url}\\n\")\n                        if r.snippet:\n                            fh.write(f\"Snippet : {r.snippet}\\n\")\n                        fh.write(\"\\n\")\n                printer.success(\n                    f\"Results saved → {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n\n            case \"csv\":\n                filepath = _SAVE_DIR / f\"websearch_{slug}_{timestamp}.csv\"\n                with filepath.open(\"w\", newline=\"\", encoding=\"utf-8\") as fh:\n                    writer = csv.writer(fh)\n                    writer.writerow([\"Label\", \"Title\", \"URL\", \"Snippet\"])\n                    for r in results:\n                        writer.writerow([r.label, r.title, r.url, r.snippet])\n                printer.success(\n                    f\"Results saved → {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n\n            case \"json\":\n                filepath = _SAVE_DIR / f\"websearch_{slug}_{timestamp}.json\"\n                payload = {\n                    \"mode\": mode_name,\n                    \"target\": target,\n                    \"timestamp\": datetime.now().isoformat(),\n                    \"total_results\": len(results),\n                    \"results\": [asdict(r) for r in results],\n                }\n                with filepath.open(\"w\", encoding=\"utf-8\") as fh:\n                    json.dump(payload, fh, indent=2, ensure_ascii=False)\n                printer.success(\n                    f\"Results saved → {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n\n            case _:\n                printer.error(f\"Unknown format '{fmt}'. Use 'txt', 'csv', or 'json'.\")\n\n    except OSError as exc:\n        printer.error(f\"Could not write results file: {exc}\")\n\n\ndef _build_query(template: str, target: dict[str, str]) -> str:\n    \"\"\"\n    Fill a dork template with values from *target*.\n\n    Unknown placeholders are silently left as-is so a partially-filled\n    template still produces a runnable (if suboptimal) query string.\n\n    :param template: Dork template containing ``{placeholder}`` fields.\n    :param target:   Mapping of placeholder names → values.\n    :return: Formatted query string.\n    \"\"\"\n    try:\n        return template.format(**target)\n    except KeyError:\n        return template\n\n\ndef _fetch_results(\n    query: str,\n    max_results: int,\n    *,\n    label: str = \"\",\n) -> list[SearchResult]:\n    \"\"\"\n    Fetch text search results via the ``ddgs`` library with exponential\n    back-off retry on rate-limit and timeout errors.\n\n    :param query:       The search query string.\n    :param max_results: Maximum number of results to request.\n    :param label:       Section label attached to each :class:`SearchResult`.\n    :return:            List of :class:`SearchResult` objects.\n    \"\"\"\n    for attempt in range(1, _MAX_RETRIES + 1):\n        try:\n            with DDGS() as ddgs:\n                raw: list[dict] = ddgs.text(query, max_results=max_results) or []\n\n            return [\n                SearchResult(\n                    title=r.get(\"title\", \"\").strip(),\n                    url=r.get(\"href\", \"\").strip(),\n                    snippet=r.get(\"body\", \"\").strip(),\n                    label=label,\n                )\n                for r in raw\n                if r.get(\"href\", \"\").strip()\n            ]\n\n        except RatelimitException:\n            delay = _RETRY_BASE_DELAY * (2 ** (attempt - 1))\n            printer.warning(\n                f\"Rate limited — waiting {delay:.0f}s \"\n                f\"(attempt {attempt}/{_MAX_RETRIES})...\"\n            )\n            time.sleep(delay)\n\n        except TimeoutException:\n            printer.warning(f\"Request timed out (attempt {attempt}/{_MAX_RETRIES}).\")\n            if attempt < _MAX_RETRIES:\n                time.sleep(_RETRY_BASE_DELAY)\n\n        except DDGSException as exc:\n            printer.error(f\"Search error: {exc}\")\n            return []\n\n        except Exception as exc:  # noqa: BLE001\n            printer.error(f\"Unexpected error during search: {exc}\")\n            return []\n\n    printer.error(\n        f\"All {_MAX_RETRIES} attempts failed\"\n        + (f\" for [{label}]\" if label else \"\")\n        + \".\"\n    )\n    return []\n\n\ndef _print_result(result: SearchResult) -> None:\n    \"\"\"\n    Print a single :class:`SearchResult` in the standard toolkit layout::\n\n        [+] Bold Title\n                https://example.com/path\n                Snippet text, truncated to 220 chars if needed…\n\n    :param result: The result to render.\n    \"\"\"\n    printer.success(f\"{Style.BRIGHT}{result.title}{Style.RESET_ALL}\")\n    printer.noprefix(f\"    {result.url}\")\n    if result.snippet:\n        snippet = (\n            result.snippet\n            if len(result.snippet) <= 220\n            else result.snippet[:217] + \"...\"\n        )\n        printer.noprefix(f\"    {snippet}\")\n    printer.noprefix(\"\")\n"
  },
  {
    "path": "utils/web_scrape.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport asyncio\nimport csv\nimport json\nfrom datetime import datetime\nfrom pathlib import Path\nfrom urllib.parse import urljoin, urlparse\n\nimport aiohttp\nfrom bs4 import BeautifulSoup\nfrom colorama import Style\n\nfrom helper import printer, randomuser, timer\n\n\ndef _export_links(links: set, base_url: str, format_type: str = \"txt\") -> None:\n    \"\"\"\n    Exports scraped links to a file in the specified format.\n\n    :param links: Set of scraped links\n    :param base_url: Base URL that was scraped\n    :param format_type: Export format ('txt', 'csv', or 'json')\n    \"\"\"\n    if not links:\n        printer.warning(\"No links to export!\")\n        return\n\n    # Create output directory if it doesn't exist\n    output_dir = Path(\"scraped_data\")\n    output_dir.mkdir(exist_ok=True)\n\n    # Generate filename with timestamp\n    timestamp = datetime.now().strftime(\"%Y%m%d_%H%M%S\")\n    domain = urlparse(base_url).netloc.replace(\".\", \"_\")\n    filename = f\"{domain}_{timestamp}\"\n\n    try:\n        match format_type.lower():\n            case \"txt\":\n                filepath = output_dir / f\"{filename}.txt\"\n                with open(filepath, \"w\", encoding=\"utf-8\") as f:\n                    f.write(f\"Scraped links from: {base_url}\\n\")\n                    f.write(f\"Date: {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}\\n\")\n                    f.write(f\"Total links: {len(links)}\\n\")\n                    f.write(\"-\" * 80 + \"\\n\\n\")\n                    for link in sorted(links):\n                        f.write(f\"{link}\\n\")\n                printer.success(\n                    f\"Links exported to {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n            case \"csv\":\n                filepath = output_dir / f\"{filename}.csv\"\n                with open(filepath, \"w\", newline=\"\", encoding=\"utf-8\") as f:\n                    writer = csv.writer(f)\n                    writer.writerow([\"URL\", \"Domain\", \"Path\"])\n                    for link in sorted(links):\n                        parsed = urlparse(link)\n                        writer.writerow([link, parsed.netloc, parsed.path])\n                printer.success(\n                    f\"Links exported to {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n            case \"json\":\n                filepath = output_dir / f\"{filename}.json\"\n                link_data = {\n                    \"metadata\": {\n                        \"source_url\": base_url,\n                        \"scraped_date\": datetime.now().isoformat(),\n                        \"total_links\": len(links),\n                    },\n                    \"links\": [\n                        {\n                            \"url\": link,\n                            \"domain\": urlparse(link).netloc,\n                            \"path\": urlparse(link).path,\n                            \"scheme\": urlparse(link).scheme,\n                        }\n                        for link in sorted(links)\n                    ],\n                }\n                with open(filepath, \"w\", encoding=\"utf-8\") as f:\n                    json.dump(link_data, f, indent=2, ensure_ascii=False)\n                printer.success(\n                    f\"Links exported to {Style.BRIGHT}{filepath}{Style.RESET_ALL}\"\n                )\n            case _:\n                printer.error(\n                    f\"Invalid format: {format_type}. Use 'txt', 'csv', or 'json'.\"\n                )\n\n    except Exception as e:\n        printer.error(f\"Error exporting links: {e}\")\n\n\n@timer.timer(require_input=True)\ndef scrape(url: str) -> None:\n    \"\"\"\n    Scrapes links from the given url.\n\n    :param url: url of the website.\n    \"\"\"\n    printer.debug(f\"Scraping {urlparse(url).netloc}\")\n\n    # Use a fresh set for every invocation so state never leaks between runs.\n    scraped_links: set[str] = set()\n\n    try:\n        response = printer.user_input(\n            \"Do you want to scrape the linked pages as well? (y/N) : \"\n        )\n        if response.lower() in {\"y\", \"yes\"}:\n            printer.info(\n                f\"Scraping links from {Style.BRIGHT}{url}{Style.RESET_ALL} and all linked pages...\"\n            )\n            printer.warning(\"This may take a while depending on the size of the site.\")\n            printer.noprefix(\"\")\n            printer.section(\"Scraped Links\")\n            asyncio.run(_scrape_links(url, scraped_links, recursive=True))\n            printer.noprefix(\"\")\n            printer.success(\"Scraping completed.\")\n        else:\n            printer.info(f\"Scraping links from {Style.BRIGHT}{url}{Style.RESET_ALL}...\")\n            printer.noprefix(\"\")\n            printer.section(\"Scraped Links\")\n            asyncio.run(_scrape_links(url, scraped_links, recursive=False))\n            printer.noprefix(\"\")\n            printer.success(\"Scraping completed.\")\n\n        # Ask user if they want to export the results\n        if scraped_links:\n            printer.info(f\"{len(scraped_links)} link(s) collected in total.\")\n            export_response = printer.user_input(\n                \"Do you want to export the scraped links? (y/N) : \"\n            )\n            if export_response.lower() in {\"y\", \"yes\"}:\n                printer.noprefix(\"\")\n                printer.section(\"Export\")\n                printer.info(\"  1 : TXT  (plain text)\")\n                printer.info(\"  2 : CSV  (comma-separated values)\")\n                printer.info(\"  3 : JSON (structured data)\")\n\n                format_choice = printer.user_input(\n                    \"Choose format (1/2/3) [default: 1] : \"\n                ).strip()\n\n                format_map = {\n                    \"1\": \"txt\",\n                    \"2\": \"csv\",\n                    \"3\": \"json\",\n                    \"\": \"txt\",  # default\n                }\n\n                export_format = format_map.get(format_choice, \"txt\")\n                _export_links(scraped_links, url, export_format)\n\n    except KeyboardInterrupt:\n        printer.error(\"Cancelled..!\")\n    except Exception as e:\n        printer.error(f\"Error : {e}\")\n\n\nasync def _fetch(session: aiohttp.ClientSession, url: str) -> str:\n    \"\"\"\n    Fetches the HTML content of a URL.\n\n    :param session: The shared aiohttp client session.\n    :param url: URL to fetch.\n    :return: Response body as text, or an empty string on error.\n    \"\"\"\n    headers = {\"User-Agent\": str(randomuser.GetUser())}\n    try:\n        async with session.get(url, headers=headers) as response:\n            return await response.text()\n    except Exception:\n        return \"\"\n\n\nasync def _parse_links(content: str, base_url: str) -> list[tuple[str, str]]:\n    \"\"\"\n    Parses all anchor tags from *content* and returns absolute (href, text) pairs.\n\n    :param content: Raw HTML string.\n    :param base_url: Base URL used to resolve relative hrefs.\n    :return: List of (absolute_url, link_text) tuples.\n    \"\"\"\n    soup = BeautifulSoup(content, \"html.parser\")\n    return [\n        (urljoin(base_url, str(link.get(\"href\"))), link.get_text(strip=True))\n        for link in soup.find_all(\"a\")\n    ]\n\n\nasync def _scrape_links(\n    url: str,\n    scraped_links: set[str],\n    recursive: bool = False,\n) -> None:\n    \"\"\"\n    Scrapes links from *url* and, optionally, from every discovered page.\n\n    :param url: The URL to scrape.\n    :param scraped_links: Shared set used to track already-visited URLs across\n                          recursive calls, preventing infinite loops.\n    :param recursive: When True, every newly discovered URL is scraped as well.\n    \"\"\"\n    async with aiohttp.ClientSession() as session:\n        html_content = await _fetch(session, url)\n        links = await _parse_links(html_content, url)\n\n        for href, text in links:\n            if href not in scraped_links:\n                scraped_links.add(href)\n                printer.success(\n                    f\"{len(scraped_links)} Link(s) found : \"\n                    f\"{Style.BRIGHT}{href} - {text}{Style.RESET_ALL}\"\n                )\n\n                if recursive:\n                    await _scrape_links(href, scraped_links, recursive=True)\n"
  },
  {
    "path": "utils/whois_lookup.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport whoisdomain\nfrom colorama import Style\n\nfrom helper import printer, timer\n\n\n@timer.timer(require_input=True)\ndef check_whois(domain: str) -> None:\n    \"\"\"\n    Looks up WhoIs information for a given domain.\n\n    :param domain: The domain name to look up.\n    \"\"\"\n    try:\n        q = whoisdomain.query(domain)\n        printer.info(f\"Looking up {Style.BRIGHT}{domain}{Style.RESET_ALL}...\")\n        printer.debug(q)\n        printer.noprefix(\"\")\n        printer.section(\"WhoIs Results\")\n\n        for key, value in q.__dict__.items():\n            # Skip None, empty strings, and empty lists/dicts.\n            if value is None or value == \"\" or value == [] or value == {}:\n                continue\n            label = key.replace(\"_\", \" \").title()\n            printer.success(f\"{label} : {value}\")\n\n    except Exception as e:\n        printer.error(f\"Error : {e}\")\n        printer.error(\n            f\"Make sure {Style.BRIGHT}whois{Style.RESET_ALL} is installed on your system.\"\n        )\n"
  },
  {
    "path": "utils/wifi_finder.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport os\nimport subprocess\n\nfrom colorama import Style\n\nfrom helper import printer, timer\n\n\n@timer.timer(require_input=True)\ndef scan_nearby_wifi() -> None:\n    \"\"\"\n    Performs a basic scan for nearby Wi-Fi networks.\n\n    Requires netsh for Windows and nmcli for Linux.\n    \"\"\"\n    match os.name:\n        case \"nt\":\n            _scan_windows()\n        case \"posix\":\n            _scan_linux()\n        case _:\n            printer.error(\"Unsupported platform..!\")\n\n\ndef _scan_windows() -> None:\n    printer.info(\n        f\"Windows system detected... Performing {Style.BRIGHT}netsh{Style.RESET_ALL} scan...\"\n    )\n    try:\n        output = subprocess.check_output([\"netsh\", \"wlan\", \"show\", \"networks\"])\n        _parse_windows_output(output.decode(\"utf-8\"))\n    except subprocess.CalledProcessError as e:\n        printer.error(f\"Error : {e.returncode} - {e.stderr}\")\n\n\ndef _scan_linux() -> None:\n    printer.info(\n        f\"Linux system detected... Performing {Style.BRIGHT}nmcli{Style.RESET_ALL} scan...\"\n    )\n    try:\n        # Use terse mode (-t) with explicit fields so the output is easy to split\n        # reliably regardless of column widths or terminal size.\n        # Fields: IN-USE, SSID, SIGNAL, SECURITY\n        output = subprocess.check_output(\n            [\"nmcli\", \"-t\", \"-f\", \"IN-USE,SSID,SIGNAL,SECURITY\", \"dev\", \"wifi\"]\n        )\n        _parse_linux_output(output.decode(\"utf-8\"))\n    except FileNotFoundError:\n        printer.error(\n            f\"Could not find {Style.BRIGHT}nmcli{Style.RESET_ALL}. \"\n            \"Is NetworkManager installed on your system?\"\n        )\n    except subprocess.CalledProcessError as e:\n        printer.error(f\"Error : {e.returncode} - {e.stderr}\")\n        printer.error(f\"Is your system using {Style.BRIGHT}nmcli{Style.RESET_ALL}?\")\n\n\ndef _parse_windows_output(output: str) -> None:\n    \"\"\"\n    Parses the output of `netsh wlan show networks` and prints each network.\n\n    :param output: Raw decoded output from netsh.\n    \"\"\"\n    networks: list[dict] = []\n    current: dict = {}\n\n    for line in output.splitlines():\n        line = line.strip()\n        if line.startswith(\"SSID\") and \"BSSID\" not in line:\n            # Start of a new network block.\n            if current:\n                networks.append(current)\n            parts = line.split(\":\", 1)\n            current = {\n                \"ssid\": parts[1].strip() if len(parts) > 1 else \"\",\n                \"signal\": \"\",\n                \"encryption\": \"\",\n            }\n        elif line.startswith(\"Signal\"):\n            parts = line.split(\":\", 1)\n            current[\"signal\"] = parts[1].strip() if len(parts) > 1 else \"\"\n        elif line.startswith(\"Authentication\"):\n            parts = line.split(\":\", 1)\n            current[\"encryption\"] = parts[1].strip() if len(parts) > 1 else \"\"\n\n    if current:\n        networks.append(current)\n\n    if not networks:\n        printer.warning(\"No Wi-Fi networks found.\")\n        return\n\n    printer.noprefix(\"\")\n    printer.section(f\"Wi-Fi Networks ({len(networks)} found)\")\n    for network in networks:\n        printer.success(\n            f\"  {network['ssid']}\"\n            f\"  (Signal: {network['signal'] or 'N/A'},\"\n            f\" Encryption: {network['encryption'] or 'N/A'})\"\n        )\n\n\ndef _parse_linux_output(output: str) -> None:\n    \"\"\"\n    Parses the terse output of `nmcli -t -f IN-USE,SSID,SIGNAL,SECURITY dev wifi`\n    and prints every visible network, marking the connected one with an asterisk.\n\n    Terse format fields are separated by ':'. Literal colons inside field values\n    are escaped as '\\\\:' by nmcli.\n\n    :param output: Raw decoded terse nmcli output.\n    \"\"\"\n    networks: list[dict] = []\n\n    for line in output.splitlines():\n        line = line.strip()\n        if not line:\n            continue\n\n        # Split on unescaped colons only.\n        # nmcli escapes literal colons inside values as '\\:', so we split on ':'\n        # that is NOT preceded by a backslash.\n        import re\n\n        parts = re.split(r\"(?<!\\\\):\", line)\n\n        # Un-escape any '\\:' sequences left in values.\n        parts = [p.replace(\"\\\\:\", \":\") for p in parts]\n\n        if len(parts) < 4:\n            continue\n\n        in_use, ssid, signal, security = (\n            parts[0],\n            parts[1],\n            parts[2],\n            \":\".join(parts[3:]),\n        )\n        connected = in_use.strip() == \"*\"\n\n        networks.append(\n            {\n                \"ssid\": ssid.strip() or \"(hidden)\",\n                \"signal\": signal.strip(),\n                \"security\": security.strip() or \"None\",\n                \"connected\": connected,\n            }\n        )\n\n    if not networks:\n        printer.warning(\"No Wi-Fi networks found.\")\n        return\n\n    printer.noprefix(\"\")\n    printer.section(f\"Wi-Fi Networks ({len(networks)} found)\")\n    for network in networks:\n        indicator = (\n            f\"{Style.BRIGHT}*{Style.RESET_ALL} \" if network[\"connected\"] else \"  \"\n        )\n        printer.success(\n            f\"{indicator}{network['ssid']}\"\n            f\"  (Signal: {network['signal']},\"\n            f\" Security: {network['security']})\"\n        )\n"
  },
  {
    "path": "utils/wifi_vault.py",
    "content": "\"\"\"\nCopyright (c) 2023-2026. Vili and contributors.\n\nThis program is free software: you can redistribute it and/or modify\nit under the terms of the GNU General Public License as published by\nthe Free Software Foundation, either version 3 of the License, or\n(at your option) any later version.\n\nThis program is distributed in the hope that it will be useful,\nbut WITHOUT ANY WARRANTY; without even the implied warranty of\nMERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\nGNU General Public License for more details.\n\nYou should have received a copy of the GNU General Public License\nalong with this program.  If not, see <https://www.gnu.org/licenses/>.\n\"\"\"\n\nimport os\nimport re\nimport subprocess\n\nfrom helper import printer, timer\n\n\n@timer.timer(require_input=True)\ndef get_local_passwords() -> None:\n    \"\"\"Retrieves saved Wi-Fi passwords on the system.\"\"\"\n    match os.name:\n        case \"nt\":\n            printer.info(\"Windows system detected.\")\n            try:\n                output = subprocess.check_output(\n                    \"netsh wlan show profiles\", shell=True\n                ).decode(\"utf-8\")\n                profile_names = [\n                    line.split(\":\")[1].strip()\n                    for line in output.splitlines()\n                    if \"All User Profile\" in line\n                ]\n\n                printer.noprefix(\"\")\n                printer.section(\"Saved Wi-Fi Passwords\")\n\n                for profile_name in profile_names:\n                    try:\n                        wifi_info = subprocess.check_output(\n                            'netsh wlan show profile name=\"{}\" key=clear'.format(\n                                profile_name\n                            ),\n                            shell=True,\n                        ).decode(\"utf-8\")\n\n                        password_index = wifi_info.find(\"Key Content\")\n                        if password_index != -1:\n                            password_start = password_index + len(\"Key Content\") + 2\n                            password = (\n                                wifi_info[password_start:].split(\"\\r\\n\")[0].strip()\n                            )\n                            printer.success(f\"Network  : {profile_name}\")\n                            printer.success(f\"Password : {password}\")\n                        else:\n                            printer.success(f\"Network  : {profile_name}\")\n                            printer.warning(\"Password : (none saved)\")\n                        printer.noprefix(\"\")\n                    except subprocess.CalledProcessError as e:\n                        printer.error(\n                            f\"Error retrieving Wi-Fi info for {profile_name} : {e}\"\n                        )\n            except subprocess.CalledProcessError as e:\n                printer.error(f\"Error retrieving profile names : {e}\")\n\n        case \"posix\":\n            printer.info(\"Linux system detected.\")\n            try:\n                output = subprocess.check_output(\n                    [\"nmcli\", \"-f\", \"NAME,UUID\", \"connection\", \"show\"]\n                )\n                connections = re.findall(r\"(\\S+)\\s+([0-9a-f-]{36})\", output.decode())\n\n                printer.noprefix(\"\")\n                printer.section(\"Saved Wi-Fi Passwords\")\n\n                for ssid, uuid in connections:\n                    try:\n                        password_output = subprocess.check_output(\n                            [\n                                \"nmcli\",\n                                \"-s\",\n                                \"-g\",\n                                \"802-11-wireless-security.psk\",\n                                \"connection\",\n                                \"show\",\n                                uuid,\n                            ]\n                        )\n                        password = password_output.decode().strip()\n\n                        printer.success(f\"Network  : {ssid}\")\n                        printer.success(f\"Password : {password}\")\n                        printer.noprefix(\"\")\n                    except subprocess.CalledProcessError as e:\n                        printer.error(f\"Error retrieving password for {ssid} : {e}\")\n\n            except subprocess.CalledProcessError as e:\n                printer.error(f\"Error retrieving saved connections : {e}\")\n                printer.error(\"Is your system using nmcli?\")\n\n        case _:\n            printer.error(\"Unsupported platform..!\")\n"
  }
]