diff --git a/LICENSE b/LICENSE
deleted file mode 100644
index 0ad25db4..00000000
--- a/LICENSE
+++ /dev/null
@@ -1,661 +0,0 @@
- 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/Makefile b/Makefile
index f6362e19..d3e5c410 100644
--- a/Makefile
+++ b/Makefile
@@ -1,3 +1,15 @@
+# Current Operator version
+VERSION ?= 0.0.1
+# Default bundle image tag
+BUNDLE_IMG ?= controller-bundle:$(VERSION)
+# Options for 'bundle-build'
+ifneq ($(origin CHANNELS), undefined)
+BUNDLE_CHANNELS := --channels=$(CHANNELS)
+endif
+ifneq ($(origin DEFAULT_CHANNEL), undefined)
+BUNDLE_DEFAULT_CHANNEL := --default-channel=$(DEFAULT_CHANNEL)
+endif
+BUNDLE_METADATA_OPTS ?= $(BUNDLE_CHANNELS) $(BUNDLE_DEFAULT_CHANNEL)
# Image URL to use all building/pushing image targets
IMG ?= controller:latest
@@ -14,8 +26,11 @@ endif
all: manager
# Run tests
+ENVTEST_ASSETS_DIR = $(shell pwd)/testbin
test: generate fmt vet manifests
- go test ./... -coverprofile cover.out
+ mkdir -p $(ENVTEST_ASSETS_DIR)
+ test -f $(ENVTEST_ASSETS_DIR)/setup-envtest.sh || curl -sSLo $(ENVTEST_ASSETS_DIR)/setup-envtest.sh https://raw.githubusercontent.com/kubernetes-sigs/controller-runtime/v0.6.3/hack/setup-envtest.sh
+ source $(ENVTEST_ASSETS_DIR)/setup-envtest.sh; fetch_envtest_tools $(ENVTEST_ASSETS_DIR); setup_envtest_env $(ENVTEST_ASSETS_DIR); go test ./... -coverprofile cover.out
# Build manager binary
manager: generate fmt vet
@@ -93,3 +108,16 @@ KUSTOMIZE=$(GOBIN)/kustomize
else
KUSTOMIZE=$(shell which kustomize)
endif
+
+# Generate bundle manifests and metadata, then validate generated files.
+.PHONY: bundle
+bundle: manifests
+ operator-sdk generate kustomize manifests -q
+ cd config/manager && $(KUSTOMIZE) edit set image controller=$(IMG)
+ $(KUSTOMIZE) build config/manifests | operator-sdk generate bundle -q --overwrite --version $(VERSION) $(BUNDLE_METADATA_OPTS)
+ operator-sdk bundle validate ./bundle
+
+# Build the bundle image.
+.PHONY: bundle-build
+bundle-build:
+ docker build -f bundle.Dockerfile -t $(BUNDLE_IMG) .
diff --git a/PROJECT b/PROJECT
index 17cf1210..ce4a8b24 100644
--- a/PROJECT
+++ b/PROJECT
@@ -1,7 +1,14 @@
-domain: k6.io
+domain: io
+layout: go.kubebuilder.io/v2
+projectName: operator
repo: github.com/k6io/operator
resources:
-- group: test
+- group: k6
kind: K6
version: v1alpha1
-version: "2"
+- group: k6
+ kind: K6Test
+ version: v1alpha1
+version: 3-alpha
+plugins:
+ go.sdk.operatorframework.io/v2-alpha: {}
diff --git a/README.md b/README.md
index 50e12a4d..31c23638 100644
--- a/README.md
+++ b/README.md
@@ -1 +1,9 @@
-# operator
\ No newline at end of file
+
+
+# k6 Operator
+
+> ### ⚠️ Experimental
+>
+> This project is **experimental** and changes a lot
+> between commits.
+
diff --git a/api/v1alpha1/groupversion_info.go b/api/v1alpha1/groupversion_info.go
index ecafd92e..f762dbe1 100644
--- a/api/v1alpha1/groupversion_info.go
+++ b/api/v1alpha1/groupversion_info.go
@@ -1,6 +1,22 @@
-// Package v1alpha1 contains API Schema definitions for the test v1alpha1 API group
+/*
+
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+// Package v1alpha1 contains API Schema definitions for the k6 v1alpha1 API group
// +kubebuilder:object:generate=true
-// +groupName=test.k6.io
+// +groupName=k6.io
package v1alpha1
import (
@@ -10,7 +26,7 @@ import (
var (
// GroupVersion is group version used to register these objects
- GroupVersion = schema.GroupVersion{Group: "test.k6.io", Version: "v1alpha1"}
+ GroupVersion = schema.GroupVersion{Group: "k6.io", Version: "v1alpha1"}
// SchemeBuilder is used to add go types to the GroupVersionKind scheme
SchemeBuilder = &scheme.Builder{GroupVersion: GroupVersion}
diff --git a/api/v1alpha1/k6_types.go b/api/v1alpha1/k6_types.go
index 3a3d194f..b5e0da04 100644
--- a/api/v1alpha1/k6_types.go
+++ b/api/v1alpha1/k6_types.go
@@ -1,3 +1,17 @@
+/*
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
package v1alpha1
import (
@@ -6,29 +20,27 @@ import (
// K6Spec defines the desired state of K6
type K6Spec struct {
- Script string `json:"script"`
- Options string `json:"options,omitempty"`
- Nodes int32 `json:"Parallelism"`
+ Script string `json:"script"`
+ Parallelism int32 `json:"parallelism"`
+ Separate bool `json:"separate,omitempty"`
}
// K6Status defines the observed state of K6
-type K6Status struct {
-}
+type K6Status struct{}
+// K6 is the Schema for the k6s API
// +kubebuilder:object:root=true
// +kubebuilder:subresource:status
-
-// K6 is the Schema for the k6s API
type K6 struct {
metav1.TypeMeta `json:",inline"`
metav1.ObjectMeta `json:"metadata,omitempty"`
- Spec K6Spec `json:"spec,omitempty"`
- Status K6Status `json:"status,omitempty"`
-}
-// +kubebuilder:object:root=true
+ Spec K6Spec `json:"spec,omitempty"`
+ Status K6Status `json:"status,omitempty"`
+}
// K6List contains a list of K6
+// +kubebuilder:object:root=true
type K6List struct {
metav1.TypeMeta `json:",inline"`
metav1.ListMeta `json:"metadata,omitempty"`
diff --git a/api/v1alpha1/zz_generated.deepcopy.go b/api/v1alpha1/zz_generated.deepcopy.go
index fd368dfd..9e2e4be9 100644
--- a/api/v1alpha1/zz_generated.deepcopy.go
+++ b/api/v1alpha1/zz_generated.deepcopy.go
@@ -1,5 +1,21 @@
// +build !ignore_autogenerated
+/*
+
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
// Code generated by controller-gen. DO NOT EDIT.
package v1alpha1
diff --git a/assets/data-flow.png b/assets/data-flow.png
new file mode 100644
index 00000000..699e772e
Binary files /dev/null and b/assets/data-flow.png differ
diff --git a/config/crd/bases/test.k6.io_k6s.yaml b/config/crd/bases/k6.io_k6s.yaml
similarity index 87%
rename from config/crd/bases/test.k6.io_k6s.yaml
rename to config/crd/bases/k6.io_k6s.yaml
index e00600be..be775d2b 100644
--- a/config/crd/bases/test.k6.io_k6s.yaml
+++ b/config/crd/bases/k6.io_k6s.yaml
@@ -6,9 +6,9 @@ metadata:
annotations:
controller-gen.kubebuilder.io/version: v0.3.0
creationTimestamp: null
- name: k6s.test.k6.io
+ name: k6s.k6.io
spec:
- group: test.k6.io
+ group: k6.io
names:
kind: K6
listKind: K6List
@@ -36,22 +36,15 @@ spec:
spec:
description: K6Spec defines the desired state of K6
properties:
- backoff:
+ parallelism:
format: int32
type: integer
- cleanup:
- type: boolean
- nodes:
- format: int32
- type: integer
- options:
- type: string
script:
type: string
+ separate:
+ type: boolean
required:
- - backoff
- - cleanup
- - nodes
+ - parallelism
- script
type: object
status:
diff --git a/config/crd/kustomization.yaml b/config/crd/kustomization.yaml
index 837c48b2..98f3a9d6 100644
--- a/config/crd/kustomization.yaml
+++ b/config/crd/kustomization.yaml
@@ -2,18 +2,20 @@
# since it depends on service name and namespace that are out of this kustomize package.
# It should be run by config/default
resources:
-- bases/test.k6.io_k6s.yaml
+- bases/k6.io_k6s.yaml
# +kubebuilder:scaffold:crdkustomizeresource
patchesStrategicMerge:
# [WEBHOOK] To enable webhook, uncomment all the sections with [WEBHOOK] prefix.
# patches here are for enabling the conversion webhook for each CRD
#- patches/webhook_in_k6s.yaml
+#- patches/webhook_in_k6tests.yaml
# +kubebuilder:scaffold:crdkustomizewebhookpatch
# [CERTMANAGER] To enable webhook, uncomment all the sections with [CERTMANAGER] prefix.
# patches here are for enabling the CA injection for each CRD
#- patches/cainjection_in_k6s.yaml
+#- patches/cainjection_in_k6tests.yaml
# +kubebuilder:scaffold:crdkustomizecainjectionpatch
# the following config is for teaching kustomize how to do kustomization for CRDs.
diff --git a/config/crd/patches/cainjection_in_k6s.yaml b/config/crd/patches/cainjection_in_k6s.yaml
index 35e201ec..9c8247c5 100644
--- a/config/crd/patches/cainjection_in_k6s.yaml
+++ b/config/crd/patches/cainjection_in_k6s.yaml
@@ -5,4 +5,4 @@ kind: CustomResourceDefinition
metadata:
annotations:
cert-manager.io/inject-ca-from: $(CERTIFICATE_NAMESPACE)/$(CERTIFICATE_NAME)
- name: k6s.test.k6.io
+ name: k6s.k6.io
diff --git a/config/crd/patches/webhook_in_k6s.yaml b/config/crd/patches/webhook_in_k6s.yaml
index d9b6d9d1..c90788df 100644
--- a/config/crd/patches/webhook_in_k6s.yaml
+++ b/config/crd/patches/webhook_in_k6s.yaml
@@ -3,7 +3,7 @@
apiVersion: apiextensions.k8s.io/v1beta1
kind: CustomResourceDefinition
metadata:
- name: k6s.test.k6.io
+ name: k6s.k6.io
spec:
conversion:
strategy: Webhook
diff --git a/config/manager/kustomization.yaml b/config/manager/kustomization.yaml
index 5c5f0b84..0ed2f2de 100644
--- a/config/manager/kustomization.yaml
+++ b/config/manager/kustomization.yaml
@@ -1,2 +1,8 @@
resources:
- manager.yaml
+images:
+ - name: controller
+ newName: ghcr.io/k6io/operator
+ newTag: latest
+apiVersion: kustomize.config.k8s.io/v1beta1
+kind: Kustomization
\ No newline at end of file
diff --git a/config/manager/manager.yaml b/config/manager/manager.yaml
index b6c85a52..f12df7d0 100644
--- a/config/manager/manager.yaml
+++ b/config/manager/manager.yaml
@@ -2,6 +2,7 @@ apiVersion: v1
kind: Namespace
metadata:
labels:
+ app.kubernetes.io/name: k6-operator
control-plane: controller-manager
name: system
---
diff --git a/config/rbac/k6_editor_role.yaml b/config/rbac/k6_editor_role.yaml
index d576034d..1bdc3be8 100644
--- a/config/rbac/k6_editor_role.yaml
+++ b/config/rbac/k6_editor_role.yaml
@@ -5,7 +5,7 @@ metadata:
name: k6-editor-role
rules:
- apiGroups:
- - test.k6.io
+ - k6.io
resources:
- k6s
verbs:
@@ -17,7 +17,7 @@ rules:
- update
- watch
- apiGroups:
- - test.k6.io
+ - k6.io
resources:
- k6s/status
verbs:
diff --git a/config/rbac/k6_viewer_role.yaml b/config/rbac/k6_viewer_role.yaml
index 684ff7ee..9aebbdf5 100644
--- a/config/rbac/k6_viewer_role.yaml
+++ b/config/rbac/k6_viewer_role.yaml
@@ -5,7 +5,7 @@ metadata:
name: k6-viewer-role
rules:
- apiGroups:
- - test.k6.io
+ - k6.io
resources:
- k6s
verbs:
@@ -13,7 +13,7 @@ rules:
- list
- watch
- apiGroups:
- - test.k6.io
+ - k6.io
resources:
- k6s/status
verbs:
diff --git a/config/rbac/role.yaml b/config/rbac/role.yaml
index 19c219e6..15d76463 100644
--- a/config/rbac/role.yaml
+++ b/config/rbac/role.yaml
@@ -7,7 +7,7 @@ metadata:
name: manager-role
rules:
- apiGroups:
- - test.k6.io
+ - k6.io
resources:
- k6s
verbs:
@@ -19,7 +19,7 @@ rules:
- update
- watch
- apiGroups:
- - test.k6.io
+ - k6.io
resources:
- k6s/status
verbs:
diff --git a/config/samples/k6_v1alpha1_configmap.yml b/config/samples/k6_v1alpha1_configmap.yml
new file mode 100644
index 00000000..f41f5d2b
--- /dev/null
+++ b/config/samples/k6_v1alpha1_configmap.yml
@@ -0,0 +1,42 @@
+apiVersion: v1
+kind: ConfigMap
+metadata:
+ namespace: k6
+ name: k6-test
+data:
+ test.js: |
+ import http from 'k6/http';
+ import { Rate } from 'k6/metrics';
+ import { check, sleep } from 'k6';
+
+ import {
+ getExecutionSegments,
+ logSegment,
+ } from 'https://jslib.k6.io/k8s-distributed-execution/0.0.1/index.js';
+
+ const failRate = new Rate('failed_requests');
+
+ export function setup() {
+ logSegment();
+ }
+
+ export let options = Object.assign({
+ stages: [
+ { target: 200, duration: '30s' },
+ { target: 0, duration: '30s' },
+ ],
+ threshold: {
+ failed_requests: ['rate<=0'],
+ http_req_duration: ['p(95)<500'],
+ },
+ }, getExecutionSegments());
+
+ export default function () {
+ const result = http.get('https://test-api.k6.io/public/crocodiles/');
+ check(result, {
+ 'http response status code is 200': result.status === 200,
+ });
+ failRate.add(result.status !== 200);
+ sleep(1);
+ }
+
diff --git a/config/samples/k6_v1alpha1_k6.yaml b/config/samples/k6_v1alpha1_k6.yaml
new file mode 100644
index 00000000..8e0894af
--- /dev/null
+++ b/config/samples/k6_v1alpha1_k6.yaml
@@ -0,0 +1,9 @@
+apiVersion: k6.io/v1alpha1
+kind: K6
+metadata:
+ namespace: k6
+ name: k6-sample
+spec:
+ parallelism: 4
+ script: k6-test
+ separate: false
\ No newline at end of file
diff --git a/config/samples/kustomization.yaml b/config/samples/kustomization.yaml
new file mode 100644
index 00000000..ef4d1300
--- /dev/null
+++ b/config/samples/kustomization.yaml
@@ -0,0 +1,5 @@
+## Append samples you want in your CSV to this file as resources ##
+resources:
+ - k6_v1alpha1_configmap.yaml
+ - k6_v1alpha1_k6.yaml
+# +kubebuilder:scaffold:manifestskustomizesamples
diff --git a/config/samples/test_v1alpha1_configmap.yaml b/config/samples/test_v1alpha1_configmap.yaml
deleted file mode 100644
index 9faba507..00000000
--- a/config/samples/test_v1alpha1_configmap.yaml
+++ /dev/null
@@ -1,55 +0,0 @@
-apiVersion: v1
-kind: ConfigMap
-metadata:
- name: k6-test
-data:
- test.js: |
- import http from 'k6/http';
-
- import { Rate } from 'k6/metrics';
- import { check, sleep } from 'k6';
-
- const failRate = new Rate('failed_requests');
-
- function getExecutionSegment() {
- const n = __ENV.K6_NODES_INDEX;
- const s = __ENV.K6_NODES_TOTAL;
-
- let sequence = [];
- for (let i=0; i <= s; i++) {
- sequence.push(`${i}/${s}`);
- }
-
- let output = {
- executionSegment: `${n-1}/${s}:${n}/${s}`,
- executionSegmentSequence: sequence.join(','),
- }
-
- console.log(JSON.stringify(output, null, 2));
- return output;
- }
-
-
- const segment = getExecutionSegment();
- export const options = {
- executionSegment: segment.executionSegment,
- executionSegmentSequence: segment.executionSegmentSequence,
- stages: [
- { target: 200, duration: '30s' },
- { target: 0, duration: '30s' },
- ],
- threshold: {
- failed_requests: ['rate<=0'],
- http_req_duration: ['p(95)<500'],
- },
- };
-
- export default function () {
- console.log(__VU);
- const result = http.get('https://test-api.k6.io/public/crocodiles/');
- check(result, {
- 'http response status code is 200': result.status === 200,
- });
- failRate.add(result.status !== 200);
- sleep(1);
- }
\ No newline at end of file
diff --git a/config/samples/test_v1alpha1_k6.yaml b/config/samples/test_v1alpha1_k6.yaml
deleted file mode 100644
index ce334fe0..00000000
--- a/config/samples/test_v1alpha1_k6.yaml
+++ /dev/null
@@ -1,7 +0,0 @@
-apiVersion: test.k6.io/v1alpha1
-kind: K6
-metadata:
- name: k6-sample
-spec:
- parallelism: 1
- script: k6-test
\ No newline at end of file
diff --git a/config/scorecard/bases/config.yaml b/config/scorecard/bases/config.yaml
new file mode 100644
index 00000000..c7704784
--- /dev/null
+++ b/config/scorecard/bases/config.yaml
@@ -0,0 +1,7 @@
+apiVersion: scorecard.operatorframework.io/v1alpha3
+kind: Configuration
+metadata:
+ name: config
+stages:
+- parallel: true
+ tests: []
diff --git a/config/scorecard/kustomization.yaml b/config/scorecard/kustomization.yaml
new file mode 100644
index 00000000..d73509ee
--- /dev/null
+++ b/config/scorecard/kustomization.yaml
@@ -0,0 +1,16 @@
+resources:
+- bases/config.yaml
+patchesJson6902:
+- path: patches/basic.config.yaml
+ target:
+ group: scorecard.operatorframework.io
+ version: v1alpha3
+ kind: Configuration
+ name: config
+- path: patches/olm.config.yaml
+ target:
+ group: scorecard.operatorframework.io
+ version: v1alpha3
+ kind: Configuration
+ name: config
+# +kubebuilder:scaffold:patchesJson6902
diff --git a/config/scorecard/patches/basic.config.yaml b/config/scorecard/patches/basic.config.yaml
new file mode 100644
index 00000000..39576a57
--- /dev/null
+++ b/config/scorecard/patches/basic.config.yaml
@@ -0,0 +1,10 @@
+- op: add
+ path: /stages/0/tests/-
+ value:
+ entrypoint:
+ - scorecard-test
+ - basic-check-spec
+ image: quay.io/operator-framework/scorecard-test:v1.1.0
+ labels:
+ suite: basic
+ test: basic-check-spec-test
diff --git a/config/scorecard/patches/olm.config.yaml b/config/scorecard/patches/olm.config.yaml
new file mode 100644
index 00000000..8102e4db
--- /dev/null
+++ b/config/scorecard/patches/olm.config.yaml
@@ -0,0 +1,50 @@
+- op: add
+ path: /stages/0/tests/-
+ value:
+ entrypoint:
+ - scorecard-test
+ - olm-bundle-validation
+ image: quay.io/operator-framework/scorecard-test:v1.1.0
+ labels:
+ suite: olm
+ test: olm-bundle-validation-test
+- op: add
+ path: /stages/0/tests/-
+ value:
+ entrypoint:
+ - scorecard-test
+ - olm-crds-have-validation
+ image: quay.io/operator-framework/scorecard-test:v1.1.0
+ labels:
+ suite: olm
+ test: olm-crds-have-validation-test
+- op: add
+ path: /stages/0/tests/-
+ value:
+ entrypoint:
+ - scorecard-test
+ - olm-crds-have-resources
+ image: quay.io/operator-framework/scorecard-test:v1.1.0
+ labels:
+ suite: olm
+ test: olm-crds-have-resources-test
+- op: add
+ path: /stages/0/tests/-
+ value:
+ entrypoint:
+ - scorecard-test
+ - olm-spec-descriptors
+ image: quay.io/operator-framework/scorecard-test:v1.1.0
+ labels:
+ suite: olm
+ test: olm-spec-descriptors-test
+- op: add
+ path: /stages/0/tests/-
+ value:
+ entrypoint:
+ - scorecard-test
+ - olm-status-descriptors
+ image: quay.io/operator-framework/scorecard-test:v1.1.0
+ labels:
+ suite: olm
+ test: olm-status-descriptors-test
diff --git a/controllers/k6_controller.go b/controllers/k6_controller.go
index fb39f789..3167ce8b 100644
--- a/controllers/k6_controller.go
+++ b/controllers/k6_controller.go
@@ -1,18 +1,33 @@
+/*
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
package controllers
import (
"context"
- "github.com/go-logr/logr"
- testv1alpha1 "github.com/k6io/operator/api/v1alpha1"
- appsv1 "k8s.io/api/apps/v1"
+ "fmt"
+ "github.com/k6io/operator/pkg/resources"
batchv1 "k8s.io/api/batch/v1"
- corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/api/errors"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/types"
+
+ "github.com/go-logr/logr"
+ "k8s.io/apimachinery/pkg/runtime"
ctrl "sigs.k8s.io/controller-runtime"
"sigs.k8s.io/controller-runtime/pkg/client"
+
+ "github.com/k6io/operator/api/v1alpha1"
)
// K6Reconciler reconciles a K6 object
@@ -22,102 +37,50 @@ type K6Reconciler struct {
Scheme *runtime.Scheme
}
-// +kubebuilder:rbac:groups=test.k6.io,resources=k6s,verbs=get;list;watch;create;update;patch;delete
-// +kubebuilder:rbac:groups=test.k6.io,resources=k6s/status,verbs=get;update;patch
+// +kubebuilder:rbac:groups=k6.io,resources=k6s,verbs=get;list;watch;create;update;patch;delete
+// +kubebuilder:rbac:groups=k6.io,resources=k6s/status,verbs=get;update;patch
func (r *K6Reconciler) Reconcile(req ctrl.Request) (ctrl.Result, error) {
ctx := context.Background()
log := r.Log.WithValues("k6", req.NamespacedName)
- k6 := &testv1alpha1.K6{}
+ result := ctrl.Result{}
+
+ // Fetch the applied CRD
+ k6 := &v1alpha1.K6{}
err := r.Get(ctx, req.NamespacedName, k6)
if err != nil {
if errors.IsNotFound(err) {
log.Info("Request deleted. Skipping requeuing.")
- return ctrl.Result{}, nil
+ return result, nil
}
log.Error(err, "Could not fetch request")
return ctrl.Result{Requeue: true}, err
}
+ // Check for previous jobs
found := &batchv1.Job{}
- err = r.Get(ctx, types.NamespacedName{Name: k6.Name, Namespace: k6.Namespace}, found)
+ err = r.Get(ctx, types.NamespacedName{Name: fmt.Sprintf("%s-1", k6.Name), Namespace: k6.Namespace}, found)
if err == nil || !errors.IsNotFound(err) {
log.Info("Could not start a new test, Make sure you've deleted your previous run.")
- return ctrl.Result{}, err
+ return result, err
}
- log.Info("Launching k6 test")
- deployment := r.newDeployment(k6)
- err = r.Create(ctx, deployment)
- if err != nil {
- log.Error(err, "Failed to launch k6 test")
- return ctrl.Result{}, err
+ // Create jobs
+ for i := 1; i <= int(k6.Spec.Parallelism); i++ {
+ log.Info(fmt.Sprintf("Launching k6 test #%d", i))
+ job := resources.NewJob(k6, i)
+ if err = r.Create(ctx, job); err != nil {
+ log.Error(err, "Failed to launch k6 test")
+ return ctrl.Result{}, err
+ }
}
- return ctrl.Result{}, nil
+
+ return result, nil
}
func (r *K6Reconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
- For(&testv1alpha1.K6{}).
- Owns(&appsv1.Deployment{}).
+ For(&v1alpha1.K6{}).
Complete(r)
}
-
-func (r *K6Reconciler) newDeployment(k *testv1alpha1.K6) *batchv1.Job {
- job := &batchv1.Job{
- ObjectMeta: metav1.ObjectMeta{
- Name: k.Name,
- Namespace: k.Namespace,
- },
- Spec: batchv1.JobSpec{
- Parallelism: &k.Spec.Nodes,
- Template: corev1.PodTemplateSpec{
- ObjectMeta: metav1.ObjectMeta{
- Labels: newLabels(k.Name),
- },
- Spec: corev1.PodSpec{
- RestartPolicy: corev1.RestartPolicyNever,
- Containers: []corev1.Container{{
- Image: "loadimpact/k6:latest",
- Name: "k6",
- Command: []string{"k6", "run", "/test/test.js"},
- VolumeMounts: []corev1.VolumeMount{{
- Name: "k6-test-volume",
- MountPath: "/test",
- }},
- Env: []corev1.EnvVar{{
- Name: "K6_NODES_INDEX",
- Value: "1",
- }, {
- Name: "K6_NODES_TOTAL",
- Value: "1",
- }},
- }},
- Volumes: []corev1.Volume{{
- Name: "k6-test-volume",
- VolumeSource: corev1.VolumeSource{
- ConfigMap: &corev1.ConfigMapVolumeSource{
- LocalObjectReference: corev1.LocalObjectReference{
- Name: k.Spec.Script,
- },
- },
- },
- }},
- },
- },
- },
- }
- err := ctrl.SetControllerReference(k, job, r.Scheme)
- if err != nil {
- return nil
- }
- return job
-}
-
-func newLabels(name string) map[string]string {
- return map[string]string{
- "app": "k6",
- "k6_cr": name,
- }
-}
diff --git a/controllers/suite_test.go b/controllers/suite_test.go
index 57ac6186..e51ab0f2 100644
--- a/controllers/suite_test.go
+++ b/controllers/suite_test.go
@@ -1,3 +1,19 @@
+/*
+
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
package controllers
import (
@@ -14,7 +30,7 @@ import (
logf "sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/log/zap"
- testv1alpha1 "github.com/k6io/operator/api/v1alpha1"
+ k6v1alpha1 "github.com/k6io/operator/api/v1alpha1"
// +kubebuilder:scaffold:imports
)
@@ -46,7 +62,10 @@ var _ = BeforeSuite(func(done Done) {
Expect(err).ToNot(HaveOccurred())
Expect(cfg).ToNot(BeNil())
- err = testv1alpha1.AddToScheme(scheme.Scheme)
+ err = k6v1alpha1.AddToScheme(scheme.Scheme)
+ Expect(err).NotTo(HaveOccurred())
+
+ err = k6v1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
// +kubebuilder:scaffold:scheme
diff --git a/go.mod b/go.mod
index 2abe0ca6..8538c2f3 100644
--- a/go.mod
+++ b/go.mod
@@ -11,3 +11,5 @@ require (
k8s.io/client-go v0.18.6
sigs.k8s.io/controller-runtime v0.6.2
)
+
+replace github.com/k6io/operator => ./
\ No newline at end of file
diff --git a/hack/boilerplate.go.txt b/hack/boilerplate.go.txt
index e69de29b..767efde9 100644
--- a/hack/boilerplate.go.txt
+++ b/hack/boilerplate.go.txt
@@ -0,0 +1,15 @@
+/*
+
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
\ No newline at end of file
diff --git a/main.go b/main.go
index dd539575..1788aa3b 100644
--- a/main.go
+++ b/main.go
@@ -1,3 +1,19 @@
+/*
+
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
package main
import (
@@ -11,7 +27,7 @@ import (
ctrl "sigs.k8s.io/controller-runtime"
"sigs.k8s.io/controller-runtime/pkg/log/zap"
- testv1alpha1 "github.com/k6io/operator/api/v1alpha1"
+ k6v1alpha1 "github.com/k6io/operator/api/v1alpha1"
"github.com/k6io/operator/controllers"
// +kubebuilder:scaffold:imports
)
@@ -24,12 +40,19 @@ var (
func init() {
utilruntime.Must(clientgoscheme.AddToScheme(scheme))
- utilruntime.Must(testv1alpha1.AddToScheme(scheme))
+ utilruntime.Must(k6v1alpha1.AddToScheme(scheme))
// +kubebuilder:scaffold:scheme
}
func main() {
- metricsAddr, enableLeaderElection := parseFlags()
+ var metricsAddr string
+ var enableLeaderElection bool
+ flag.StringVar(&metricsAddr, "metrics-addr", ":8080", "The address the metric endpoint binds to.")
+ flag.BoolVar(&enableLeaderElection, "enable-leader-election", false,
+ "Enable leader election for controller manager. "+
+ "Enabling this will ensure there is only one active controller manager.")
+ flag.Parse()
+
ctrl.SetLogger(zap.New(zap.UseDevMode(true)))
mgr, err := ctrl.NewManager(ctrl.GetConfigOrDie(), ctrl.Options{
@@ -37,7 +60,7 @@ func main() {
MetricsBindAddress: metricsAddr,
Port: 9443,
LeaderElection: enableLeaderElection,
- LeaderElectionID: "fcdfce80.k6.io",
+ LeaderElectionID: "fcdfce80.io",
})
if err != nil {
setupLog.Error(err, "unable to start manager")
@@ -60,23 +83,3 @@ func main() {
os.Exit(1)
}
}
-
-func parseFlags() (string, bool) {
- var metricsAddr string
- flag.StringVar(
- &metricsAddr,
- "metrics-addr",
- ":8080",
- "The address the metric endpoint binds to.",
- )
-
- var enableLeaderElection bool
- flag.BoolVar(
- &enableLeaderElection,
- "enable-leader-election",
- false,
- "Enable leader election for controller manager. "+
- "Enabling this will ensure there is only one active controller manager.")
- flag.Parse()
- return metricsAddr, enableLeaderElection
-}
diff --git a/pkg/resources/resources.go b/pkg/resources/resources.go
new file mode 100644
index 00000000..864e7634
--- /dev/null
+++ b/pkg/resources/resources.go
@@ -0,0 +1,104 @@
+package resources
+
+import (
+ "fmt"
+ "github.com/k6io/operator/api/v1alpha1"
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+)
+
+// NewJob creates a new k6 job from a CRD
+func NewJob(k *v1alpha1.K6, index int) *batchv1.Job {
+ name := fmt.Sprintf("%s-%d", k.Name, index)
+
+ job := &batchv1.Job{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: name,
+ Namespace: k.Namespace,
+ },
+ Spec: batchv1.JobSpec{
+ Template: corev1.PodTemplateSpec{
+ ObjectMeta: metav1.ObjectMeta{
+ Labels: newLabels(k.Name),
+ },
+ Spec: corev1.PodSpec{
+ RestartPolicy: corev1.RestartPolicyNever,
+ Containers: []corev1.Container{{
+ Image: "loadimpact/k6:latest",
+ Name: "k6",
+ Command: []string{"k6", "run", "/test/test.js"},
+ VolumeMounts: []corev1.VolumeMount{{
+ Name: "k6-test-volume",
+ MountPath: "/test",
+ }},
+ Env: newEnvVars(k.Spec.Parallelism, index),
+ }},
+ Volumes: newVolumeSpec(k.Spec.Script),
+ },
+ },
+ },
+ }
+
+ if k.Spec.Separate {
+ job.Spec.Template.Spec.Affinity = newAntiAffinity()
+ }
+
+ return job
+}
+
+func newEnvVars(parallelism int32, index int) []corev1.EnvVar {
+ return []corev1.EnvVar{
+ {
+ Name: "K6_INSTANCES_INDEX",
+ Value: fmt.Sprintf("%d", index),
+ },
+ {
+ Name: "K6_INSTANCES_TOTAL",
+ Value: fmt.Sprintf("%d", parallelism),
+ },
+ }
+}
+
+func newVolumeSpec(script string) []corev1.Volume {
+ return []corev1.Volume{{
+ Name: "k6-test-volume",
+ VolumeSource: corev1.VolumeSource{
+ ConfigMap: &corev1.ConfigMapVolumeSource{
+ LocalObjectReference: corev1.LocalObjectReference{
+ Name: script,
+ },
+ },
+ },
+ }}
+}
+
+func newAntiAffinity() *corev1.Affinity {
+ return &corev1.Affinity{
+ PodAntiAffinity: &corev1.PodAntiAffinity{
+ RequiredDuringSchedulingIgnoredDuringExecution: []corev1.PodAffinityTerm{
+ {
+ LabelSelector: &metav1.LabelSelector{
+ MatchExpressions: []metav1.LabelSelectorRequirement{
+ {
+ Key: "app",
+ Operator: "In",
+ Values: []string{
+ "k6",
+ },
+ },
+ },
+ },
+ TopologyKey: "kubernetes.io/hostname",
+ },
+ },
+ },
+ }
+}
+
+func newLabels(name string) map[string]string {
+ return map[string]string{
+ "app": "k6",
+ "k6_cr": name,
+ }
+}
diff --git a/testbin/setup-envtest.sh b/testbin/setup-envtest.sh
new file mode 100644
index 00000000..61f69db0
--- /dev/null
+++ b/testbin/setup-envtest.sh
@@ -0,0 +1,96 @@
+#!/usr/bin/env bash
+
+# Copyright 2020 The Kubernetes 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
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+set -o errexit
+set -o pipefail
+
+# Turn colors in this script off by setting the NO_COLOR variable in your
+# environment to any value:
+#
+# $ NO_COLOR=1 test.sh
+NO_COLOR=${NO_COLOR:-""}
+if [ -z "$NO_COLOR" ]; then
+ header=$'\e[1;33m'
+ reset=$'\e[0m'
+else
+ header=''
+ reset=''
+fi
+
+function header_text {
+ echo "$header$*$reset"
+}
+
+function setup_envtest_env {
+ header_text "setting up env vars"
+
+ # Setup env vars
+ KUBEBUILDER_ASSETS=${KUBEBUILDER_ASSETS:-""}
+ if [[ -z "${KUBEBUILDER_ASSETS}" ]]; then
+ export KUBEBUILDER_ASSETS=$1/bin
+ fi
+}
+
+# fetch k8s API gen tools and make it available under envtest_root_dir/bin.
+#
+# Skip fetching and untaring the tools by setting the SKIP_FETCH_TOOLS variable
+# in your environment to any value:
+#
+# $ SKIP_FETCH_TOOLS=1 ./check-everything.sh
+#
+# If you skip fetching tools, this script will use the tools already on your
+# machine.
+function fetch_envtest_tools {
+ SKIP_FETCH_TOOLS=${SKIP_FETCH_TOOLS:-""}
+ if [ -n "$SKIP_FETCH_TOOLS" ]; then
+ return 0
+ fi
+
+ tmp_root=/tmp
+ envtest_root_dir=$tmp_root/envtest
+
+ k8s_version="${ENVTEST_K8S_VERSION:-1.16.4}"
+ goarch="$(go env GOARCH)"
+ goos="$(go env GOOS)"
+
+ if [[ "$goos" != "linux" && "$goos" != "darwin" ]]; then
+ echo "OS '$goos' not supported. Aborting." >&2
+ return 1
+ fi
+
+ local dest_dir="${1}"
+
+ # use the pre-existing version in the temporary folder if it matches our k8s version
+ if [[ -x "${dest_dir}/bin/kube-apiserver" ]]; then
+ version=$("${dest_dir}"/bin/kube-apiserver --version)
+ if [[ $version == *"${k8s_version}"* ]]; then
+ header_text "Using cached envtest tools from ${dest_dir}"
+ return 0
+ fi
+ fi
+
+ header_text "fetching envtest tools@${k8s_version} (into '${dest_dir}')"
+ envtest_tools_archive_name="kubebuilder-tools-$k8s_version-$goos-$goarch.tar.gz"
+ envtest_tools_download_url="https://storage.googleapis.com/kubebuilder-tools/$envtest_tools_archive_name"
+
+ envtest_tools_archive_path="$tmp_root/$envtest_tools_archive_name"
+ if [ ! -f $envtest_tools_archive_path ]; then
+ curl -sL ${envtest_tools_download_url} -o "$envtest_tools_archive_path"
+ fi
+
+ mkdir -p "${dest_dir}"
+ tar -C "${dest_dir}" --strip-components=1 -zvxf "$envtest_tools_archive_path"
+}