From 2ee6afa8accf019fdad0f9d8185830b11c4c31dc Mon Sep 17 00:00:00 2001 From: IoI_xD Date: Sat, 17 Jun 2023 19:02:39 -0700 Subject: [PATCH] Squashed all previous commits. --- .gitattributes | 1 + .gitignore | 25 + .../BlackBoxAndInternalRust.code-workspace | 20 + .vscode/rust.code-workspace | 27 + .vscode/settings.json | 12 + LICENSE | 661 ++++++++++++ README.md | 15 + build.gradle | 80 ++ build.sh | 12 + gradle.properties | 0 gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 62076 bytes gradle/wrapper/gradle-wrapper.properties | 6 + gradlew | 245 +++++ gradlew.bat | 92 ++ include/net_ioixd_blackbox_Native.h | 69 ++ native/.gitignore | 2 + native/.vscode/settings.json | 3 + native/Cargo.toml | 16 + native/settings.gradle | 0 native/src/lib.rs | 362 +++++++ settings.gradle | 1 + .../java/net/ioixd/blackbox/BlackBox.java | 81 ++ .../net/ioixd/blackbox/BlackBoxPlugin.java | 379 +++++++ .../ioixd/blackbox/BlackBoxPluginLoader.java | 178 ++++ .../ioixd/blackbox/BlackBoxPluginMeta.java | 101 ++ src/main/java/net/ioixd/blackbox/Native.java | 14 + .../exceptions/MissingFunctionException.java | 10 + .../NativeLibraryLoadException.java | 10 + .../NativeLibrarySymbolLoadException.java | 10 + .../extendables/ExtendableBiomeProvider.java | 49 + .../extendables/ExtendableBukkitRunnable.java | 28 + .../ExtendableCommandExecutor.java | 31 + .../ExtendableConfigurationSerializable.java | 29 + .../extendables/ExtendableConsumer.java | 30 + .../ExtendableConversationCanceller.java | 53 + .../ExtendableConversationPrefix.java | 32 + .../ExtendableFileConfiguration.java | 40 + .../extendables/ExtendableHelpTopic.java | 31 + .../ExtendableHelpTopicFactory.java | 31 + .../extendables/ExtendableMapRenderer.java | 29 + .../extendables/ExtendableMetadataValue.java | 137 +++ .../extendables/ExtendableNoiseGenerator.java | 32 + .../ExtendablePersistentDataType.java | 66 ++ .../extendables/ExtendablePlugin.java | 259 +++++ .../extendables/ExtendablePluginBase.java | 257 +++++ .../extendables/ExtendablePluginLoader.java | 93 ++ .../extendables/ExtendablePrompt.java | 57 + .../extendables/ExtendableTabCompleter.java | 31 + .../extendables/ExtendableTabExecutor.java | 44 + .../net/ioixd/blackbox/extendables/Misc.java | 23 + .../__skipped__ExtendableConfiguration | 981 ++++++++++++++++++ src/main/resources/.gitignore | 2 + src/main/resources/plugin.yml | 4 + 53 files changed, 4801 insertions(+) create mode 100644 .gitattributes create mode 100644 .gitignore create mode 100644 .vscode/BlackBoxAndInternalRust.code-workspace create mode 100644 .vscode/rust.code-workspace create mode 100644 .vscode/settings.json create mode 100644 LICENSE create mode 100644 README.md create mode 100755 build.gradle create mode 100755 build.sh create mode 100755 gradle.properties create mode 100755 gradle/wrapper/gradle-wrapper.jar create mode 100755 gradle/wrapper/gradle-wrapper.properties create mode 100755 gradlew create mode 100755 gradlew.bat create mode 100644 include/net_ioixd_blackbox_Native.h create mode 100755 native/.gitignore create mode 100644 native/.vscode/settings.json create mode 100755 native/Cargo.toml create mode 100644 native/settings.gradle create mode 100755 native/src/lib.rs create mode 100755 settings.gradle create mode 100755 src/main/java/net/ioixd/blackbox/BlackBox.java create mode 100644 src/main/java/net/ioixd/blackbox/BlackBoxPlugin.java create mode 100644 src/main/java/net/ioixd/blackbox/BlackBoxPluginLoader.java create mode 100644 src/main/java/net/ioixd/blackbox/BlackBoxPluginMeta.java create mode 100755 src/main/java/net/ioixd/blackbox/Native.java create mode 100644 src/main/java/net/ioixd/blackbox/exceptions/MissingFunctionException.java create mode 100644 src/main/java/net/ioixd/blackbox/exceptions/NativeLibraryLoadException.java create mode 100644 src/main/java/net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableBiomeProvider.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableBukkitRunnable.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableCommandExecutor.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableConfigurationSerializable.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableConsumer.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableConversationCanceller.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableConversationPrefix.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableFileConfiguration.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableHelpTopic.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableHelpTopicFactory.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableMapRenderer.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableMetadataValue.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableNoiseGenerator.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendablePersistentDataType.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendablePlugin.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendablePluginBase.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendablePluginLoader.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendablePrompt.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableTabCompleter.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/ExtendableTabExecutor.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/Misc.java create mode 100644 src/main/java/net/ioixd/blackbox/extendables/__skipped__ExtendableConfiguration create mode 100755 src/main/resources/.gitignore create mode 100755 src/main/resources/plugin.yml diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..f4d278d --- /dev/null +++ b/.gitattributes @@ -0,0 +1 @@ +libraries/* linguist-vendored diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..3e8b464 --- /dev/null +++ b/.gitignore @@ -0,0 +1,25 @@ +.idea/ + +bin + +.gradle +**/build/ +!src/**/build/ + +# Ignore Gradle GUI config +gradle-app.setting + +# Avoid ignoring Gradle wrapper jar file (.jar files are usually ignored) +!gradle-wrapper.jar + +# Avoid ignore Gradle wrappper properties +!gradle-wrapper.properties + +# Cache of project +.gradletasknamecache + +# Eclipse Gradle plugin generated files +# Eclipse Core +.project +# JDT-specific (Eclipse Java Development Tools) +.classpath diff --git a/.vscode/BlackBoxAndInternalRust.code-workspace b/.vscode/BlackBoxAndInternalRust.code-workspace new file mode 100644 index 0000000..4c35e01 --- /dev/null +++ b/.vscode/BlackBoxAndInternalRust.code-workspace @@ -0,0 +1,20 @@ +{ + "folders": [ + { + "path": ".." + }, + { + "path": "../libraries/spigotjsongen" + } + ], + "settings": { + "java.compile.nullAnalysis.mode": "automatic", + "java.configuration.updateBuildConfiguration": "automatic", + "rust-analyzer.linkedProjects": [ + "./native/Cargo.toml", + "./libraries/blackbox-rs/Cargo.toml", + "./examples/block_break_event/Cargo.toml", + "./libraries/blackbox-rs/Cargo.toml" + ] + } +} \ No newline at end of file diff --git a/.vscode/rust.code-workspace b/.vscode/rust.code-workspace new file mode 100644 index 0000000..75c1bea --- /dev/null +++ b/.vscode/rust.code-workspace @@ -0,0 +1,27 @@ +{ + "folders": [ + { + "path": ".." + }, + { + "path": "../libraries/blackbox-rs" + }, + { + "path": "../libraries/spigotjsongen" + } + ], + "settings": { + "java.compile.nullAnalysis.mode": "automatic", + "java.configuration.updateBuildConfiguration": "automatic", + "rust-analyzer.linkedProjects": [ + "./native/Cargo.toml", + "./examples/block_break_event/Cargo.toml", + "./libraries/blackbox-rs/Cargo.toml", + "./Cargo.toml" + ], + "[python]": { + "editor.defaultFormatter": "ms-python.autopep8" + }, + "python.formatting.provider": "none" + } +} \ No newline at end of file diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..ddbc294 --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,12 @@ +{ + "java.compile.nullAnalysis.mode": "automatic", + "files.autoGuessEncoding": true, + "files.encoding": "utf8", + "java.jdt.ls.vmargs": "-Dfile.encoding=UTF-8", + "javac-linter.javac": "javac -Dfile.encoding=UTF-8", + "java.configuration.updateBuildConfiguration": "automatic", + "[python]": { + "editor.defaultFormatter": "ms-python.autopep8" + }, + "python.formatting.provider": "none", +} \ No newline at end of file diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..be3f7b2 --- /dev/null +++ b/LICENSE @@ -0,0 +1,661 @@ + GNU AFFERO GENERAL PUBLIC LICENSE + Version 3, 19 November 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU Affero General Public License is a free, copyleft license for +software and other kinds of works, specifically designed to ensure +cooperation with the community in the case of network server software. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +our General Public Licenses are intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + Developers that use our General Public Licenses protect your rights +with two steps: (1) assert copyright on the software, and (2) offer +you this License which gives you legal permission to copy, distribute +and/or modify the software. + + A secondary benefit of defending all users' freedom is that +improvements made in alternate versions of the program, if they +receive widespread use, become available for other developers to +incorporate. Many developers of free software are heartened and +encouraged by the resulting cooperation. However, in the case of +software used on network servers, this result may fail to come about. +The GNU General Public License permits making a modified version and +letting the public access it on a server without ever releasing its +source code to the public. + + The GNU Affero General Public License is designed specifically to +ensure that, in such cases, the modified source code becomes available +to the community. It requires the operator of a network server to +provide the source code of the modified version running there to the +users of that server. Therefore, public use of a modified version, on +a publicly accessible server, gives the public access to the source +code of the modified version. + + An older license, called the Affero General Public License and +published by Affero, was designed to accomplish similar goals. This is +a different license, not a version of the Affero GPL, but Affero has +released a new version of the Affero GPL which permits relicensing under +this license. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU Affero General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Remote Network Interaction; Use with the GNU General Public License. + + Notwithstanding any other provision of this License, if you modify the +Program, your modified version must prominently offer all users +interacting with it remotely through a computer network (if your version +supports such interaction) an opportunity to receive the Corresponding +Source of your version by providing access to the Corresponding Source +from a network server at no charge, through some standard or customary +means of facilitating copying of software. This Corresponding Source +shall include the Corresponding Source for any work covered by version 3 +of the GNU General Public License that is incorporated pursuant to the +following paragraph. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the work with which it is combined will remain governed by version +3 of the GNU General Public License. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU Affero General Public License from time to time. Such new versions +will be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU Affero General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU Affero General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU Affero General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright (C) + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU Affero General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU Affero General Public License for more details. + + You should have received a copy of the GNU Affero General Public License + along with this program. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If your software can interact with users remotely through a computer +network, you should also make sure that it provides a way for users to +get its source. For example, if your program is a web application, its +interface could display a "Source" link that leads users to an archive +of the code. There are many ways you could offer source, and different +solutions will be better for different programs; see section 13 for the +specific requirements. + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU AGPL, see +. diff --git a/README.md b/README.md new file mode 100644 index 0000000..058ad5f --- /dev/null +++ b/README.md @@ -0,0 +1,15 @@ +# BlackBox + +BlackBox is a Minecraft plugin that will allow you to write plugins in many other compiled languages besides Java. It does what you could do with JNI, but it handles all the Minecraft stuff for you automatically, so you don't even have to use Java's tooling ever again. Because JNI can be very unruley to work with, its developed alongside a [Rust library](https://github.com/BlackBoxMC/blackbox-rs/), and a C++ library is planned, although it is low priority. + +The repo is licensed under AGPL. I understand this may is a controversional, but I've put too much work into this project to risk somebody bringing it into closed source (and GPL is useless since this is a plugin). I'd like to make it clear that I don't care about your users knowing their rights, I don't care about GitHub forks or etc. violating the license, and I don't think a lot of the loopholes against the AGPL apply here. All I care about is if you modify the plugin, I'd like to see the source. + +--- + +**This plugin is made knowing why this hasn't been done before and why you might not want to do it.** + +- Distributing plugins made this way becomes much harder. +- You technically lose performance since JNI calls are slow (...they take about 15ms on a good machine but this might matter in some scenarios) +- You lose the ability to link to, and therefore create addons for, other plugins, unless you want to fork this repo and add support for the plugin in question (I am considering adding support in the helper library for the most popular plugins). + +**It's assumed that you don't care.** This is for the people who are sick of Java, never want to work with Gradle or IntelliJ ever again, and just want to use a language that they actually like (assuming they dislike Java). diff --git a/build.gradle b/build.gradle new file mode 100755 index 0000000..d9f88fd --- /dev/null +++ b/build.gradle @@ -0,0 +1,80 @@ +plugins { + id 'com.github.johnrengelman.shadow' version '7.1.0' + id 'kr.entree.spigradle' version '2.2.4' + id 'io.freefair.lombok' version '6.3.0' + id 'java' + id 'jacoco' + id 'idea' +} + +group = 'net.ioixd' +version = '1.0-SNAPSHOT' + +compileJava { + options.compilerArgs += ["-h", file("include")] +} + +repositories { + mavenCentral() + maven { + url = 'https://hub.spigotmc.org/nexus/content/repositories/snapshots/' + + // As of Gradle 5.1, you can limit this to only those + // dependencies you expect from it + content { + includeGroup 'org.bukkit' + includeGroup 'org.spigotmc' + } + } + maven { + name = "sonatype" + url = "https://oss.sonatype.org/content/groups/public/" + } +} +dependencies { + implementation "org.spigotmc:spigot-api:1.20.1-R0.1-SNAPSHOT" + implementation "io.github.classgraph:classgraph:4.8.160" + implementation "org.reflections:reflections:0.10.2" + implementation "io.papermc.paper:paper-api:1.20.1-R0.1-SNAPSHOT" + + implementation "org.slf4j:slf4j-api:2.0.7" + implementation "org.eclipse.aether:aether-api:1.1.0" +} + + +shadowJar { + dependencies { + include(dependency("io.papermc.paper:paper-api:1.20.1-R0.1-SNAPSHOT")) + include(dependency("io.github.classgraph:classgraph:4.8.160")) + } +} + +tasks.build.dependsOn(shadowJar) +tasks.prepareSpigotPlugins.dependsOn(shadowJar) + + +def targetJavaVersion = 17 +java { + def javaVersion = JavaVersion.toVersion(targetJavaVersion) + sourceCompatibility = javaVersion + targetCompatibility = javaVersion + if (JavaVersion.current() < javaVersion) { + toolchain.languageVersion = JavaLanguageVersion.of(targetJavaVersion) + } +} + +tasks.withType(JavaCompile).configureEach { + if (targetJavaVersion >= 10 || JavaVersion.current().isJava10Compatible()) { + options.release = targetJavaVersion + } +} + +processResources { + def props = [version: version] + inputs.properties props + filteringCharset 'UTF-8' + filesMatching('plugin.yml') { + expand props + } +} +defaultTasks 'shadowJar' diff --git a/build.sh b/build.sh new file mode 100755 index 0000000..a2fcad4 --- /dev/null +++ b/build.sh @@ -0,0 +1,12 @@ +cd native + +rustup target add x86_64-pc-windows-gnu +cargo build --release --target x86_64-pc-windows-gnu +cp ../native/target/x86_64-pc-windows-gnu/release/native.dll ../src/main/resources/native.dll +rustup target add x86_64-unknown-linux-gnu +cargo build --release --target x86_64-unknown-linux-gnu +cp ../native/target/x86_64-unknown-linux-gnu/release/libnative.so ../src/main/resources/libnative.so + +cd .. + +./gradlew shadowJar diff --git a/gradle.properties b/gradle.properties new file mode 100755 index 0000000..e69de29 diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar new file mode 100755 index 0000000000000000000000000000000000000000..c1962a79e29d3e0ab67b14947c167a862655af9b GIT binary patch literal 62076 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 literal 0 HcmV?d00001 diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties new file mode 100755 index 0000000..37aef8d --- /dev/null +++ b/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,6 @@ +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-8.1.1-bin.zip +networkTimeout=10000 +zipStoreBase=GRADLE_USER_HOME +zipStorePath=wrapper/dists diff --git a/gradlew b/gradlew new file mode 100755 index 0000000..aeb74cb --- /dev/null +++ b/gradlew @@ -0,0 +1,245 @@ +#!/bin/sh + +# +# Copyright © 2015-2021 the original authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +############################################################################## +# +# Gradle start up script for POSIX generated by Gradle. +# +# Important for running: +# +# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is +# noncompliant, but you have some other compliant shell such as ksh or +# bash, then to run this script, type that shell name before the whole +# command line, like: +# +# ksh Gradle +# +# Busybox and similar reduced shells will NOT work, because this script +# requires all of these POSIX shell features: +# * functions; +# * expansions «$var», «${var}», «${var:-default}», «${var+SET}», +# «${var#prefix}», «${var%suffix}», and «$( cmd )»; +# * compound commands having a testable exit status, especially «case»; +# * various built-in commands including «command», «set», and «ulimit». +# +# Important for patching: +# +# (2) This script targets any POSIX shell, so it avoids extensions provided +# by Bash, Ksh, etc; in particular arrays are avoided. +# +# The "traditional" practice of packing multiple parameters into a +# space-separated string is a well documented source of bugs and security +# problems, so this is (mostly) avoided, by progressively accumulating +# options in "$@", and eventually passing that to Java. +# +# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, +# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; +# see the in-line comments for details. +# +# There are tweaks for specific operating systems such as AIX, CygWin, +# Darwin, MinGW, and NonStop. +# +# (3) This script is generated from the Groovy template +# https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt +# within the Gradle project. +# +# You can find Gradle at https://github.com/gradle/gradle/. +# +############################################################################## + +# Attempt to set APP_HOME + +# Resolve links: $0 may be a link +app_path=$0 + +# Need this for daisy-chained symlinks. +while + APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path + [ -h "$app_path" ] +do + ls=$( ls -ld "$app_path" ) + link=${ls#*' -> '} + case $link in #( + /*) app_path=$link ;; #( + *) app_path=$APP_HOME$link ;; + esac +done + +# This is normally unused +# shellcheck disable=SC2034 +APP_BASE_NAME=${0##*/} +APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD=maximum + +warn () { + echo "$*" +} >&2 + +die () { + echo + echo "$*" + echo + exit 1 +} >&2 + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +nonstop=false +case "$( uname )" in #( + CYGWIN* ) cygwin=true ;; #( + Darwin* ) darwin=true ;; #( + MSYS* | MINGW* ) msys=true ;; #( + NONSTOP* ) nonstop=true ;; +esac + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD=$JAVA_HOME/jre/sh/java + else + JAVACMD=$JAVA_HOME/bin/java + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD=java + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." +fi + +# Increase the maximum file descriptors if we can. +if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then + case $MAX_FD in #( + max*) + # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC3045 + MAX_FD=$( ulimit -H -n ) || + warn "Could not query maximum file descriptor limit" + esac + case $MAX_FD in #( + '' | soft) :;; #( + *) + # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC3045 + ulimit -n "$MAX_FD" || + warn "Could not set maximum file descriptor limit to $MAX_FD" + esac +fi + +# Collect all arguments for the java command, stacking in reverse order: +# * args from the command line +# * the main class name +# * -classpath +# * -D...appname settings +# * --module-path (only if needed) +# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. + +# For Cygwin or MSYS, switch paths to Windows format before running java +if "$cygwin" || "$msys" ; then + APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) + CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) + + JAVACMD=$( cygpath --unix "$JAVACMD" ) + + # Now convert the arguments - kludge to limit ourselves to /bin/sh + for arg do + if + case $arg in #( + -*) false ;; # don't mess with options #( + /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath + [ -e "$t" ] ;; #( + *) false ;; + esac + then + arg=$( cygpath --path --ignore --mixed "$arg" ) + fi + # Roll the args list around exactly as many times as the number of + # args, so each arg winds up back in the position where it started, but + # possibly modified. + # + # NB: a `for` loop captures its iteration list before it begins, so + # changing the positional parameters here affects neither the number of + # iterations, nor the values presented in `arg`. + shift # remove old arg + set -- "$@" "$arg" # push replacement arg + done +fi + + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + +# Collect all arguments for the java command; +# * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of +# shell script including quotes and variable substitutions, so put them in +# double quotes to make sure that they get re-expanded; and +# * put everything else in single quotes, so that it's not re-expanded. + +set -- \ + "-Dorg.gradle.appname=$APP_BASE_NAME" \ + -classpath "$CLASSPATH" \ + org.gradle.wrapper.GradleWrapperMain \ + "$@" + +# Stop when "xargs" is not available. +if ! command -v xargs >/dev/null 2>&1 +then + die "xargs is not available" +fi + +# Use "xargs" to parse quoted args. +# +# With -n1 it outputs one arg per line, with the quotes and backslashes removed. +# +# In Bash we could simply go: +# +# readarray ARGS < <( xargs -n1 <<<"$var" ) && +# set -- "${ARGS[@]}" "$@" +# +# but POSIX shell has neither arrays nor command substitution, so instead we +# post-process each arg (as a line of input to sed) to backslash-escape any +# character that might be a shell metacharacter, then use eval to reverse +# that process (while maintaining the separation between arguments), and wrap +# the whole thing up as a single "set" statement. +# +# This will of course break if any of these variables contains a newline or +# an unmatched quote. +# + +eval "set -- $( + printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | + xargs -n1 | + sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | + tr '\n' ' ' + )" '"$@"' + +exec "$JAVACMD" "$@" diff --git a/gradlew.bat b/gradlew.bat new file mode 100755 index 0000000..6689b85 --- /dev/null +++ b/gradlew.bat @@ -0,0 +1,92 @@ +@rem +@rem Copyright 2015 the original author or authors. +@rem +@rem Licensed under the Apache License, Version 2.0 (the "License"); +@rem you may not use this file except in compliance with the License. +@rem You may obtain a copy of the License at +@rem +@rem https://www.apache.org/licenses/LICENSE-2.0 +@rem +@rem Unless required by applicable law or agreed to in writing, software +@rem distributed under the License is distributed on an "AS IS" BASIS, +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +@rem See the License for the specific language governing permissions and +@rem limitations under the License. +@rem + +@if "%DEBUG%"=="" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +set DIRNAME=%~dp0 +if "%DIRNAME%"=="" set DIRNAME=. +@rem This is normally unused +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Resolve any "." and ".." in APP_HOME to make it shorter. +for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if %ERRORLEVEL% equ 0 goto execute + +echo. +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto execute + +echo. +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* + +:end +@rem End local scope for the variables with windows NT shell +if %ERRORLEVEL% equ 0 goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +set EXIT_CODE=%ERRORLEVEL% +if %EXIT_CODE% equ 0 set EXIT_CODE=1 +if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% +exit /b %EXIT_CODE% + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/include/net_ioixd_blackbox_Native.h b/include/net_ioixd_blackbox_Native.h new file mode 100644 index 0000000..aedfe58 --- /dev/null +++ b/include/net_ioixd_blackbox_Native.h @@ -0,0 +1,69 @@ +/* DO NOT EDIT THIS FILE - it is machine generated */ +#include +/* Header for class net_ioixd_blackbox_Native */ + +#ifndef _Included_net_ioixd_blackbox_Native +#define _Included_net_ioixd_blackbox_Native +#ifdef __cplusplus +extern "C" { +#endif +/* + * Class: net_ioixd_blackbox_Native + * Method: loadPlugin + * Signature: (Ljava/lang/String;)Ljava/lang/String; + */ +JNIEXPORT jstring JNICALL Java_net_ioixd_blackbox_Native_loadPlugin + (JNIEnv *, jclass, jstring); + +/* + * Class: net_ioixd_blackbox_Native + * Method: enablePlugin + * Signature: (Ljava/lang/String;)V + */ +JNIEXPORT void JNICALL Java_net_ioixd_blackbox_Native_enablePlugin + (JNIEnv *, jclass, jstring); + +/* + * Class: net_ioixd_blackbox_Native + * Method: disablePlugin + * Signature: (Ljava/lang/String;)V + */ +JNIEXPORT void JNICALL Java_net_ioixd_blackbox_Native_disablePlugin + (JNIEnv *, jclass, jstring); + +/* + * Class: net_ioixd_blackbox_Native + * Method: libraryNames + * Signature: ()Ljava/lang/String; + */ +JNIEXPORT jstring JNICALL Java_net_ioixd_blackbox_Native_libraryNames + (JNIEnv *, jclass); + +/* + * Class: net_ioixd_blackbox_Native + * Method: libraryHasFunction + * Signature: (Ljava/lang/String;Ljava/lang/String;)Z + */ +JNIEXPORT jboolean JNICALL Java_net_ioixd_blackbox_Native_libraryHasFunction + (JNIEnv *, jclass, jstring, jstring); + +/* + * Class: net_ioixd_blackbox_Native + * Method: sendEvent + * Signature: (Ljava/lang/String;Ljava/lang/String;Ljava/lang/Object;)Z + */ +JNIEXPORT jboolean JNICALL Java_net_ioixd_blackbox_Native_sendEvent + (JNIEnv *, jclass, jstring, jstring, jobject); + +/* + * Class: net_ioixd_blackbox_Native + * Method: execute + * Signature: (Ljava/lang/String;Ljava/lang/String;Lorg/bukkit/plugin/Plugin;[Ljava/lang/Object;)Ljava/lang/Object; + */ +JNIEXPORT jobject JNICALL Java_net_ioixd_blackbox_Native_execute + (JNIEnv *, jclass, jstring, jstring, jobject, jobjectArray); + +#ifdef __cplusplus +} +#endif +#endif diff --git a/native/.gitignore b/native/.gitignore new file mode 100755 index 0000000..4fffb2f --- /dev/null +++ b/native/.gitignore @@ -0,0 +1,2 @@ +/target +/Cargo.lock diff --git a/native/.vscode/settings.json b/native/.vscode/settings.json new file mode 100644 index 0000000..0db5873 --- /dev/null +++ b/native/.vscode/settings.json @@ -0,0 +1,3 @@ +{ + "cmake.configureOnOpen": true +} \ No newline at end of file diff --git a/native/Cargo.toml b/native/Cargo.toml new file mode 100755 index 0000000..3ea7dc9 --- /dev/null +++ b/native/Cargo.toml @@ -0,0 +1,16 @@ +[package] +name = "native" +version = "0.1.0" +edition = "2021" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +jni = "0.21.1" +lazy_static = "1.4.0" +libloading = "0.8.0" +once_cell = "1.18.0" +parking_lot = "0.12.1" + +[lib] +crate_type = ["cdylib"] diff --git a/native/settings.gradle b/native/settings.gradle new file mode 100644 index 0000000..e69de29 diff --git a/native/src/lib.rs b/native/src/lib.rs new file mode 100755 index 0000000..ea66d9b --- /dev/null +++ b/native/src/lib.rs @@ -0,0 +1,362 @@ +struct SwitchableLibrary { + library: Mutex, + enabled: Mutex, +} + +impl SwitchableLibrary { + fn new(library: Library, enabled: bool) -> Self { + Self { + library: Mutex::new(library), + enabled: Mutex::new(enabled), + } + } + fn enabled(&self) -> bool { + self.enabled.lock().clone() + } + fn library(&self) -> MutexGuard { + self.library.lock() + } + fn set_enabled(&self, enabled: bool) { + *self.enabled.lock() = enabled; + } +} + +struct LibraryManager { + loaded_libraries: Mutex>, +} + +impl LibraryManager { + fn new() -> Self { + Self { + loaded_libraries: Mutex::new(HashMap::new()), + } + } + fn push_lib(&mut self, libname: String, lib: (Library, bool)) { + let libs = &mut self.loaded_libraries.lock(); + libs.insert(libname, SwitchableLibrary::new(lib.0, lib.1)); + } + fn set_enabled(&mut self, file: String, enabled: bool) { + let libs = &mut self.loaded_libraries.lock(); + libs.get(&file).unwrap().set_enabled(enabled); + } +} + +static mut LIBRARY_MANAGER: Lazy = Lazy::new(|| LibraryManager::new()); + +use jni::{ + objects::{JObject, JString}, + sys::jboolean, + JNIEnv, +}; +use libloading::Library; +use once_cell::sync::Lazy; +use parking_lot::{Mutex, MutexGuard}; + +use std::{collections::HashMap, error::Error, fmt::Display, fs, path::Path}; + +#[no_mangle] +pub extern "system" fn Java_net_ioixd_blackbox_Native_loadPlugin<'a>( + mut env: JNIEnv<'a>, + _obj: JObject, + filename: JString, +) { + let file = env + .get_string(&filename) + .unwrap() + .to_string_lossy() + .to_string(); + unsafe { + let lib = libloading::Library::new(&file); + if let Err(e) = &lib { + throw_libloading_error(&mut env, &file, e); + return; + } + let lib = lib.unwrap(); + LIBRARY_MANAGER.push_lib(file, (lib, true)); + } +} + +#[no_mangle] +pub extern "system" fn Java_net_ioixd_blackbox_Native_enablePlugin<'a>( + mut env: JNIEnv<'a>, + _obj: JObject, + filename: JString, +) { + let file = env + .get_string(&filename) + .unwrap() + .to_string_lossy() + .to_string(); + unsafe { + let map = LIBRARY_MANAGER + .loaded_libraries + .lock() + .get(&file) + .unwrap() + .set_enabled(true); + } +} + +#[no_mangle] +pub extern "system" fn Java_net_ioixd_blackbox_Native_disablePlugin<'a>( + mut env: JNIEnv<'a>, + _obj: JObject, + filename: JString, +) { + let file = env + .get_string(&filename) + .unwrap() + .to_string_lossy() + .to_string(); + unsafe { + let map = LIBRARY_MANAGER + .loaded_libraries + .lock() + .get(&file) + .unwrap() + .set_enabled(false); + } +} + +#[no_mangle] +pub extern "system" fn Java_net_ioixd_blackbox_Native_libraryNames<'a>( + env: JNIEnv<'a>, + _obj: JObject, +) -> JString<'a> { + unsafe { + env.new_string( + LIBRARY_MANAGER + .loaded_libraries + .lock() + .iter() + .map(|f| f.0.clone()) + .collect::>() + .join(","), + ) + .unwrap() + } +} + +#[no_mangle] +pub extern "system" fn Java_net_ioixd_blackbox_Native_libraryHasFunction<'a>( + mut env: JNIEnv<'a>, + _obj: JObject, + libname_raw: JString, + funcname_raw: JString, +) -> jboolean { + let libname = &env + .get_string(&libname_raw) + .unwrap() + .to_string_lossy() + .to_string(); + let funcname = &env + .get_string(&funcname_raw) + .unwrap() + .to_string_lossy() + .to_string(); + + let manager = unsafe { LIBRARY_MANAGER.loaded_libraries.lock() }; + let lib = manager.get(libname).unwrap().library(); + let func: Result< + libloading::Symbol, JObject<'_>)>, + libloading::Error, + > = unsafe { lib.get(funcname.as_bytes()) }; + if let Err(e) = &func { + match e { + libloading::Error::DlSym { desc: d } => { + return false.into(); + } + libloading::Error::GetProcAddress { source: s } => { + return false.into(); + } + _ => { + throw_libloading_error(&mut env, &libname, e); + return false.into(); + } + } + } + return true.into(); +} + +#[no_mangle] +pub extern "system" fn Java_net_ioixd_blackbox_Native_sendEvent<'a>( + mut env: JNIEnv<'a>, + _obj: JObject, + libname_raw: JString, + funcname_raw: JString, + ev: JObject, +) -> jboolean { + let libname_raw = env.get_string(&libname_raw); + let libname = unwrap_result_or_java_error(&mut env, "unknown library", libname_raw) + .to_string_lossy() + .to_string(); + let funcname_raw = env.get_string(&funcname_raw); + let funcname = unwrap_result_or_java_error(&mut env, &libname, funcname_raw) + .to_string_lossy() + .to_string(); + + let manager = unsafe { LIBRARY_MANAGER.loaded_libraries.lock() }; + let lib = unwrap_option_or_java_error( + &mut env, + &libname, + &"library".to_string(), + manager.get(&libname), + ) + .library(); + let func: Result< + libloading::Symbol, JObject<'_>)>, + libloading::Error, + > = unsafe { lib.get(funcname.as_bytes()) }; + if let Err(e) = &func { + match e { + libloading::Error::DlSym { desc: _ } => { + return false.into(); + } + libloading::Error::GetProcAddress { source: _ } => { + return false.into(); + } + _ => { + throw_libloading_error(&mut env, &libname, e); + return false.into(); + } + } + } + let func = func.unwrap(); + func(env, ev); + return true.into(); +} + +#[no_mangle] +pub extern "system" fn Java_net_ioixd_blackbox_Native_execute<'a>( + mut env: JNIEnv<'a>, + _obj: JObject, + libname_raw: JString, + funcname_raw: JString, + _plugin: JObject, + ev: JObject, +) -> JObject<'a> { + let libname_raw = env.get_string(&libname_raw); + let libname = unwrap_result_or_java_error(&mut env, "unknown library", libname_raw) + .to_string_lossy() + .to_string(); + let funcname_raw = env.get_string(&funcname_raw); + let funcname = unwrap_result_or_java_error(&mut env, &libname, funcname_raw) + .to_string_lossy() + .to_string(); + + let manager = unsafe { LIBRARY_MANAGER.loaded_libraries.lock() }; + let lib = + unwrap_option_or_java_error(&mut env, &libname, &libname, manager.get(&libname)).library(); + let func: Result< + libloading::Symbol, JObject<'_>) -> JObject<'a>>, + libloading::Error, + > = unsafe { lib.get(funcname.as_bytes()) }; + if let Err(e) = &func { + match e { + _ => { + throw_libloading_error(&mut env, &libname, e); + } + } + } + let func = func.unwrap(); + return func(env, ev); +} + +pub fn unwrap_result_or_java_error( + mut env: &mut JNIEnv, + libname: S, + opt: Result, +) -> V +where + S: Into + Display, + E: Error, +{ + match opt { + Ok(v) => v, + Err(err) => { + throw_with_error( + &mut env, + "net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException", + format!("error loading {}: {:?}", &libname, err), + ); + unreachable!(); + } + } +} + +pub fn unwrap_option_or_java_error( + mut env: &mut JNIEnv, + libname: S, + valname: S, + opt: Option, +) -> V +where + S: Into + Display, +{ + match opt { + Some(v) => v, + None => { + throw_with_error( + &mut env, + "net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException", + format!("error loading {}: {} is None", &libname, valname), + ); + unreachable!(); + } + } +} + +pub fn throw_libloading_error(mut env: &mut JNIEnv, libname: &String, e: &libloading::Error) { + let err_desc = e.to_string(); + match e { + libloading::Error::DlSym { desc } + | libloading::Error::DlOpen { desc } + | libloading::Error::DlClose { desc } => { + throw_with_error( + &mut env, + "net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException", + format!("error loading {}: {}. {:?}", &libname, err_desc, desc), + ); + } + libloading::Error::LoadLibraryExW { source } + | libloading::Error::GetModuleHandleExW { source } + | libloading::Error::FreeLibrary { source } + | libloading::Error::GetProcAddress { source } => { + throw_with_error( + &mut env, + "net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException", + format!("error loading {}: {}. {:?}", &libname, err_desc, source), + ); + } + libloading::Error::CreateCString { source } => { + throw_with_error( + &mut env, + "net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException", + format!("error loading {}: {}. {:?}", &libname, err_desc, source), + ); + } + libloading::Error::CreateCStringWithTrailing { source } => { + throw_with_error( + &mut env, + "net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException", + format!("error loading {}: {}. {:?}", &libname, err_desc, source), + ); + } + _ => { + throw_with_error( + &mut env, + "net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException", + format!("{}; no other information given", err_desc), + ); + } + } +} + +pub fn throw_with_error(env: &mut JNIEnv, exception_name: &str, exception_message: String) { + let er = env.throw((exception_name, exception_message)); + if let Err(_) = er { + env.exception_describe().unwrap(); + } + // TEMPORARY: just describe the error + env.exception_describe().unwrap(); +} diff --git a/settings.gradle b/settings.gradle new file mode 100755 index 0000000..b3f8cd6 --- /dev/null +++ b/settings.gradle @@ -0,0 +1 @@ +rootProject.name = 'BlackBox' diff --git a/src/main/java/net/ioixd/blackbox/BlackBox.java b/src/main/java/net/ioixd/blackbox/BlackBox.java new file mode 100755 index 0000000..c38ea6d --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/BlackBox.java @@ -0,0 +1,81 @@ +package net.ioixd.blackbox; + +import org.bukkit.event.Event; +import org.bukkit.event.EventPriority; +import org.bukkit.event.HandlerList; +import org.bukkit.event.Listener; +import org.bukkit.plugin.RegisteredListener; +import org.bukkit.plugin.java.JavaPlugin; + +import io.github.classgraph.ClassGraph; +import io.github.classgraph.ClassInfoList; + +import java.io.File; +import java.io.IOException; +import java.io.InputStream; +import java.nio.file.FileSystems; +import java.nio.file.Files; +import java.util.ArrayList; +import java.util.Arrays; + +public final class BlackBox extends JavaPlugin { + @Override + public void onEnable() { + // Make sure everything has been loaded. + ClassGraph clsgraph = new ClassGraph(); + clsgraph.acceptPackages("org.bukkit"); + clsgraph.initializeLoadedClasses(); + ClassInfoList info = clsgraph.scan().getSubclasses(Event.class.getName()); + info.forEach(cls -> { + cls.loadClass(); + }); + + // extract the .dll or .so from the .jar into the plugins folder and run it. + // i do not care if this is the wrong way. i have spent so fucking long trying + // to do it + // ""the proper way"" and nobody is gonna use this plugin so i don't give a fuck + String filename = ""; + String os = System.getProperty("os.name").toLowerCase(); + if (os.contains("win")) { + filename = "native.dll"; + } else if (os.contains("linux")) { + filename = "libnative.so"; + } else { + getLogger().severe("Unsupported operating system \"" + os + "\". Stopping now."); + getServer().shutdown(); + } + File dir = new File(getDataFolder().getAbsolutePath()); + if (!dir.exists()) { + dir.mkdir(); + } + File file = new File( + getDataFolder().getAbsolutePath() + + FileSystems.getDefault().getSeparator() + + filename); + if (file.exists()) { + file.delete(); + } + InputStream link = (BlackBox.class.getResourceAsStream("/" + filename)); + try { + Files.copy(link, file.getAbsoluteFile().toPath()); + } catch (IOException e) { + getLogger().severe(e.toString()); + getLogger().severe("Disabling self"); + setEnabled(false); + } + + System.load(file.getAbsolutePath()); + + this.getServer().getPluginManager().registerInterface(BlackBoxPluginLoader.class); + } + + @Override + public void onDisable() { + // Plugin shutdown logic + } + + public void disable() { + this.setEnabled(false); + } + +} diff --git a/src/main/java/net/ioixd/blackbox/BlackBoxPlugin.java b/src/main/java/net/ioixd/blackbox/BlackBoxPlugin.java new file mode 100644 index 0000000..4d3215d --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/BlackBoxPlugin.java @@ -0,0 +1,379 @@ +package net.ioixd.blackbox; + +import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.InputStreamReader; +import java.io.OutputStream; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.HashMap; +import java.util.List; +import java.util.logging.Level; +import java.util.logging.Logger; + +import org.bukkit.Server; +import org.bukkit.command.Command; +import org.bukkit.command.CommandSender; +import org.bukkit.configuration.file.FileConfiguration; +import org.bukkit.configuration.file.YamlConfiguration; +import org.bukkit.event.Event; +import org.bukkit.event.EventPriority; +import org.bukkit.event.HandlerList; +import org.bukkit.generator.BiomeProvider; +import org.bukkit.generator.ChunkGenerator; +import org.bukkit.plugin.InvalidDescriptionException; +import org.bukkit.plugin.Plugin; +import org.bukkit.plugin.PluginDescriptionFile; +import org.bukkit.plugin.PluginLoader; +import org.bukkit.event.Listener; +import org.bukkit.plugin.RegisteredListener; + +import com.google.common.base.Charsets; + +import io.papermc.paper.plugin.configuration.PluginMeta; + +public class BlackBoxPlugin implements Plugin { + String library; + BlackBox parent; + + PluginLoader loader; + + private boolean isEnabled = false; + private boolean naggable = true; + + private File file = null; + private PluginDescriptionFile description = null; + private File dataFolder = null; + private FileConfiguration newConfig = null; + private File configFile = null; + + private PluginMeta pluginMeta = null; + + RegisteredListener registeredListener = null; + + BlackBoxPlugin(String library, BlackBox parent, PluginLoader loader) throws IOException { + this.library = library; + this.isEnabled = true; + this.naggable = true; + this.parent = parent; + File pluginsFolder = parent.getDataFolder().getParentFile(); + if (!Paths.get(pluginsFolder.getPath(), library).toFile().exists()) { + Paths.get(pluginsFolder.getPath(), library).toFile().createNewFile(); + } + this.dataFolder = Paths.get(pluginsFolder.getPath(), library).toFile(); + this.loader = loader; + + this.pluginMeta = new BlackBoxPluginMeta(library); + } + + public void setRegisteredListener(RegisteredListener registeredListener) { + this.registeredListener = registeredListener; + } + + public void onEvent(Listener listener, Event event) { + // get the event name + String name = event.getEventName(); + String hookName = "__on__" + name; + + try { + boolean result = Native.libraryHasFunction(library, hookName); + if (result == false) { + HandlerList handler = event.getHandlers(); + handler.unregister(registeredListener); + } else { + Native.sendEvent(library, hookName, event); + } + } catch (Exception ex) { + this.getLogger().severe(ex.toString()); + } + } + + Object execNative(String functionName, Object[] objects) { + try { + boolean hasFunc = Native.libraryHasFunction(library, "getConfig"); + if (hasFunc) { + return Native.execute(this.library, "__getConfig", this, objects); + } else { + // default + return null; + } + } catch (Exception ex) { + this.getLogger().severe(ex.toString()); + // this.getLogger().severe("Disabling self"); + // this.disable(); + } + return null; + } + + public String getInnerLibraryName() { + return library; + } + + @Override + public String getName() { + return library; + } + + @Override + public FileConfiguration getConfig() { + Object func = execNative("getConfig", new Object[] {}); + if (func == null) { + if (newConfig == null) { + this.reloadConfig(); + } + return newConfig; + } else { + return (FileConfiguration) func; + } + } + + @Override + public BiomeProvider getDefaultBiomeProvider(String worldName, String id) { + Object func = execNative("getDefaultBiomeProvider", new Object[] { worldName, id }); + if (func == null) { + return parent.getDefaultBiomeProvider(worldName, id); + } else { + return (BiomeProvider) func; + } + } + + @Override + public ChunkGenerator getDefaultWorldGenerator(String worldName, String id) { + Object func = execNative("getDefaultWorldGenerator", new Object[] { worldName, id }); + if (func == null) { + return parent.getDefaultWorldGenerator(worldName, id); + } else { + return (ChunkGenerator) func; + } + } + + @Override + public Logger getLogger() { + Object func = execNative("getLogger", new Object[] {}); + if (func == null) { + return parent.getLogger(); + } else { + return (Logger) func; + } + } + + @Override + public InputStream getResource(String filename) { + Object func = execNative("getResource", new Object[] { filename }); + if (func == null) { + throw new UnsupportedOperationException( + "BlackBox plugins do not have embedded resources; the plugin author may choose to add an abstraction for them, but if you're seeing this then they don't. (\"Embedded resources\" refer to resources that would normally come from a .jar; they are not to be confused with anything that might embed a file into a library, you have to handle those yourself)."); + } else { + return (InputStream) func; + } + } + + @Override + public boolean onCommand(CommandSender sender, Command command, String label, String[] args) { + Object func = execNative("onCommand", new Object[] { sender, command, label, args }); + if (func == null) { + return false; + } else { + return (boolean) func; + } + } + + @Override + public void onDisable() { + // if it doesn't exist, nothing happens + execNative("onDisable", new Object[] {}); + } + + @Override + public void onEnable() { + // if it doesn't exist, nothing happens + execNative("onEnable", new Object[] {}); + } + + @Override + public void onLoad() { + // if it doesn't exist, nothing happens + execNative("onLoad", new Object[] {}); + } + + @Override + public List onTabComplete(CommandSender sender, Command command, String alias, String[] args) { + Object func = execNative("onTabComplete", new Object[] { sender, command, alias, args }); + if (func == null) { + return null; + } else { + return (List) func; + } + } + + @Override + public void reloadConfig() { + Object func = execNative("reloadConfig", new Object[] {}); + if (func == null) { + newConfig = YamlConfiguration.loadConfiguration(configFile); + + final InputStream defConfigStream = getResource("config.yml"); + if (defConfigStream == null) { + return; + } + + newConfig.setDefaults( + YamlConfiguration.loadConfiguration(new InputStreamReader(defConfigStream, Charsets.UTF_8))); + } + } + + @Override + public void saveConfig() { + Object func = execNative("saveConfig", new Object[] {}); + if (func == null) { + try { + getConfig().save(configFile); + } catch (IOException ex) { + this.getLogger().log(Level.SEVERE, "Could not save config to " + configFile, ex); + } + } + } + + @Override + public void saveDefaultConfig() { + Object func = execNative("saveDefaultConfig", new Object[] {}); + if (func == null) { + if (!configFile.exists()) { + saveResource("config.yml", false); + } + } + } + + @Override + public void saveResource(String resourcePath, boolean replace) { + Object func = execNative("saveResource", new Object[] {}); + if (func == null) { + if (resourcePath == null || resourcePath.equals("")) { + throw new IllegalArgumentException("ResourcePath cannot be null or empty"); + } + + resourcePath = resourcePath.replace('\\', '/'); + InputStream in = getResource(resourcePath); + if (in == null) { + throw new IllegalArgumentException( + "The embedded resource '" + resourcePath + "' cannot be found in " + file); + } + + File outFile = new File(dataFolder, resourcePath); + int lastIndex = resourcePath.lastIndexOf('/'); + File outDir = new File(dataFolder, resourcePath.substring(0, lastIndex >= 0 ? lastIndex : 0)); + + if (!outDir.exists()) { + outDir.mkdirs(); + } + + try { + if (!outFile.exists() || replace) { + OutputStream out = new FileOutputStream(outFile); + byte[] buf = new byte[1024]; + int len; + while ((len = in.read(buf)) > 0) { + out.write(buf, 0, len); + } + out.close(); + in.close(); + } else { + this.getLogger().log(Level.WARNING, "Could not save " + outFile.getName() + " to " + outFile + + " because " + outFile.getName() + " already exists."); + } + } catch (IOException ex) { + this.getLogger().log(Level.SEVERE, "Could not save " + outFile.getName() + " to " + outFile, ex); + } + } + } + + public PluginMeta getPluginMeta() { + return this.pluginMeta; + } + + @Override + public String toString() { + Object func = execNative("toString", new Object[] {}); + if (func == null) { + return description.getFullName(); + } else { + return (String) func; + } + } + + @Override + public File getDataFolder() { + Object func = execNative("getDataFolder", new Object[] {}); + if (func == null) { + return dataFolder; + } else { + return (File) func; + } + } + + @Override + public PluginDescriptionFile getDescription() { + Object func = execNative("getDescription", new Object[] {}); + if (func == null) { + try { + return loader.getPluginDescription(file); + } catch (InvalidDescriptionException e) { + e.printStackTrace(); + return null; + } + } else { + return (PluginDescriptionFile) func; + } + } + + @Override + public PluginLoader getPluginLoader() { + Object func = execNative("getPluginLoader", new Object[] {}); + if (func == null) { + return loader; + } else { + return (PluginLoader) func; + } + } + + @Override + public Server getServer() { + Object func = execNative("getServer", new Object[] {}); + if (func == null) { + return parent.getServer(); + } else { + return (Server) func; + } + } + + @Override + public boolean isEnabled() { + Object func = execNative("isEnabled", new Object[] {}); + if (func == null) { + return isEnabled; + } else { + return (boolean) func; + } + } + + @Override + public boolean isNaggable() { + Object func = execNative("isNaggable", new Object[] {}); + if (func == null) { + return naggable; + } else { + return (boolean) func; + } + } + + @Override + public void setNaggable(boolean canNag) { + Object func = execNative("setNaggable", new Object[] {}); + if (func == null) { + naggable = canNag; + } + } + +} diff --git a/src/main/java/net/ioixd/blackbox/BlackBoxPluginLoader.java b/src/main/java/net/ioixd/blackbox/BlackBoxPluginLoader.java new file mode 100644 index 0000000..e2deeaf --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/BlackBoxPluginLoader.java @@ -0,0 +1,178 @@ +package net.ioixd.blackbox; + +import com.google.common.base.Preconditions; +import com.google.common.base.Strings; + +import java.io.File; +import java.io.IOException; +import java.lang.reflect.Method; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Set; +import java.util.regex.Pattern; +import org.bukkit.Server; +import org.bukkit.event.Event; +import org.bukkit.event.EventHandler; +import org.bukkit.event.Listener; +import org.bukkit.plugin.InvalidDescriptionException; +import org.bukkit.plugin.InvalidPluginException; +import org.bukkit.plugin.Plugin; +import org.bukkit.plugin.PluginDescriptionFile; +import org.bukkit.plugin.PluginLoader; +import org.bukkit.plugin.RegisteredListener; +import org.bukkit.plugin.TimedRegisteredListener; +import org.jetbrains.annotations.NotNull; + +/** + * Represents a BlackBox plugin loader, allowing plugins in the form of .dll and + * .so + */ +public final class BlackBoxPluginLoader implements PluginLoader { + final Server server; + private HashMap libNameMap = new HashMap<>(); + private final Pattern[] windowsFileFilters = new Pattern[] { Pattern.compile("\\.dll$") }; + private final Pattern[] macFileFilters = new Pattern[] { Pattern.compile("\\.dylib$") }; + private final Pattern[] linuxFileFilters = new Pattern[] { Pattern.compile("\\.so$") }; + + public BlackBoxPluginLoader(@NotNull Server instance) { + Preconditions.checkArgument(instance != null, "Server cannot be null"); + server = instance; + } + + @Override + @NotNull + public Plugin loadPlugin(@NotNull final File file) throws InvalidPluginException { + server.getLogger().info("Loading " + file.getName()); + Native.loadPlugin(file.getName()); + BlackBoxPlugin plugin; + try { + plugin = new BlackBoxPlugin(file.getName(), + (BlackBox) server.getPluginManager().getPlugin("BlackBox"), this); + } catch (IOException e) { + server.getLogger().severe(e.getStackTrace().toString()); + return null; + } + libNameMap.put(plugin, plugin); + return plugin; + } + + @Override + public void enablePlugin(@NotNull final Plugin plugin) { + if (libNameMap.containsKey(plugin)) { + Native.enablePlugin(libNameMap.get(plugin).getInnerLibraryName()); + } else { + server.getLogger().severe("BlackBoxPluginLoader tried to enable plugin that isn't registered in it."); + } + } + + @Override + public void disablePlugin(@NotNull Plugin plugin) { + if (libNameMap.containsKey(plugin)) { + Native.disablePlugin(libNameMap.get(plugin).getInnerLibraryName()); + } else { + server.getLogger().severe("BlackBoxPluginLoader tried to disable plugin that isn't registered in it."); + } + } + + @Override + @NotNull + public PluginDescriptionFile getPluginDescription(@NotNull File file) throws InvalidDescriptionException { + /* + * GAME PLAN + * library is expected to export some c types under certain names. + */ + return null; + } + + @Override + @NotNull + public Pattern[] getPluginFileFilters() { + String os = System.getProperty("os.name").toLowerCase(); + server.getLogger().info("OS is " + os); + if (os.contains("win")) { + return windowsFileFilters; + } + if (os.contains("mac")) { + return macFileFilters; + } + if (os.contains("linux") || os.contains("nix")) { + return linuxFileFilters; + } + throw new UnsupportedOperationException("Unknown OS \"" + os + "\""); + } + + @Override + @NotNull + public Map, Set> createRegisteredListeners(@NotNull Listener listener, + @NotNull final Plugin plugin) { + Preconditions.checkArgument(plugin != null, "Plugin can not be null"); + Preconditions.checkArgument(listener != null, "Listener can not be null"); + + boolean useTimings = server.getPluginManager().useTimings(); + + Map, Set> ret = new HashMap, Set>(); + + Set methods; + try { + Method[] publicMethods = listener.getClass().getMethods(); + Method[] privateMethods = listener.getClass().getDeclaredMethods(); + methods = new HashSet(publicMethods.length + privateMethods.length, 1.0f); + for (Method method : publicMethods) { + methods.add(method); + } + for (Method method : privateMethods) { + methods.add(method); + } + } catch (NoClassDefFoundError e) { + plugin.getLogger() + .severe("Plugin " + plugin.getDescription().getFullName() + " has failed to register events for " + + listener.getClass() + " because " + e.getMessage() + " does not exist."); + return ret; + } + + for (final Method method : methods) { + final EventHandler eh = method.getAnnotation(EventHandler.class); + if (eh == null) + continue; + // Do not register bridge or synthetic methods to avoid event duplication + // Fixes SPIGOT-893 + if (method.isBridge() || method.isSynthetic()) { + continue; + } + final Class checkClass; + if (method.getParameterTypes().length != 1 + || !Event.class.isAssignableFrom(checkClass = method.getParameterTypes()[0])) { + plugin.getLogger() + .severe(plugin.getDescription().getFullName() + + " attempted to register an invalid EventHandler method signature \"" + + method.toGenericString() + "\" in " + listener.getClass()); + continue; + } + final Class eventClass = checkClass.asSubclass(Event.class); + method.setAccessible(true); + Set eventSet = ret.get(eventClass); + if (eventSet == null) { + eventSet = new HashSet(); + ret.put(eventClass, eventSet); + } + + if (libNameMap.get(plugin) == null) { + server.getLogger().severe( + "BlackBoxPluginLoader tried to create listeners for a plugin that isn't registered in it."); + continue; + } + ; + if (useTimings) { + eventSet.add( + new TimedRegisteredListener(listener, libNameMap.get(plugin)::onEvent, eh.priority(), plugin, + eh.ignoreCancelled())); + } else { + eventSet.add(new RegisteredListener(listener, libNameMap.get(plugin)::onEvent, eh.priority(), plugin, + eh.ignoreCancelled())); + } + } + return ret; + + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/BlackBoxPluginMeta.java b/src/main/java/net/ioixd/blackbox/BlackBoxPluginMeta.java new file mode 100644 index 0000000..5b8a836 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/BlackBoxPluginMeta.java @@ -0,0 +1,101 @@ +package net.ioixd.blackbox; + +import java.util.ArrayList; +import java.util.List; + +import org.bukkit.permissions.Permission; +import org.bukkit.permissions.PermissionDefault; +import org.bukkit.plugin.PluginLoadOrder; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import io.papermc.paper.plugin.configuration.PluginMeta; + +public class BlackBoxPluginMeta implements PluginMeta { + String library; + + BlackBoxPluginMeta(String library) { + this.library = library; + } + + @Override + public @Nullable String getAPIVersion() { + return "0.0.0"; + } + + @Override + public @NotNull List getAuthors() { + return new ArrayList(); + } + + @Override + public @NotNull List getContributors() { + return new ArrayList(); + } + + @Override + public @Nullable String getDescription() { + return "A native library loaded by BlackBox. No other information is avaliable."; + } + + @Override + public @NotNull List getLoadBeforePlugins() { + return new ArrayList(); + } + + @Override + public @NotNull PluginLoadOrder getLoadOrder() { + return PluginLoadOrder.STARTUP; + } + + @Override + public @Nullable String getLoggerPrefix() { + return library; + } + + @Override + public @NotNull String getMainClass() { + return null; + } + + @Override + public @NotNull String getName() { + return library; + } + + @Override + public @NotNull PermissionDefault getPermissionDefault() { + return null; + } + + @Override + public @NotNull List getPermissions() { + return new ArrayList(); + } + + @Override + public @NotNull List getPluginDependencies() { + return new ArrayList(); + } + + @Override + public @NotNull List getPluginSoftDependencies() { + return new ArrayList(); + } + + @Override + public @NotNull List getProvidedPlugins() { + return new ArrayList(); + } + + @Override + public @NotNull String getVersion() { + return "0.0.0"; + } + + @Override + public @Nullable String getWebsite() { + return "https://lemmykoopa.com"; + } + +} diff --git a/src/main/java/net/ioixd/blackbox/Native.java b/src/main/java/net/ioixd/blackbox/Native.java new file mode 100755 index 0000000..f8b3a55 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/Native.java @@ -0,0 +1,14 @@ +package net.ioixd.blackbox; + +import org.bukkit.plugin.Plugin; + +public class Native { + public static native String loadPlugin(String filename); + public static native void enablePlugin(String libname); + public static native void disablePlugin(String libname); + + public static native String libraryNames(); + public static native boolean libraryHasFunction(String libName, String functionName); + public static native boolean sendEvent(String libName, String functionName, Object object); + public static native Object execute(String libName, String functionName, Plugin plugin, Object[] objects) throws Exception; +} diff --git a/src/main/java/net/ioixd/blackbox/exceptions/MissingFunctionException.java b/src/main/java/net/ioixd/blackbox/exceptions/MissingFunctionException.java new file mode 100644 index 0000000..51cd2a6 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/exceptions/MissingFunctionException.java @@ -0,0 +1,10 @@ +package net.ioixd.blackbox.exceptions; + +public class MissingFunctionException extends RuntimeException { + public MissingFunctionException(String errorMessage) { + super(errorMessage); + } + public MissingFunctionException(String errorMessage, Throwable err) { + super(errorMessage, err); + } +} diff --git a/src/main/java/net/ioixd/blackbox/exceptions/NativeLibraryLoadException.java b/src/main/java/net/ioixd/blackbox/exceptions/NativeLibraryLoadException.java new file mode 100644 index 0000000..89d58df --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/exceptions/NativeLibraryLoadException.java @@ -0,0 +1,10 @@ +package net.ioixd.blackbox.exceptions; + +public class NativeLibraryLoadException extends RuntimeException { + public NativeLibraryLoadException(String errorMessage) { + super(errorMessage); + } + public NativeLibraryLoadException(String errorMessage, Throwable err) { + super(errorMessage, err); + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException.java b/src/main/java/net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException.java new file mode 100644 index 0000000..cb3f080 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/exceptions/NativeLibrarySymbolLoadException.java @@ -0,0 +1,10 @@ +package net.ioixd.blackbox.exceptions; + +public class NativeLibrarySymbolLoadException extends RuntimeException { + public NativeLibrarySymbolLoadException(String errorMessage) { + super(errorMessage); + } + public NativeLibrarySymbolLoadException(String errorMessage, Throwable err) { + super(errorMessage, err); + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableBiomeProvider.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableBiomeProvider.java new file mode 100644 index 0000000..44df599 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableBiomeProvider.java @@ -0,0 +1,49 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; + +import org.bukkit.generator.BiomeProvider; +import org.bukkit.plugin.Plugin; + +import net.ioixd.blackbox.exceptions.MissingFunctionException; + + +public class ExtendableBiomeProvider extends BiomeProvider { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableBiomeProvider(Plugin blackBox, String name, String inLibName) throws MissingFunctionException { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public org.bukkit.block.Biome getBiome(org.bukkit.generator.WorldInfo arg0, int arg1, int arg2, int arg3) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__BiomeProvider__"+this.name+"__getBiome", this.blackBox, new Object[] { + arg0, arg1, arg2, arg3 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.block.Biome) result; + } + public java.util.List getBiomes(org.bukkit.generator.WorldInfo arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__BiomeProvider__"+this.name+"__getBiomes", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableBukkitRunnable.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableBukkitRunnable.java new file mode 100644 index 0000000..172293a --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableBukkitRunnable.java @@ -0,0 +1,28 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; + +import org.bukkit.plugin.Plugin; +import org.bukkit.scheduler.BukkitRunnable; +public class ExtendableBukkitRunnable extends BukkitRunnable { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableBukkitRunnable(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public void run() { + try { + Native.execute(this.inLibName, "__extends__BukkitRunnable__"+this.name+"__run", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableCommandExecutor.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableCommandExecutor.java new file mode 100644 index 0000000..7282a39 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableCommandExecutor.java @@ -0,0 +1,31 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.command.CommandExecutor; +import org.bukkit.plugin.Plugin; +public class ExtendableCommandExecutor implements CommandExecutor { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableCommandExecutor(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public boolean onCommand(org.bukkit.command.CommandSender arg0, org.bukkit.command.Command arg1, java.lang.String arg2, java.lang.String[] arg3) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__CommandExecutor__"+this.name+"__onCommand", this.blackBox, new Object[] { + arg0, arg1, arg2, arg3 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableConfigurationSerializable.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableConfigurationSerializable.java new file mode 100644 index 0000000..fc1553f --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableConfigurationSerializable.java @@ -0,0 +1,29 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.configuration.serialization.ConfigurationSerializable; +import org.bukkit.plugin.Plugin; +public class ExtendableConfigurationSerializable implements ConfigurationSerializable { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableConfigurationSerializable(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public java.util.Map serialize() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__ConfigurationSerializable__"+this.name+"__serialize", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.Map) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableConsumer.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableConsumer.java new file mode 100644 index 0000000..24beb1f --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableConsumer.java @@ -0,0 +1,30 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; + +import org.bukkit.plugin.Plugin; +import org.bukkit.util.Consumer; +public class ExtendableConsumer implements Consumer { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableConsumer(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public void accept(java.lang.Object arg0) { + try { + Native.execute(this.inLibName, "__extends__Consumer__"+this.name+"__accept", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableConversationCanceller.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableConversationCanceller.java new file mode 100644 index 0000000..dae4e00 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableConversationCanceller.java @@ -0,0 +1,53 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.conversations.ConversationCanceller; +import org.bukkit.plugin.Plugin; +public class ExtendableConversationCanceller implements ConversationCanceller { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableConversationCanceller(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public void setConversation(org.bukkit.conversations.Conversation arg0) { + try { + Native.execute(this.inLibName, "__extends__ConversationCanceller__"+this.name+"__setConversation", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public boolean cancelBasedOnInput(org.bukkit.conversations.ConversationContext arg0, java.lang.String arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__ConversationCanceller__"+this.name+"__cancelBasedOnInput", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public org.bukkit.conversations.ConversationCanceller clone() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__ConversationCanceller__"+this.name+"__clone", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.conversations.ConversationCanceller) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableConversationPrefix.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableConversationPrefix.java new file mode 100644 index 0000000..f11005d --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableConversationPrefix.java @@ -0,0 +1,32 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.conversations.ConversationPrefix; +import org.bukkit.plugin.Plugin; +public class ExtendableConversationPrefix implements ConversationPrefix { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableConversationPrefix(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + + public java.lang.String getPrefix(org.bukkit.conversations.ConversationContext arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__ConversationPrefix__"+this.name+"__getPrefix", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.String) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableFileConfiguration.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableFileConfiguration.java new file mode 100644 index 0000000..81cb5a0 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableFileConfiguration.java @@ -0,0 +1,40 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.configuration.file.FileConfiguration; +import org.bukkit.plugin.Plugin; +public class ExtendableFileConfiguration extends FileConfiguration { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableFileConfiguration(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public java.lang.String saveToString() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__FileConfiguration__"+this.name+"__saveToString", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.String) result; + } + public void loadFromString(java.lang.String arg0) { + try { + Native.execute(this.inLibName, "__extends__FileConfiguration__"+this.name+"__loadFromString", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableHelpTopic.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableHelpTopic.java new file mode 100644 index 0000000..b2e916b --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableHelpTopic.java @@ -0,0 +1,31 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.help.HelpTopic; +import org.bukkit.plugin.Plugin; +public class ExtendableHelpTopic extends HelpTopic { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableHelpTopic(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public boolean canSee(org.bukkit.command.CommandSender arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__HelpTopic__"+this.name+"__canSee", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableHelpTopicFactory.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableHelpTopicFactory.java new file mode 100644 index 0000000..e0d37c8 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableHelpTopicFactory.java @@ -0,0 +1,31 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.help.HelpTopicFactory; +import org.bukkit.plugin.Plugin; +public class ExtendableHelpTopicFactory implements HelpTopicFactory { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableHelpTopicFactory(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public org.bukkit.help.HelpTopic createTopic(org.bukkit.command.Command arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__HelpTopicFactory__"+this.name+"__createTopic", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.help.HelpTopic) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableMapRenderer.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableMapRenderer.java new file mode 100644 index 0000000..ce4258f --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableMapRenderer.java @@ -0,0 +1,29 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.map.MapRenderer; +import org.bukkit.plugin.Plugin; +public class ExtendableMapRenderer extends MapRenderer { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableMapRenderer(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public void render(org.bukkit.map.MapView arg0, org.bukkit.map.MapCanvas arg1, org.bukkit.entity.Player arg2) { + try { + Native.execute(this.inLibName, "__extends__MapRenderer__"+this.name+"__render", this.blackBox, new Object[] { + arg0, arg1, arg2 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableMetadataValue.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableMetadataValue.java new file mode 100644 index 0000000..61fb7de --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableMetadataValue.java @@ -0,0 +1,137 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.metadata.MetadataValue; +import org.bukkit.plugin.Plugin; +public class ExtendableMetadataValue implements MetadataValue { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableMetadataValue(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public java.lang.String asString() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__asString", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.String) result; + } + public int asInt() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__asInt", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (int) result; + } + public org.bukkit.plugin.Plugin getOwningPlugin() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__getOwningPlugin", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.plugin.Plugin) result; + } + public void invalidate() { + try { + Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__invalidate", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public float asFloat() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__asFloat", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (float) result; + } + public double asDouble() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__asDouble", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (double) result; + } + public long asLong() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__asLong", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (long) result; + } + public short asShort() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__asShort", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (short) result; + } + public byte asByte() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__asByte", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (byte) result; + } + public boolean asBoolean() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__asBoolean", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public java.lang.Object value() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__MetadataValue__"+this.name+"__value", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.Object) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableNoiseGenerator.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableNoiseGenerator.java new file mode 100644 index 0000000..69c683d --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableNoiseGenerator.java @@ -0,0 +1,32 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; + +import org.bukkit.plugin.Plugin; +import org.bukkit.util.noise.NoiseGenerator; +public class ExtendableNoiseGenerator extends NoiseGenerator { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableNoiseGenerator(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public double noise(double arg0, double arg1, double arg2) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__NoiseGenerator__"+this.name+"__noise", this.blackBox, new Object[] { + arg0, arg1, arg2 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (double) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendablePersistentDataType.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendablePersistentDataType.java new file mode 100644 index 0000000..f36b349 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendablePersistentDataType.java @@ -0,0 +1,66 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.persistence.PersistentDataType; +import org.bukkit.plugin.Plugin; +public class ExtendablePersistentDataType implements PersistentDataType { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendablePersistentDataType(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public java.lang.Class getPrimitiveType() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PersistentDataType__"+this.name+"__getPrimitiveType", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.Class) result; + } + public java.lang.Class getComplexType() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PersistentDataType__"+this.name+"__getComplexType", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.Class) result; + } + public java.lang.Object toPrimitive(java.lang.Object arg0, org.bukkit.persistence.PersistentDataAdapterContext arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PersistentDataType__"+this.name+"__toPrimitive", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.Object) result; + } + public java.lang.Object fromPrimitive(java.lang.Object arg0, org.bukkit.persistence.PersistentDataAdapterContext arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PersistentDataType__"+this.name+"__fromPrimitive", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.Object) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendablePlugin.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendablePlugin.java new file mode 100644 index 0000000..6ef9fbf --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendablePlugin.java @@ -0,0 +1,259 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.plugin.Plugin; +public class ExtendablePlugin implements Plugin { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendablePlugin(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public org.bukkit.Server getServer() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__getServer", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.Server) result; + } + public org.bukkit.plugin.PluginDescriptionFile getDescription() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__getDescription", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.plugin.PluginDescriptionFile) result; + } + public org.bukkit.generator.BiomeProvider getDefaultBiomeProvider(java.lang.String arg0, java.lang.String arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__getDefaultBiomeProvider", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.generator.BiomeProvider) result; + } + public boolean isEnabled() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__isEnabled", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public org.bukkit.configuration.file.FileConfiguration getConfig() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__getConfig", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.file.FileConfiguration) result; + } + public java.io.File getDataFolder() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__getDataFolder", this.blackBox, new Object[] { + + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.io.File) result; + } + public void saveConfig() { + try { + Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__saveConfig", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void saveDefaultConfig() { + try { + Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__saveDefaultConfig", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void saveResource(java.lang.String arg0, boolean arg1) { + try { + Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__saveResource", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void reloadConfig() { + try { + Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__reloadConfig", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public org.bukkit.plugin.PluginLoader getPluginLoader() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__getPluginLoader", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.plugin.PluginLoader) result; + } + public void onDisable() { + try { + Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__onDisable", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void onLoad() { + try { + Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__onLoad", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void onEnable() { + try { + Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__onEnable", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public boolean isNaggable() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__isNaggable", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public void setNaggable(boolean arg0) { + try { + Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__setNaggable", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public org.bukkit.generator.ChunkGenerator getDefaultWorldGenerator(java.lang.String arg0, java.lang.String arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__getDefaultWorldGenerator", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.generator.ChunkGenerator) result; + } + public java.lang.String getName() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__getName", this.blackBox, new Object[] { }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.String) result; + } + public java.io.InputStream getResource(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__getResource", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.io.InputStream) result; + } + public java.util.logging.Logger getLogger() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__getLogger", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.logging.Logger) result; + } + public java.util.List onTabComplete(org.bukkit.command.CommandSender arg0, org.bukkit.command.Command arg1, java.lang.String arg2, java.lang.String[] arg3) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__onTabComplete", this.blackBox, new Object[] { + arg0, arg1, arg2, arg3 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public boolean onCommand(org.bukkit.command.CommandSender arg0, org.bukkit.command.Command arg1, java.lang.String arg2, java.lang.String[] arg3) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Plugin__"+this.name+"__onCommand", this.blackBox, new Object[] { + arg0, arg1, arg2, arg3 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendablePluginBase.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendablePluginBase.java new file mode 100644 index 0000000..a7354c4 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendablePluginBase.java @@ -0,0 +1,257 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; + +import org.bukkit.plugin.Plugin; +import org.bukkit.plugin.PluginBase; +public class ExtendablePluginBase extends PluginBase { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendablePluginBase(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + + public org.bukkit.Server getServer() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__getServer", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.Server) result; + } + public org.bukkit.plugin.PluginDescriptionFile getDescription() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__getDescription", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.plugin.PluginDescriptionFile) result; + } + public org.bukkit.generator.BiomeProvider getDefaultBiomeProvider(java.lang.String arg0, java.lang.String arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__getDefaultBiomeProvider", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.generator.BiomeProvider) result; + } + public boolean isEnabled() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__isEnabled", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public org.bukkit.configuration.file.FileConfiguration getConfig() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__getConfig", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.file.FileConfiguration) result; + } + public java.io.File getDataFolder() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__getDataFolder", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.io.File) result; + } + public void saveConfig() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__saveConfig", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void saveDefaultConfig() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__saveDefaultConfig", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void saveResource(java.lang.String arg0, boolean arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__saveResource", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void reloadConfig() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__reloadConfig", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public org.bukkit.plugin.PluginLoader getPluginLoader() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__getPluginLoader", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.plugin.PluginLoader) result; + } + public void onDisable() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__onDisable", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void onLoad() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__onLoad", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void onEnable() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__onEnable", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public boolean isNaggable() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__isNaggable", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public void setNaggable(boolean arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__setNaggable", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public org.bukkit.generator.ChunkGenerator getDefaultWorldGenerator(java.lang.String arg0, java.lang.String arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__getDefaultWorldGenerator", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.generator.ChunkGenerator) result; + } + public java.io.InputStream getResource(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__getResource", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.io.InputStream) result; + } + public java.util.logging.Logger getLogger() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__getLogger", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.logging.Logger) result; + } + public java.util.List onTabComplete(org.bukkit.command.CommandSender arg0, org.bukkit.command.Command arg1, java.lang.String arg2, java.lang.String[] arg3) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__onTabComplete", this.blackBox, new Object[] { + arg0, arg1, arg2, arg3 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public boolean onCommand(org.bukkit.command.CommandSender arg0, org.bukkit.command.Command arg1, java.lang.String arg2, java.lang.String[] arg3) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginBase__"+this.name+"__onCommand", this.blackBox, new Object[] { + arg0, arg1, arg2, arg3 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendablePluginLoader.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendablePluginLoader.java new file mode 100644 index 0000000..219a84f --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendablePluginLoader.java @@ -0,0 +1,93 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; + +import org.bukkit.plugin.Plugin; +import org.bukkit.plugin.PluginLoader; +public class ExtendablePluginLoader implements PluginLoader { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendablePluginLoader(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public org.bukkit.plugin.Plugin loadPlugin(java.io.File arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginLoader__"+this.name+"__loadPlugin", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.plugin.Plugin) result; + } + public org.bukkit.plugin.PluginDescriptionFile getPluginDescription(java.io.File arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginLoader__"+this.name+"__getPluginDescription", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.plugin.PluginDescriptionFile) result; + } + public java.util.regex.Pattern[] getPluginFileFilters() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginLoader__"+this.name+"__getPluginFileFilters", this.blackBox, new Object[] {}); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.regex.Pattern[]) result; + } + public java.util.Map, java.util.Set> createRegisteredListeners(org.bukkit.event.Listener arg0, org.bukkit.plugin.Plugin arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginLoader__"+this.name+"__createRegisteredListeners", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.Map, java.util.Set>) result; + } + public void enablePlugin(org.bukkit.plugin.Plugin arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginLoader__"+this.name+"__enablePlugin", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void disablePlugin(org.bukkit.plugin.Plugin arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__PluginLoader__"+this.name+"__disablePlugin", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendablePrompt.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendablePrompt.java new file mode 100644 index 0000000..6a8662e --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendablePrompt.java @@ -0,0 +1,57 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.conversations.Prompt; +import org.bukkit.plugin.Plugin; +public class ExtendablePrompt implements Prompt { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendablePrompt(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public java.lang.String getPromptText(org.bukkit.conversations.ConversationContext arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Prompt__"+this.name+"__getPromptText", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.String) result; + } + public boolean blocksForInput(org.bukkit.conversations.ConversationContext arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Prompt__"+this.name+"__blocksForInput", this.blackBox, new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public org.bukkit.conversations.Prompt acceptInput(org.bukkit.conversations.ConversationContext arg0, java.lang.String arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Prompt__"+this.name+"__acceptInput", this.blackBox, new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.conversations.Prompt) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableTabCompleter.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableTabCompleter.java new file mode 100644 index 0000000..5ea3151 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableTabCompleter.java @@ -0,0 +1,31 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.command.TabCompleter; +import org.bukkit.plugin.Plugin; +public class ExtendableTabCompleter implements TabCompleter { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableTabCompleter(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public java.util.List onTabComplete(org.bukkit.command.CommandSender arg0, org.bukkit.command.Command arg1, java.lang.String arg2, java.lang.String[] arg3) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__TabCompleter__"+this.name+"__onTabComplete", this.blackBox, new Object[] { + arg0, arg1, arg2, arg3 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/ExtendableTabExecutor.java b/src/main/java/net/ioixd/blackbox/extendables/ExtendableTabExecutor.java new file mode 100644 index 0000000..eb13a5f --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/ExtendableTabExecutor.java @@ -0,0 +1,44 @@ +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.command.TabExecutor; +import org.bukkit.plugin.Plugin; +public class ExtendableTabExecutor implements TabExecutor { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableTabExecutor(Plugin blackBox, String name, String inLibName) { + this.blackBox = (BlackBox) blackBox; + this.name = name; + this.inLibName = inLibName; + Misc.throwIfFuncsNotBound(this.inLibName, this.name, this.getClass()); + } + + public java.util.List onTabComplete(org.bukkit.command.CommandSender arg0, org.bukkit.command.Command arg1, java.lang.String arg2, java.lang.String[] arg3) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__TabExecutor__"+this.name+"__onTabComplete", this.blackBox, new Object[] { + arg0, arg1, arg2, arg3 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public boolean onCommand(org.bukkit.command.CommandSender arg0, org.bukkit.command.Command arg1, java.lang.String arg2, java.lang.String[] arg3) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__TabExecutor__"+this.name+"__onCommand", this.blackBox, new Object[] { + arg0, arg1, arg2, arg3 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } +} \ No newline at end of file diff --git a/src/main/java/net/ioixd/blackbox/extendables/Misc.java b/src/main/java/net/ioixd/blackbox/extendables/Misc.java new file mode 100644 index 0000000..d1dbfa0 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/Misc.java @@ -0,0 +1,23 @@ +package net.ioixd.blackbox.extendables; + +import java.lang.reflect.Method; +import java.lang.reflect.Modifier; + +import net.ioixd.blackbox.Native; + +import net.ioixd.blackbox.exceptions.MissingFunctionException; + +public class Misc { + public static void throwIfFuncsNotBound(String libName, String className, Class extendsClass) throws MissingFunctionException { + String extendsName = extendsClass.getName().replace("Extendable",""); + Method[] methods = extendsClass.getMethods(); + for (Method method : methods) { + if((method.getModifiers()&Modifier.ABSTRACT) != Modifier.ABSTRACT) { + continue; + } + if (!Native.libraryHasFunction(libName, "__extends__"+extendsName+"__"+className+"__"+method.getName())) { + throw new MissingFunctionException(libName+" does not have a function titled __extends__"+extendsName+"__"+className+"__"+method.getName()); + } + } + } +} diff --git a/src/main/java/net/ioixd/blackbox/extendables/__skipped__ExtendableConfiguration b/src/main/java/net/ioixd/blackbox/extendables/__skipped__ExtendableConfiguration new file mode 100644 index 0000000..2efce26 --- /dev/null +++ b/src/main/java/net/ioixd/blackbox/extendables/__skipped__ExtendableConfiguration @@ -0,0 +1,981 @@ +/* + Configuration utilizes operator overloading and it's unclear how this should be taken +*/ + +package net.ioixd.blackbox.extendables; +import net.ioixd.blackbox.BlackBox; +import net.ioixd.blackbox.Native; +import org.bukkit.configuration.Configuration; +import org.bukkit.configuration.serialization.ConfigurationSerializable; +public class ExtendableConfiguration implements Configuration { + public String name; + public String inLibName; + public BlackBox blackBox; + + ExtendableConfiguration(BlackBox blackBox, String name, String inLibName) { + this.blackBox = blackBox; + this.name = name; + this.inLibName = inLibName; + } + + void tryExecute(String funcName, Object[] objects) { + try { + Native.execute(this.inLibName, funcName, objects); + } catch(Exception ex) { + this.blackBox.getLogger().severe(ex.toString()); + this.blackBox.getLogger().severe("Disabling self"); + this.blackBox.disable(); + } + } + public void addDefault(java.lang.String arg0, java.lang.Object arg1) { + try { + Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__addDefault", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public org.bukkit.configuration.Configuration getDefaults() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getDefaults", new Object[] { + + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.Configuration) result; + } + public void addDefaults(java.util.Map arg0) { + try { + Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__addDefaults", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void addDefaults(org.bukkit.configuration.Configuration arg0) { + try { + Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__addDefaults", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void setDefaults(org.bukkit.configuration.Configuration arg0) { + try { + Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__setDefaults", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public org.bukkit.configuration.ConfigurationOptions options() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__options", new Object[] { + + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.ConfigurationOptions) result; + } + public java.lang.String getString(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getString", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.String) result; + } + public java.lang.String getString(java.lang.String arg0, java.lang.String arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getString", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.String) result; + } + public java.util.Set getKeys(boolean arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getKeys", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.Set) result; + } + public org.bukkit.Color getColor(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getColor", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.Color) result; + } + public org.bukkit.Color getColor(java.lang.String arg0, org.bukkit.Color arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getColor", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.Color) result; + } + public org.bukkit.inventory.ItemStack getItemStack(java.lang.String arg0, org.bukkit.inventory.ItemStack arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getItemStack", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.inventory.ItemStack) result; + } + public org.bukkit.inventory.ItemStack getItemStack(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getItemStack", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.inventory.ItemStack) result; + } + public org.bukkit.OfflinePlayer getOfflinePlayer(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getOfflinePlayer", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.OfflinePlayer) result; + } + public org.bukkit.OfflinePlayer getOfflinePlayer(java.lang.String arg0, org.bukkit.OfflinePlayer arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getOfflinePlayer", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.OfflinePlayer) result; + } + public java.util.Map getValues(boolean arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getValues", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.Map) result; + } + public org.bukkit.configuration.ConfigurationSection getDefaultSection() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getDefaultSection", new Object[] { + + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.ConfigurationSection) result; + } + public java.lang.String getCurrentPath() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getCurrentPath", new Object[] { + + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.String) result; + } + public boolean isConfigurationSection(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isConfigurationSection", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public org.bukkit.configuration.ConfigurationSection getConfigurationSection(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getConfigurationSection", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.ConfigurationSection) result; + } + public org.bukkit.configuration.ConfigurationSection createSection(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__createSection", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.ConfigurationSection) result; + } + public org.bukkit.configuration.ConfigurationSection createSection(java.lang.String arg0, java.util.Map arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__createSection", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.ConfigurationSection) result; + } + public boolean isString(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isString", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public boolean isInt(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isInt", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public boolean isBoolean(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isBoolean", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public boolean isDouble(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isDouble", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public boolean isLong(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isLong", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public java.util.List getList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List getList(java.lang.String arg0, java.util.List arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getList", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public boolean isList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public java.util.List getStringList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getStringList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List getIntegerList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getIntegerList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List getBooleanList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getBooleanList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List getDoubleList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getDoubleList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List getFloatList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getFloatList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List getLongList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getLongList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List getByteList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getByteList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List getCharacterList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getCharacterList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List getShortList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getShortList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List> getMapList(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getMapList", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List>) result; + } + public ConfigurationSerializable getSerializable(java.lang.String arg0, java.lang.Class arg1, org.bukkit.configuration.serialization.ConfigurationSerializable arg2) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getSerializable", new Object[] { + arg0, arg1, arg2 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.serialization.ConfigurationSerializable) result; + } + public org.bukkit.configuration.serialization.ConfigurationSerializable getSerializable(java.lang.String arg0, java.lang.Class arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getSerializable", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.serialization.ConfigurationSerializable) result; + } + public org.bukkit.util.Vector getVector(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getVector", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.util.Vector) result; + } + public org.bukkit.util.Vector getVector(java.lang.String arg0, org.bukkit.util.Vector arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getVector", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.util.Vector) result; + } + public boolean isVector(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isVector", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public boolean isOfflinePlayer(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isOfflinePlayer", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public boolean isItemStack(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isItemStack", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public boolean isColor(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isColor", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public boolean isLocation(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isLocation", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public java.util.List getComments(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getComments", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public java.util.List getInlineComments(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getInlineComments", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.util.List) result; + } + public void setComments(java.lang.String arg0, java.util.List arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__setComments", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public void setInlineComments(java.lang.String arg0, java.util.List arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__setInlineComments", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public java.lang.String getName() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getName", new Object[] { + + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.String) result; + } + public java.lang.Object get(java.lang.String arg0, java.lang.Object arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__get", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.Object) result; + } + public java.lang.Object get(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__get", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.Object) result; + } + public boolean getBoolean(java.lang.String arg0, boolean arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getBoolean", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public boolean getBoolean(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getBoolean", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public int getInt(java.lang.String arg0, int arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getInt", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (int) result; + } + public int getInt(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getInt", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (int) result; + } + public long getLong(java.lang.String arg0, long arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getLong", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (long) result; + } + public long getLong(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getLong", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (long) result; + } + public double getDouble(java.lang.String arg0, double arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getDouble", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (double) result; + } + public double getDouble(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getDouble", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (double) result; + } + public boolean contains(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__contains", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public boolean contains(java.lang.String arg0, boolean arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__contains", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public org.bukkit.Location getLocation(java.lang.String arg0, org.bukkit.Location arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getLocation", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.Location) result; + } + public org.bukkit.Location getLocation(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getLocation", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.Location) result; + } + public org.bukkit.configuration.ConfigurationSection getParent() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getParent", new Object[] { + + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.ConfigurationSection) result; + } + public void set(java.lang.String arg0, java.lang.Object arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__set", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + } + public boolean isSet(java.lang.String arg0) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__isSet", new Object[] { + arg0 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (boolean) result; + } + public org.bukkit.configuration.Configuration getRoot() { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getRoot", new Object[] { + + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (org.bukkit.configuration.Configuration) result; + } + public java.lang.Object getObject(java.lang.String arg0, java.lang.Class arg1, java.lang.Object arg2) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getObject", new Object[] { + arg0, arg1, arg2 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.Object) result; + } + public java.lang.Object getObject(java.lang.String arg0, java.lang.Class arg1) { + Object result = null; + try { + result = Native.execute(this.inLibName, "__extends__Configuration__"+this.name+"__getObject", new Object[] { + arg0, arg1 + }); + } catch(Exception ex) { + ex.printStackTrace(); + this.blackBox.getLogger().severe("The fact that an error was thrown at this stage indicates a severe error. Assuming the plugin can no longer run safely, it is being shut off."); + this.blackBox.disable(); + } + return (java.lang.Object) result; + } +} \ No newline at end of file diff --git a/src/main/resources/.gitignore b/src/main/resources/.gitignore new file mode 100755 index 0000000..d6fc3c7 --- /dev/null +++ b/src/main/resources/.gitignore @@ -0,0 +1,2 @@ +native* +libnative* \ No newline at end of file diff --git a/src/main/resources/plugin.yml b/src/main/resources/plugin.yml new file mode 100755 index 0000000..0427701 --- /dev/null +++ b/src/main/resources/plugin.yml @@ -0,0 +1,4 @@ +name: BlackBox +version: '${version}' +main: net.ioixd.blackbox.BlackBox +api-version: 1.13