diff --git a/.github/workflows/build_and_test.yml b/.github/workflows/build_and_test.yml
index 0cf6fd9e..0f680ccd 100644
--- a/.github/workflows/build_and_test.yml
+++ b/.github/workflows/build_and_test.yml
@@ -13,13 +13,14 @@ jobs:
linux:
runs-on: ubuntu-latest
+ timeout-minutes: 30
steps:
- uses: actions/checkout@v4
- name: Clone spacebar server
run: |
git clone https://github.com/bitfl0wer/server.git
- - uses: actions/setup-node@v3
+ - uses: actions/setup-node@v4
with:
node-version: 18
cache: 'npm'
@@ -52,7 +53,7 @@ jobs:
# - name: Clone spacebar server
# run: |
# git clone https://github.com/bitfl0wer/server.git
- # - uses: actions/setup-node@v3
+ # - uses: actions/setup-node@v4
# with:
# node-version: 18
# cache: 'npm'
@@ -75,12 +76,13 @@ jobs:
# SAFARIDRIVER=$(which safaridriver) cargo test --target wasm32-unknown-unknown --no-default-features --features="client, rt" --no-fail-fast
wasm-gecko:
runs-on: macos-latest
+ timeout-minutes: 30
steps:
- uses: actions/checkout@v4
- name: Clone spacebar server
run: |
git clone https://github.com/bitfl0wer/server.git
- - uses: actions/setup-node@v3
+ - uses: actions/setup-node@v4
with:
node-version: 18
cache: 'npm'
@@ -100,15 +102,16 @@ jobs:
rustup target add wasm32-unknown-unknown
curl -L --proto '=https' --tlsv1.2 -sSf https://raw.githubusercontent.com/cargo-bins/cargo-binstall/main/install-from-binstall-release.sh | bash
cargo binstall --no-confirm wasm-bindgen-cli --version "0.2.88" --force
- GECKODRIVER=$(which geckodriver) cargo test --target wasm32-unknown-unknown --no-default-features --features="client, rt"
+ GECKODRIVER=$(which geckodriver) cargo test --target wasm32-unknown-unknown --no-default-features --features="client, rt, voice_gateway"
wasm-chrome:
runs-on: macos-latest
+ timeout-minutes: 30
steps:
- uses: actions/checkout@v4
- name: Clone spacebar server
run: |
git clone https://github.com/bitfl0wer/server.git
- - uses: actions/setup-node@v3
+ - uses: actions/setup-node@v4
with:
node-version: 18
cache: 'npm'
@@ -128,4 +131,4 @@ jobs:
rustup target add wasm32-unknown-unknown
curl -L --proto '=https' --tlsv1.2 -sSf https://raw.githubusercontent.com/cargo-bins/cargo-binstall/main/install-from-binstall-release.sh | bash
cargo binstall --no-confirm wasm-bindgen-cli --version "0.2.88" --force
- CHROMEDRIVER=$(which chromedriver) cargo test --target wasm32-unknown-unknown --no-default-features --features="client, rt"
\ No newline at end of file
+ CHROMEDRIVER=$(which chromedriver) cargo test --target wasm32-unknown-unknown --no-default-features --features="client, rt, voice_gateway"
diff --git a/.github/workflows/rust-clippy.yml b/.github/workflows/rust-clippy.yml
index 4c45b0bc..b12c0ed6 100644
--- a/.github/workflows/rust-clippy.yml
+++ b/.github/workflows/rust-clippy.yml
@@ -29,12 +29,9 @@ jobs:
uses: actions/checkout@v4
- name: Install Rust toolchain
- uses: actions-rs/toolchain@v1
+ uses: dtolnay/rust-toolchain@stable
with:
- profile: minimal
- toolchain: stable
components: clippy
- override: true
- name: Install required cargo
run: cargo install clippy-sarif sarif-fmt
@@ -47,7 +44,7 @@ jobs:
continue-on-error: true
- name: Upload analysis results to GitHub
- uses: github/codeql-action/upload-sarif@v2
+ uses: github/codeql-action/upload-sarif@v3
with:
sarif_file: rust-clippy-results.sarif
wait-for-processing: true
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
new file mode 100644
index 00000000..34a1153a
--- /dev/null
+++ b/CONTRIBUTING.md
@@ -0,0 +1,10 @@
+
+# Contributing
+
+**Please refer to the [contribution guidelines](https://github.com/polyphony-chat/.github/blob/main/CONTRIBUTION_GUIDELINES.md) and [our Code of Conduct](https://github.com/polyphony-chat/.github/blob/main/CODE_OF_CONDUCT.md) before making a contribution.**
+
+Chorus is currently missing voice support and a lot of API endpoints, many of which should be trivial to implement,
+ever since [we streamlined the process of doing so](https://github.com/polyphony-chat/chorus/discussions/401).
+
+If you'd like to contribute new functionality, check out [The 'Meta'-issues.](https://github.com/polyphony-chat/chorus/issues?q=is%3Aissue+label%3A%22Type%3A+Meta%22+) They contain a comprehensive list of all features which are yet missing for full Discord.com compatibility.
+Please feel free to open an Issue with the idea you have, or a Pull Request.
\ No newline at end of file
diff --git a/Cargo.lock b/Cargo.lock
index 9a7b2bdb..456ae439 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -17,6 +17,16 @@ version = "1.0.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe"
+[[package]]
+name = "aead"
+version = "0.5.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d122413f284cf2d62fb1b7db97e02edb8cda96d769b16e443a4f6195e35662b0"
+dependencies = [
+ "crypto-common",
+ "generic-array",
+]
+
[[package]]
name = "ahash"
version = "0.8.7"
@@ -199,14 +209,16 @@ checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
[[package]]
name = "chorus"
-version = "0.14.0"
+version = "0.15.0"
dependencies = [
"async-trait",
"base64 0.21.7",
"bitflags 2.4.1",
"chorus-macros",
"chrono",
+ "crypto_secretbox",
"custom_error",
+ "discortp",
"futures-util",
"getrandom",
"hostname",
@@ -264,6 +276,17 @@ dependencies = [
"windows-targets 0.48.5",
]
+[[package]]
+name = "cipher"
+version = "0.4.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "773f3b9af64447d2ce9850330c473515014aa235e6a783b02db81ff39e4a3dad"
+dependencies = [
+ "crypto-common",
+ "inout",
+ "zeroize",
+]
+
[[package]]
name = "console_error_panic_hook"
version = "0.1.7"
@@ -342,9 +365,25 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3"
dependencies = [
"generic-array",
+ "rand_core",
"typenum",
]
+[[package]]
+name = "crypto_secretbox"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b9d6cf87adf719ddf43a805e92c6870a531aedda35ff640442cbaf8674e141e1"
+dependencies = [
+ "aead",
+ "cipher",
+ "generic-array",
+ "poly1305",
+ "salsa20",
+ "subtle",
+ "zeroize",
+]
+
[[package]]
name = "custom_error"
version = "1.9.2"
@@ -425,6 +464,16 @@ dependencies = [
"subtle",
]
+[[package]]
+name = "discortp"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "524b9439c09174aede2c88d58cfc6b83575b06569d1af4d07562f76595b2896b"
+dependencies = [
+ "pnet_macros",
+ "pnet_macros_support",
+]
+
[[package]]
name = "dotenvy"
version = "0.15.7"
@@ -643,6 +692,7 @@ checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a"
dependencies = [
"typenum",
"version_check",
+ "zeroize",
]
[[package]]
@@ -666,9 +716,9 @@ checksum = "4271d37baee1b8c7e4b708028c57d816cf9d2434acb33a549475f78c181f6253"
[[package]]
name = "h2"
-version = "0.3.23"
+version = "0.3.24"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "b553656127a00601c8ae5590fcfdc118e4083a7924b6cf4ffc1ea4b99dc429d7"
+checksum = "bb2c4422095b67ee78da96fbb51a4cc413b3b25883c7717ff7ca1ab31022c9c9"
dependencies = [
"bytes",
"fnv",
@@ -923,6 +973,15 @@ dependencies = [
"serde",
]
+[[package]]
+name = "inout"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a0c10553d664a4d0bcff9f4215d0aac67a639cc68ef660840afe309b807bc9f5"
+dependencies = [
+ "generic-array",
+]
+
[[package]]
name = "ipnet"
version = "2.9.0"
@@ -1085,9 +1144,9 @@ dependencies = [
[[package]]
name = "mio"
-version = "0.8.10"
+version = "0.8.11"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "8f3d0b296e374a4e6f3c7b0a1f5a51d748a0d34c85e7dc48fc3fa9a87657fe09"
+checksum = "a4a650543ca06a924e8b371db273b2756685faae30f8487da1b56505a8f78b0c"
dependencies = [
"libc",
"wasi",
@@ -1123,6 +1182,12 @@ dependencies = [
"libc",
]
+[[package]]
+name = "no-std-net"
+version = "0.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "43794a0ace135be66a25d3ae77d41b91615fb68ae937f904090203e81f755b65"
+
[[package]]
name = "nom"
version = "7.1.3"
@@ -1217,6 +1282,12 @@ version = "1.19.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3fdb12b2476b595f9358c5161aa467c2438859caa136dec86c26fdd2efe17b92"
+[[package]]
+name = "opaque-debug"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5"
+
[[package]]
name = "openssl"
version = "0.10.62"
@@ -1363,6 +1434,36 @@ version = "0.3.28"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "69d3587f8a9e599cc7ec2c00e331f71c4e69a5f9a4b8a6efd5b07466b9736f9a"
+[[package]]
+name = "pnet_base"
+version = "0.31.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f9d3a993d49e5fd5d4d854d6999d4addca1f72d86c65adf224a36757161c02b6"
+dependencies = [
+ "no-std-net",
+]
+
+[[package]]
+name = "pnet_macros"
+version = "0.31.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "48dd52a5211fac27e7acb14cfc9f30ae16ae0e956b7b779c8214c74559cef4c3"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "regex",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "pnet_macros_support"
+version = "0.31.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "89de095dc7739349559913aed1ef6a11e73ceade4897dadc77c5e09de6740750"
+dependencies = [
+ "pnet_base",
+]
+
[[package]]
name = "poem"
version = "1.3.59"
@@ -1406,6 +1507,17 @@ dependencies = [
"syn 2.0.48",
]
+[[package]]
+name = "poly1305"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8159bd90725d2df49889a078b54f4f79e87f1f8a8444194cdca81d38f5393abf"
+dependencies = [
+ "cpufeatures",
+ "opaque-debug",
+ "universal-hash",
+]
+
[[package]]
name = "powerfmt"
version = "0.2.0"
@@ -1688,6 +1800,15 @@ version = "1.0.16"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f98d2aa92eebf49b69786be48e4477826b256916e84a57ff2a4f21923b48eb4c"
+[[package]]
+name = "salsa20"
+version = "0.10.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "97a22f5af31f73a954c10289c93e8a50cc23d971e80ee446f1f6f7137a088213"
+dependencies = [
+ "cipher",
+]
+
[[package]]
name = "schannel"
version = "0.1.23"
@@ -2526,6 +2647,16 @@ version = "0.1.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "39ec24b3121d976906ece63c9daad25b85969647682eee313cb5779fdd69e14e"
+[[package]]
+name = "universal-hash"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fc1de2c688dc15305988b563c3854064043356019f97a4b46276fe734c4f07ea"
+dependencies = [
+ "crypto-common",
+ "subtle",
+]
+
[[package]]
name = "untrusted"
version = "0.7.1"
@@ -2588,6 +2719,12 @@ version = "0.11.0+wasi-snapshot-preview1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423"
+[[package]]
+name = "wasite"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b8dad83b4f25e74f184f64c43b150b91efe7647395b42289f38e50566d82855b"
+
[[package]]
name = "wasm-bindgen"
version = "0.2.89"
@@ -2705,9 +2842,13 @@ dependencies = [
[[package]]
name = "whoami"
-version = "1.4.1"
+version = "1.5.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
-checksum = "22fc3756b8a9133049b26c7f61ab35416c130e8c09b660f5b3958b446f52cc50"
+checksum = "0fec781d48b41f8163426ed18e8fc2864c12937df9ce54c88ede7bd47270893e"
+dependencies = [
+ "redox_syscall",
+ "wasite",
+]
[[package]]
name = "wildmatch"
diff --git a/Cargo.toml b/Cargo.toml
index f14d9b7d..45060d45 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -1,13 +1,14 @@
[package]
name = "chorus"
description = "A library for interacting with multiple Spacebar-compatible Instances at once."
-version = "0.14.0"
-license = "AGPL-3.0"
+version = "0.15.0"
+license = "MPL-2.0"
edition = "2021"
repository = "https://github.com/polyphony-chat/chorus"
readme = "README.md"
keywords = ["spacebar", "discord", "polyphony"]
website = ["https://discord.com/invite/m3FpcapGDD"]
+rust-version = "1.67.1"
[features]
@@ -16,6 +17,9 @@ backend = ["dep:poem", "dep:sqlx"]
rt-multi-thread = ["tokio/rt-multi-thread"]
rt = ["tokio/rt"]
client = []
+voice = ["voice_udp", "voice_gateway"]
+voice_udp = ["dep:discortp", "dep:crypto_secretbox"]
+voice_gateway = []
[dependencies]
tokio = { version = "1.35.1", features = ["macros", "sync"] }
@@ -24,10 +28,7 @@ serde_json = { version = "1.0.111", features = ["raw_value"] }
serde-aux = "4.3.1"
serde_with = "3.4.0"
serde_repr = "0.1.18"
-reqwest = { features = [
- "multipart",
- "json",
-], version = "0.11.23" }
+reqwest = { features = ["multipart", "json"], version = "0.11.23" }
url = "2.5.0"
chrono = { version = "0.4.31", features = ["serde"] }
regex = "1.10.2"
@@ -52,6 +53,8 @@ sqlx = { version = "0.7.3", features = [
"runtime-tokio-native-tls",
"any",
], optional = true }
+discortp = { version = "0.5.0", optional = true, features = ["rtp", "discord", "demux"] }
+crypto_secretbox = { version = "0.1.1", optional = true }
rand = "0.8.5"
[target.'cfg(not(target_arch = "wasm32"))'.dependencies]
@@ -63,6 +66,7 @@ tokio-tungstenite = { version = "0.20.1", features = [
] }
native-tls = "0.2.11"
hostname = "0.3.1"
+getrandom = { version = "0.2.12" }
[target.'cfg(target_arch = "wasm32")'.dependencies]
getrandom = { version = "0.2.12", features = ["js"] }
diff --git a/LICENSE b/LICENSE
index 0ad25db4..a612ad98 100644
--- a/LICENSE
+++ b/LICENSE
@@ -1,661 +1,373 @@
- 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
-.
+Mozilla Public License Version 2.0
+==================================
+
+1. Definitions
+--------------
+
+1.1. "Contributor"
+ means each individual or legal entity that creates, contributes to
+ the creation of, or owns Covered Software.
+
+1.2. "Contributor Version"
+ means the combination of the Contributions of others (if any) used
+ by a Contributor and that particular Contributor's Contribution.
+
+1.3. "Contribution"
+ means Covered Software of a particular Contributor.
+
+1.4. "Covered Software"
+ means Source Code Form to which the initial Contributor has attached
+ the notice in Exhibit A, the Executable Form of such Source Code
+ Form, and Modifications of such Source Code Form, in each case
+ including portions thereof.
+
+1.5. "Incompatible With Secondary Licenses"
+ means
+
+ (a) that the initial Contributor has attached the notice described
+ in Exhibit B to the Covered Software; or
+
+ (b) that the Covered Software was made available under the terms of
+ version 1.1 or earlier of the License, but not also under the
+ terms of a Secondary License.
+
+1.6. "Executable Form"
+ means any form of the work other than Source Code Form.
+
+1.7. "Larger Work"
+ means a work that combines Covered Software with other material, in
+ a separate file or files, that is not Covered Software.
+
+1.8. "License"
+ means this document.
+
+1.9. "Licensable"
+ means having the right to grant, to the maximum extent possible,
+ whether at the time of the initial grant or subsequently, any and
+ all of the rights conveyed by this License.
+
+1.10. "Modifications"
+ means any of the following:
+
+ (a) any file in Source Code Form that results from an addition to,
+ deletion from, or modification of the contents of Covered
+ Software; or
+
+ (b) any new file in Source Code Form that contains any Covered
+ Software.
+
+1.11. "Patent Claims" of a Contributor
+ means any patent claim(s), including without limitation, method,
+ process, and apparatus claims, in any patent Licensable by such
+ Contributor that would be infringed, but for the grant of the
+ License, by the making, using, selling, offering for sale, having
+ made, import, or transfer of either its Contributions or its
+ Contributor Version.
+
+1.12. "Secondary License"
+ means either the GNU General Public License, Version 2.0, the GNU
+ Lesser General Public License, Version 2.1, the GNU Affero General
+ Public License, Version 3.0, or any later versions of those
+ licenses.
+
+1.13. "Source Code Form"
+ means the form of the work preferred for making modifications.
+
+1.14. "You" (or "Your")
+ means an individual or a legal entity exercising rights under this
+ License. For legal entities, "You" includes any entity that
+ controls, is controlled by, or is under common control with You. For
+ purposes of this definition, "control" means (a) the power, direct
+ or indirect, to cause the direction or management of such entity,
+ whether by contract or otherwise, or (b) ownership of more than
+ fifty percent (50%) of the outstanding shares or beneficial
+ ownership of such entity.
+
+2. License Grants and Conditions
+--------------------------------
+
+2.1. Grants
+
+Each Contributor hereby grants You a world-wide, royalty-free,
+non-exclusive license:
+
+(a) under intellectual property rights (other than patent or trademark)
+ Licensable by such Contributor to use, reproduce, make available,
+ modify, display, perform, distribute, and otherwise exploit its
+ Contributions, either on an unmodified basis, with Modifications, or
+ as part of a Larger Work; and
+
+(b) under Patent Claims of such Contributor to make, use, sell, offer
+ for sale, have made, import, and otherwise transfer either its
+ Contributions or its Contributor Version.
+
+2.2. Effective Date
+
+The licenses granted in Section 2.1 with respect to any Contribution
+become effective for each Contribution on the date the Contributor first
+distributes such Contribution.
+
+2.3. Limitations on Grant Scope
+
+The licenses granted in this Section 2 are the only rights granted under
+this License. No additional rights or licenses will be implied from the
+distribution or licensing of Covered Software under this License.
+Notwithstanding Section 2.1(b) above, no patent license is granted by a
+Contributor:
+
+(a) for any code that a Contributor has removed from Covered Software;
+ or
+
+(b) for infringements caused by: (i) Your and any other third party's
+ modifications of Covered Software, or (ii) the combination of its
+ Contributions with other software (except as part of its Contributor
+ Version); or
+
+(c) under Patent Claims infringed by Covered Software in the absence of
+ its Contributions.
+
+This License does not grant any rights in the trademarks, service marks,
+or logos of any Contributor (except as may be necessary to comply with
+the notice requirements in Section 3.4).
+
+2.4. Subsequent Licenses
+
+No Contributor makes additional grants as a result of Your choice to
+distribute the Covered Software under a subsequent version of this
+License (see Section 10.2) or under the terms of a Secondary License (if
+permitted under the terms of Section 3.3).
+
+2.5. Representation
+
+Each Contributor represents that the Contributor believes its
+Contributions are its original creation(s) or it has sufficient rights
+to grant the rights to its Contributions conveyed by this License.
+
+2.6. Fair Use
+
+This License is not intended to limit any rights You have under
+applicable copyright doctrines of fair use, fair dealing, or other
+equivalents.
+
+2.7. Conditions
+
+Sections 3.1, 3.2, 3.3, and 3.4 are conditions of the licenses granted
+in Section 2.1.
+
+3. Responsibilities
+-------------------
+
+3.1. Distribution of Source Form
+
+All distribution of Covered Software in Source Code Form, including any
+Modifications that You create or to which You contribute, must be under
+the terms of this License. You must inform recipients that the Source
+Code Form of the Covered Software is governed by the terms of this
+License, and how they can obtain a copy of this License. You may not
+attempt to alter or restrict the recipients' rights in the Source Code
+Form.
+
+3.2. Distribution of Executable Form
+
+If You distribute Covered Software in Executable Form then:
+
+(a) such Covered Software must also be made available in Source Code
+ Form, as described in Section 3.1, and You must inform recipients of
+ the Executable Form how they can obtain a copy of such Source Code
+ Form by reasonable means in a timely manner, at a charge no more
+ than the cost of distribution to the recipient; and
+
+(b) You may distribute such Executable Form under the terms of this
+ License, or sublicense it under different terms, provided that the
+ license for the Executable Form does not attempt to limit or alter
+ the recipients' rights in the Source Code Form under this License.
+
+3.3. Distribution of a Larger Work
+
+You may create and distribute a Larger Work under terms of Your choice,
+provided that You also comply with the requirements of this License for
+the Covered Software. If the Larger Work is a combination of Covered
+Software with a work governed by one or more Secondary Licenses, and the
+Covered Software is not Incompatible With Secondary Licenses, this
+License permits You to additionally distribute such Covered Software
+under the terms of such Secondary License(s), so that the recipient of
+the Larger Work may, at their option, further distribute the Covered
+Software under the terms of either this License or such Secondary
+License(s).
+
+3.4. Notices
+
+You may not remove or alter the substance of any license notices
+(including copyright notices, patent notices, disclaimers of warranty,
+or limitations of liability) contained within the Source Code Form of
+the Covered Software, except that You may alter any license notices to
+the extent required to remedy known factual inaccuracies.
+
+3.5. Application of Additional Terms
+
+You may choose to offer, and to charge a fee for, warranty, support,
+indemnity or liability obligations to one or more recipients of Covered
+Software. However, You may do so only on Your own behalf, and not on
+behalf of any Contributor. You must make it absolutely clear that any
+such warranty, support, indemnity, or liability obligation is offered by
+You alone, and You hereby agree to indemnify every Contributor for any
+liability incurred by such Contributor as a result of warranty, support,
+indemnity or liability terms You offer. You may include additional
+disclaimers of warranty and limitations of liability specific to any
+jurisdiction.
+
+4. Inability to Comply Due to Statute or Regulation
+---------------------------------------------------
+
+If it is impossible for You to comply with any of the terms of this
+License with respect to some or all of the Covered Software due to
+statute, judicial order, or regulation then You must: (a) comply with
+the terms of this License to the maximum extent possible; and (b)
+describe the limitations and the code they affect. Such description must
+be placed in a text file included with all distributions of the Covered
+Software under this License. Except to the extent prohibited by statute
+or regulation, such description must be sufficiently detailed for a
+recipient of ordinary skill to be able to understand it.
+
+5. Termination
+--------------
+
+5.1. The rights granted under this License will terminate automatically
+if You fail to comply with any of its terms. However, if You become
+compliant, then the rights granted under this License from a particular
+Contributor are reinstated (a) provisionally, unless and until such
+Contributor explicitly and finally terminates Your grants, and (b) on an
+ongoing basis, if such Contributor fails to notify You of the
+non-compliance by some reasonable means prior to 60 days after You have
+come back into compliance. Moreover, Your grants from a particular
+Contributor are reinstated on an ongoing basis if such Contributor
+notifies You of the non-compliance by some reasonable means, this is the
+first time You have received notice of non-compliance with this License
+from such Contributor, and You become compliant prior to 30 days after
+Your receipt of the notice.
+
+5.2. If You initiate litigation against any entity by asserting a patent
+infringement claim (excluding declaratory judgment actions,
+counter-claims, and cross-claims) alleging that a Contributor Version
+directly or indirectly infringes any patent, then the rights granted to
+You by any and all Contributors for the Covered Software under Section
+2.1 of this License shall terminate.
+
+5.3. In the event of termination under Sections 5.1 or 5.2 above, all
+end user license agreements (excluding distributors and resellers) which
+have been validly granted by You or Your distributors under this License
+prior to termination shall survive termination.
+
+************************************************************************
+* *
+* 6. Disclaimer of Warranty *
+* ------------------------- *
+* *
+* Covered Software is provided under this License on an "as is" *
+* basis, without warranty of any kind, either expressed, implied, or *
+* statutory, including, without limitation, warranties that the *
+* Covered Software is free of defects, merchantable, fit for a *
+* particular purpose or non-infringing. The entire risk as to the *
+* quality and performance of the Covered Software is with You. *
+* Should any Covered Software prove defective in any respect, You *
+* (not any Contributor) assume the cost of any necessary servicing, *
+* repair, or correction. This disclaimer of warranty constitutes an *
+* essential part of this License. No use of any Covered Software is *
+* authorized under this License except under this disclaimer. *
+* *
+************************************************************************
+
+************************************************************************
+* *
+* 7. Limitation of Liability *
+* -------------------------- *
+* *
+* Under no circumstances and under no legal theory, whether tort *
+* (including negligence), contract, or otherwise, shall any *
+* Contributor, or anyone who distributes Covered Software as *
+* permitted above, be liable to You for any direct, indirect, *
+* special, incidental, or consequential damages of any character *
+* including, without limitation, damages for lost profits, loss of *
+* goodwill, work stoppage, computer failure or malfunction, or any *
+* and all other commercial damages or losses, even if such party *
+* shall have been informed of the possibility of such damages. This *
+* limitation of liability shall not apply to liability for death or *
+* personal injury resulting from such party's negligence to the *
+* extent applicable law prohibits such limitation. Some *
+* jurisdictions do not allow the exclusion or limitation of *
+* incidental or consequential damages, so this exclusion and *
+* limitation may not apply to You. *
+* *
+************************************************************************
+
+8. Litigation
+-------------
+
+Any litigation relating to this License may be brought only in the
+courts of a jurisdiction where the defendant maintains its principal
+place of business and such litigation shall be governed by laws of that
+jurisdiction, without reference to its conflict-of-law provisions.
+Nothing in this Section shall prevent a party's ability to bring
+cross-claims or counter-claims.
+
+9. Miscellaneous
+----------------
+
+This License represents the complete agreement concerning the subject
+matter hereof. If any provision of this License is held to be
+unenforceable, such provision shall be reformed only to the extent
+necessary to make it enforceable. Any law or regulation which provides
+that the language of a contract shall be construed against the drafter
+shall not be used to construe this License against a Contributor.
+
+10. Versions of the License
+---------------------------
+
+10.1. New Versions
+
+Mozilla Foundation is the license steward. Except as provided in Section
+10.3, no one other than the license steward has the right to modify or
+publish new versions of this License. Each version will be given a
+distinguishing version number.
+
+10.2. Effect of New Versions
+
+You may distribute the Covered Software under the terms of the version
+of the License under which You originally received the Covered Software,
+or under the terms of any subsequent version published by the license
+steward.
+
+10.3. Modified Versions
+
+If you create software not governed by this License, and you want to
+create a new license for such software, you may create and use a
+modified version of this License if you rename the license and remove
+any references to the name of the license steward (except to note that
+such modified license differs from this License).
+
+10.4. Distributing Source Code Form that is Incompatible With Secondary
+Licenses
+
+If You choose to distribute Source Code Form that is Incompatible With
+Secondary Licenses under the terms of this version of the License, the
+notice described in Exhibit B of this License must be attached.
+
+Exhibit A - Source Code Form License Notice
+-------------------------------------------
+
+ This Source Code Form is subject to the terms of the Mozilla Public
+ License, v. 2.0. If a copy of the MPL was not distributed with this
+ file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+If it is not possible or desirable to put the notice in a particular
+file, then You may include the notice in a location (such as a LICENSE
+file in a relevant directory) where a recipient would be likely to look
+for such a notice.
+
+You may add additional accurate notices of copyright ownership.
+
+Exhibit B - "Incompatible With Secondary Licenses" Notice
+---------------------------------------------------------
+
+ This Source Code Form is "Incompatible With Secondary Licenses", as
+ defined by the Mozilla Public License, v. 2.0.
diff --git a/README.md b/README.md
index f512f0a6..dc492304 100644
--- a/README.md
+++ b/README.md
@@ -44,7 +44,7 @@ To get started with Chorus, import it into your project by adding the following
```toml
[dependencies]
-chorus = "0.14.0"
+chorus = "0.15.0"
```
### Establishing a Connection
@@ -53,16 +53,10 @@ To connect to a Spacebar compatible server, you need to create an [`Instance`](h
```rs
use chorus::instance::Instance;
-use chorus::UrlBundle;
#[tokio::main]
async fn main() {
- let bundle = UrlBundle::new(
- "https://example.com/api".to_string(),
- "wss://example.com/".to_string(),
- "https://example.com/cdn".to_string(),
- );
- let instance = Instance::new(bundle)
+ let instance = Instance::new("https://example.com")
.await
.expect("Failed to connect to the Spacebar server");
// You can create as many instances of `Instance` as you want, but each `Instance` should likely be unique.
@@ -87,7 +81,7 @@ let login_schema = LoginSchema {
password: "Correct-Horse-Battery-Staple".to_string(),
..Default::default()
};
-// Each user connects to the Gateway. The Gateway connection lives on a seperate thread. Depending on
+// Each user connects to the Gateway. The Gateway connection lives on a separate thread. Depending on
// the runtime feature you choose, this can potentially take advantage of all of your computers' threads.
let user = instance
.login_account(login_schema)
@@ -125,7 +119,7 @@ like "proxy connection checking" are already disabled on this version, which oth
### wasm
To test for wasm, you will need to `cargo install wasm-pack`. You can then run
-`wasm-pack test -- --headless -- --target wasm32-unknown-unknown --features="rt, client" --no-default-features`
+`wasm-pack test -- --headless -- --target wasm32-unknown-unknown --features="rt, client, voice_gateway" --no-default-features`
to run the tests for wasm.
## Versioning
@@ -134,11 +128,7 @@ This crate uses Semantic Versioning 2.0.0 as its versioning scheme. You can read
## Contributing
-Chorus is currently missing voice support and a lot of API endpoints, many of which should be trivial to implement,
-ever since [we streamlined the process of doing so](https://github.com/polyphony-chat/chorus/discussions/401).
-
-If you'd like to contribute new functionality, check out [The 'Meta'-issues.](https://github.com/polyphony-chat/chorus/issues?q=is%3Aissue+label%3A%22Type%3A+Meta%22+) They contain a comprehensive list of all features which are yet missing for full Discord.com compatibility.
-Please feel free to open an Issue with the idea you have, or a Pull Request. Please keep our [contribution guidelines](https://github.com/polyphony-chat/.github/blob/main/CONTRIBUTION_GUIDELINES.md) in mind. Your contribution might not be accepted if it violates these guidelines or [our Code of Conduct](https://github.com/polyphony-chat/.github/blob/main/CODE_OF_CONDUCT.md).
+See [CONTRIBUTING.md](./CONTRIBUTING.md).
Progress Tracker/Roadmap
diff --git a/chorus-macros/src/lib.rs b/chorus-macros/src/lib.rs
index fd68df03..ba8f27e6 100644
--- a/chorus-macros/src/lib.rs
+++ b/chorus-macros/src/lib.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use proc_macro::TokenStream;
use quote::quote;
use syn::{parse_macro_input, Data, DeriveInput, Field, Fields, FieldsNamed};
diff --git a/examples/gateway_observers.rs b/examples/gateway_observers.rs
index a26ecfb2..0f15759b 100644
--- a/examples/gateway_observers.rs
+++ b/examples/gateway_observers.rs
@@ -1,3 +1,16 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+// This example showcase how to properly use gateway observers.
+//
+// To properly run it, you will need to change the token below.
+
+const TOKEN: &str = "";
+
+/// Find the gateway websocket url of the server we want to connect to
+const GATEWAY_URL: &str = "wss://gateway.old.server.spacebar.chat/";
+
use async_trait::async_trait;
use chorus::gateway::Gateway;
use chorus::{
@@ -32,11 +45,10 @@ impl Observer for ExampleObserver {
#[tokio::main(flavor = "current_thread")]
async fn main() {
- // Find the gateway websocket url of the server we want to connect to
- let websocket_url_spacebar = "wss://gateway.old.server.spacebar.chat/".to_string();
+ let gateway_websocket_url = GATEWAY_URL.to_string();
// Initiate the gateway connection
- let gateway = Gateway::spawn(websocket_url_spacebar).await.unwrap();
+ let gateway = Gateway::spawn(gateway_websocket_url).await.unwrap();
// Create an instance of our observer
let observer = ExampleObserver {};
@@ -55,7 +67,7 @@ async fn main() {
.subscribe(shared_observer);
// Authenticate so we will receive any events
- let token = "SecretToken".to_string();
+ let token = TOKEN.to_string();
let mut identify = GatewayIdentifyPayload::common();
identify.token = token;
gateway.send_identify(identify).await;
diff --git a/examples/gateway_simple.rs b/examples/gateway_simple.rs
index affa850b..e8ff59a0 100644
--- a/examples/gateway_simple.rs
+++ b/examples/gateway_simple.rs
@@ -1,3 +1,17 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+// This example showcases how to initiate a gateway connection manually
+// (e. g. not through ChorusUser)
+//
+// To properly run it, you will need to modify the token below.
+
+const TOKEN: &str = "";
+
+/// Find the gateway websocket url of the server we want to connect to
+const GATEWAY_URL: &str = "wss://gateway.old.server.spacebar.chat/";
+
use std::time::Duration;
use chorus::gateway::Gateway;
@@ -11,16 +25,15 @@ use wasmtimer::tokio::sleep;
/// This example creates a simple gateway connection and a session with an Identify event
#[tokio::main(flavor = "current_thread")]
async fn main() {
- // Find the gateway websocket url of the server we want to connect to
- let websocket_url_spacebar = "wss://gateway.old.server.spacebar.chat/".to_string();
+ let gateway_websocket_url = GATEWAY_URL.to_string();
// Initiate the gateway connection, starting a listener in one thread and a heartbeat handler in another
- let gateway = Gateway::spawn(websocket_url_spacebar).await.unwrap();
+ let gateway = Gateway::spawn(gateway_websocket_url).await.unwrap();
// At this point, we are connected to the server and are sending heartbeats, however we still haven't authenticated
// Get a token for an account on the server
- let token = "SecretToken".to_string();
+ let token = TOKEN.to_string();
// Create an identify event
// An Identify event is how the server authenticates us and gets info about our os and browser, along with our intents / capabilities
@@ -31,8 +44,8 @@ async fn main() {
identify.token = token;
// Send off the event
- gateway.send_identify(identify).await;
-
+ gateway.send_identify(identify).await;
+
// Do something on the main thread so we don't quit
loop {
sleep(Duration::from_secs(3600)).await;
diff --git a/examples/instance.rs b/examples/instance.rs
index 2ec45a3b..0bbdc17c 100644
--- a/examples/instance.rs
+++ b/examples/instance.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use chorus::instance::Instance;
#[tokio::main(flavor = "current_thread")]
diff --git a/examples/login.rs b/examples/login.rs
index 144030b8..e89d8d20 100644
--- a/examples/login.rs
+++ b/examples/login.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use chorus::instance::Instance;
use chorus::types::LoginSchema;
@@ -13,7 +17,7 @@ async fn main() {
password: "Correct-Horse-Battery-Staple".to_string(),
..Default::default()
};
- // Each user connects to the Gateway. The Gateway connection lives on a seperate thread. Depending on
+ // Each user connects to the Gateway. The Gateway connection lives on a separate thread. Depending on
// the runtime feature you choose, this can potentially take advantage of all of your computers' threads.
let user = instance
.login_account(login_schema)
diff --git a/examples/voice_simple/Cargo.toml b/examples/voice_simple/Cargo.toml
new file mode 100644
index 00000000..dc488e66
--- /dev/null
+++ b/examples/voice_simple/Cargo.toml
@@ -0,0 +1,13 @@
+[package]
+name = "voice_simple"
+version = "0.1.0"
+edition = "2021"
+
+# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
+
+[dependencies]
+async-trait = "*"
+chorus = { path = "../../", features = ["rt", "client", "voice"] }
+tokio = { version = "*", features = ["full"] }
+simplelog = "*"
+log = "*"
diff --git a/examples/voice_simple/src/main.rs b/examples/voice_simple/src/main.rs
new file mode 100644
index 00000000..a6f77db7
--- /dev/null
+++ b/examples/voice_simple/src/main.rs
@@ -0,0 +1,311 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+// This example showcases how to use the voice udp channel.
+//
+// To use this to properly communicate with voice, you will need to bring your own opus bindings
+// along with potentially sending some other events, like Speaking
+//
+// To properly run this example, you will need to change some values below,
+// like the token, guild and channel ids.
+
+const TOKEN: &str = "";
+
+const VOICE_GUILD_ID: Option = None;
+const VOICE_CHANNEL_ID: Option = Some(Snowflake(0_u64));
+
+const GATEWAY_URL: &str = "wss://gateway.discord.gg";
+
+use async_trait::async_trait;
+use simplelog::{TermLogger, Config, WriteLogger};
+use std::{net::SocketAddrV4, sync::Arc, fs::File, time::Duration};
+
+use chorus::{
+ gateway::{Observer, Gateway},
+ types::{
+ GatewayReady, SelectProtocol, SelectProtocolData, SessionDescription, Snowflake, Speaking,
+ SpeakingBitflags, SsrcDefinition, VoiceEncryptionMode, VoiceIdentify, VoiceProtocol,
+ VoiceReady, VoiceServerUpdate, GatewayIdentifyPayload, UpdateVoiceState,
+ },
+ voice::{
+ gateway::{VoiceGateway, VoiceGatewayHandle},
+ udp::{UdpHandle, UdpHandler},
+ voice_data::VoiceData,
+ },
+};
+use log::{info, LevelFilter};
+use tokio::sync::{Mutex, RwLock};
+
+extern crate chorus;
+extern crate tokio;
+
+/// Handles in between connections between the gateway and UDP modules
+#[derive(Debug, Clone)]
+pub struct VoiceHandler {
+ pub voice_gateway_connection: Arc>>,
+ pub voice_udp_connection: Arc>>,
+ pub data: Arc>,
+}
+
+impl VoiceHandler {
+ /// Creates a new [VoiceHandler], only initializing the data
+ pub fn new() -> VoiceHandler {
+ Self {
+ data: Arc::new(RwLock::new(VoiceData::default())),
+ voice_gateway_connection: Arc::new(Mutex::new(None)),
+ voice_udp_connection: Arc::new(Mutex::new(None)),
+ }
+ }
+}
+
+impl Default for VoiceHandler {
+ fn default() -> Self {
+ Self::new()
+ }
+}
+
+#[async_trait]
+// On [VoiceServerUpdate] we get our starting data and URL for the voice gateway server.
+impl Observer for VoiceHandler {
+ async fn update(&self, data: &VoiceServerUpdate) {
+ let mut data_lock = self.data.write().await;
+
+ data_lock.server_data = Some(data.clone());
+ let user_id = data_lock.user_id;
+ let session_id = data_lock.session_id.clone();
+
+ drop(data_lock);
+
+ // Create and connect to the voice gateway
+ let voice_gateway_handle = VoiceGateway::spawn(data.endpoint.clone().unwrap())
+ .await
+ .unwrap();
+
+ let server_id: Snowflake;
+
+ if data.guild_id.is_some() {
+ server_id = data.guild_id.unwrap();
+ } else {
+ server_id = data.channel_id.unwrap();
+ }
+
+ let voice_identify = VoiceIdentify {
+ server_id,
+ user_id,
+ session_id,
+ token: data.token.clone(),
+ video: Some(false),
+ };
+
+ voice_gateway_handle.send_identify(voice_identify).await;
+
+ let cloned_gateway_handle = voice_gateway_handle.clone();
+
+ let mut voice_events = cloned_gateway_handle.events.lock().await;
+
+ let self_reference = Arc::new(self.clone());
+
+ // Subscribe to voice gateway events
+ voice_events.voice_ready.subscribe(self_reference.clone());
+ voice_events
+ .session_description
+ .subscribe(self_reference.clone());
+ voice_events.speaking.subscribe(self_reference.clone());
+ voice_events
+ .ssrc_definition
+ .subscribe(self_reference.clone());
+
+ *self.voice_gateway_connection.lock().await = Some(voice_gateway_handle);
+ }
+}
+
+#[async_trait]
+// On [VoiceReady] we get info for establishing a UDP connection, and we immediately need said UDP
+// connection for ip discovery.
+impl Observer for VoiceHandler {
+ async fn update(&self, data: &VoiceReady) {
+ let mut data_lock = self.data.write().await;
+
+ data_lock.ready_data = Some(data.clone());
+
+ drop(data_lock);
+
+ // Create a udp connection and perform ip discovery
+ let udp_handle = UdpHandler::spawn(
+ self.data.clone(),
+ std::net::SocketAddr::V4(SocketAddrV4::new(data.ip, data.port)),
+ data.ssrc,
+ )
+ .await
+ .unwrap();
+
+ // Subscribe ourself to receiving rtp data
+ udp_handle
+ .events
+ .lock()
+ .await
+ .rtp
+ .subscribe(Arc::new(self.clone()));
+
+ let ip_discovery = self.data.read().await.ip_discovery.clone().unwrap();
+
+ *self.voice_udp_connection.lock().await = Some(udp_handle.clone());
+
+ let string_ip_address =
+ String::from_utf8(ip_discovery.address).expect("Ip discovery gave non string ip");
+
+ // Send a select protocol, which tells the server where we'll be receiving data and what
+ // mode to encrypt data in
+ self.voice_gateway_connection
+ .lock()
+ .await
+ .clone()
+ .unwrap()
+ .send_select_protocol(SelectProtocol {
+ protocol: VoiceProtocol::Udp,
+ data: SelectProtocolData {
+ address: string_ip_address,
+ port: ip_discovery.port,
+ // There are several other voice encryption modes available, though not all are
+ // implemented in chorus
+ mode: VoiceEncryptionMode::Xsalsa20Poly1305,
+ },
+ ..Default::default()
+ })
+ .await;
+ }
+}
+
+#[async_trait]
+// Session descryption gives us final info regarding codecs and our encryption key
+impl Observer for VoiceHandler {
+ async fn update(&self, data: &SessionDescription) {
+ let mut data_write = self.data.write().await;
+
+ data_write.session_description = Some(data.clone());
+
+ drop(data_write);
+ }
+}
+
+#[async_trait]
+// Ready is used just to obtain some info, like the user id and session id
+impl Observer for VoiceHandler {
+ async fn update(&self, data: &GatewayReady) {
+ let mut lock = self.data.write().await;
+ lock.user_id = data.user.id;
+ lock.session_id = data.session_id.clone();
+ drop(lock);
+ }
+}
+
+#[async_trait]
+// This is the received voice data
+impl Observer for VoiceHandler {
+ async fn update(&self, data: &chorus::voice::discortp::rtp::Rtp) {
+ info!(
+ "Received decrypted voice data! {:?} (SSRC: {})",
+ data.payload.clone(),
+ data.ssrc,
+ );
+ }
+}
+
+#[async_trait]
+// This event gives extra info about who is speaking
+impl Observer for VoiceHandler {
+ async fn update(&self, data: &Speaking) {
+ println!(
+ "Received Speaking! (SRRC: {}, flags: {:?})",
+ data.ssrc,
+ SpeakingBitflags::from_bits(data.speaking).unwrap()
+ );
+ }
+}
+
+#[async_trait]
+// This event gives some info about which user has which ssrc
+impl Observer for VoiceHandler {
+ async fn update(&self, data: &SsrcDefinition) {
+ println!(
+ "Received SSRC Definition! (User {} has audio ssrc {})",
+ data.user_id.unwrap(),
+ data.audio_ssrc
+ );
+ }
+}
+
+#[tokio::main]
+async fn main() {
+ simplelog::CombinedLogger::init(vec![
+ TermLogger::new(
+ LevelFilter::Debug,
+ Config::default(),
+ simplelog::TerminalMode::Mixed,
+ simplelog::ColorChoice::Auto,
+ ),
+ WriteLogger::new(
+ LevelFilter::Trace,
+ Config::default(),
+ File::create("latest.log").unwrap(),
+ ),
+ ])
+ .unwrap();
+
+ let gateway = Gateway::spawn(GATEWAY_URL.to_string())
+ .await
+ .unwrap();
+
+ let mut identify = GatewayIdentifyPayload::common();
+ identify.token = TOKEN.to_string();
+
+ gateway.send_identify(identify).await;
+
+ let voice_handler = Arc::new(VoiceHandler::new());
+
+ // Voice handler needs voice server update
+ gateway
+ .events
+ .lock()
+ .await
+ .voice
+ .server_update
+ .subscribe(voice_handler.clone());
+
+ // It also needs a bit of the data in ready
+ gateway
+ .events
+ .lock()
+ .await
+ .session
+ .ready
+ .subscribe(voice_handler.clone());
+
+ // Data which channel to update the local user to be joined into.
+ //
+ // guild_id and channel_id can be some to join guild voice channels
+ //
+ // guild_id can be none and channel id some to join dm calls
+ //
+ // both can be none to leave all voice channels
+ let voice_state_update = UpdateVoiceState {
+ guild_id: VOICE_GUILD_ID,
+ channel_id: VOICE_CHANNEL_ID,
+ self_mute: false,
+ self_deaf: false,
+ ..Default::default()
+ };
+
+ gateway.send_update_voice_state(voice_state_update).await;
+
+ loop {
+ tokio::time::sleep(Duration::from_millis(1000)).await;
+
+ // Potentially send some data here
+ /*let voice_udp_option = voice_handler.voice_udp_connection.lock().await.clone();
+ if voice_udp_option.is_some() {
+ voice_udp_option.unwrap().send_opus_data(0, vec![1, 2, 3, 4, 5]).await.unwrap();
+ }*/
+ }
+}
diff --git a/src/api/auth/login.rs b/src/api/auth/login.rs
index ff99be8b..7689af75 100644
--- a/src/api/auth/login.rs
+++ b/src/api/auth/login.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use std::sync::{Arc, RwLock};
use reqwest::Client;
diff --git a/src/api/auth/mod.rs b/src/api/auth/mod.rs
index ae3b2195..5bd539f6 100644
--- a/src/api/auth/mod.rs
+++ b/src/api/auth/mod.rs
@@ -1,6 +1,13 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use std::sync::{Arc, RwLock};
+#[allow(unused_imports)]
pub use login::*;
+
+#[allow(unused_imports)]
pub use register::*;
use crate::gateway::Gateway;
diff --git a/src/api/auth/register.rs b/src/api/auth/register.rs
index aa0b4838..deece4db 100644
--- a/src/api/auth/register.rs
+++ b/src/api/auth/register.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use std::sync::{Arc, RwLock};
use reqwest::Client;
diff --git a/src/api/channels/channels.rs b/src/api/channels/channels.rs
index 9560d746..6c415763 100644
--- a/src/api/channels/channels.rs
+++ b/src/api/channels/channels.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use reqwest::Client;
use serde_json::to_string;
diff --git a/src/api/channels/messages.rs b/src/api/channels/messages.rs
index 6dfdfbfc..feabc371 100644
--- a/src/api/channels/messages.rs
+++ b/src/api/channels/messages.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use http::header::CONTENT_DISPOSITION;
use http::HeaderMap;
use reqwest::{multipart, Client};
@@ -36,7 +40,7 @@ impl Message {
chorus_request.deserialize_response::(user).await
} else {
for (index, attachment) in message.attachments.iter_mut().enumerate() {
- attachment.get_mut(index).unwrap().set_id(index as i16);
+ attachment.get_mut(index).unwrap().id = Some(index as i16);
}
let mut form = reqwest::multipart::Form::new();
let payload_json = to_string(&message).unwrap();
@@ -45,8 +49,8 @@ impl Message {
form = form.part("payload_json", payload_field);
for (index, attachment) in message.attachments.unwrap().into_iter().enumerate() {
- let (attachment_content, current_attachment) = attachment.move_content();
- let (attachment_filename, _) = current_attachment.move_filename();
+ let attachment_content = attachment.content;
+ let attachment_filename = attachment.filename;
let part_name = format!("files[{}]", index);
let content_disposition = format!(
"form-data; name=\"{}\"'; filename=\"{}\"",
diff --git a/src/api/channels/mod.rs b/src/api/channels/mod.rs
index b72861e7..a8c3047b 100644
--- a/src/api/channels/mod.rs
+++ b/src/api/channels/mod.rs
@@ -1,3 +1,8 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+#![allow(unused_imports)]
pub use channels::*;
pub use messages::*;
pub use permissions::*;
diff --git a/src/api/channels/permissions.rs b/src/api/channels/permissions.rs
index 53608908..03465b80 100644
--- a/src/api/channels/permissions.rs
+++ b/src/api/channels/permissions.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use reqwest::Client;
use serde_json::to_string;
diff --git a/src/api/channels/reactions.rs b/src/api/channels/reactions.rs
index 94d30870..b7c42e10 100644
--- a/src/api/channels/reactions.rs
+++ b/src/api/channels/reactions.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use crate::{
errors::ChorusResult,
instance::ChorusUser,
diff --git a/src/api/guilds/guilds.rs b/src/api/guilds/guilds.rs
index b433c849..e2ff9bad 100644
--- a/src/api/guilds/guilds.rs
+++ b/src/api/guilds/guilds.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use reqwest::Client;
use serde_json::from_str;
use serde_json::to_string;
@@ -14,6 +18,25 @@ use crate::types::{
use crate::types::{GuildBan, Snowflake};
impl Guild {
+ /// Fetches a guild by its id.
+ ///
+ /// # Reference
+ /// See
+ pub async fn get(guild_id: Snowflake, user: &mut ChorusUser) -> ChorusResult {
+ let chorus_request = ChorusRequest {
+ request: Client::new()
+ .get(format!(
+ "{}/guilds/{}",
+ user.belongs_to.read().unwrap().urls.api,
+ guild_id
+ ))
+ .header("Authorization", user.token()),
+ limit_type: LimitType::Guild(guild_id),
+ };
+ let response = chorus_request.deserialize_response::(user).await?;
+ Ok(response)
+ }
+
/// Creates a new guild.
///
/// # Reference
@@ -34,6 +57,35 @@ impl Guild {
chorus_request.deserialize_response::(user).await
}
+ /// Modify a guild's settings.
+ ///
+ /// Requires the [MANAGE_GUILD](crate::types::PermissionFlags::MANAGE_GUILD) permission.
+ ///
+ /// Returns the updated guild.
+ ///
+ /// # Reference
+ ///
+ pub async fn modify(
+ guild_id: Snowflake,
+ schema: GuildModifySchema,
+ user: &mut ChorusUser,
+ ) -> ChorusResult {
+ let chorus_request = ChorusRequest {
+ request: Client::new()
+ .patch(format!(
+ "{}/guilds/{}",
+ user.belongs_to.read().unwrap().urls.api,
+ guild_id,
+ ))
+ .header("Authorization", user.token())
+ .header("Content-Type", "application/json")
+ .body(to_string(&schema).unwrap()),
+ limit_type: LimitType::Guild(guild_id),
+ };
+ let response = chorus_request.deserialize_response::(user).await?;
+ Ok(response)
+ }
+
/// Deletes a guild by its id.
///
/// User must be the owner.
@@ -123,77 +175,11 @@ impl Guild {
};
}
- /// Fetches a guild by its id.
+ /// Returns a guild preview object for the given guild ID.
///
- /// # Reference
- /// See
- pub async fn get(guild_id: Snowflake, user: &mut ChorusUser) -> ChorusResult {
- let chorus_request = ChorusRequest {
- request: Client::new()
- .get(format!(
- "{}/guilds/{}",
- user.belongs_to.read().unwrap().urls.api,
- guild_id
- ))
- .header("Authorization", user.token()),
- limit_type: LimitType::Guild(guild_id),
- };
- let response = chorus_request.deserialize_response::(user).await?;
- Ok(response)
- }
-
- pub async fn create_ban(
- guild_id: Snowflake,
- user_id: Snowflake,
- audit_log_reason: Option,
- schema: GuildBanCreateSchema,
- user: &mut ChorusUser,
- ) -> ChorusResult<()> {
- // FIXME: Return GuildBan instead of (). Requires to be resolved.
- let request = ChorusRequest::new(
- http::Method::PUT,
- format!(
- "{}/guilds/{}/bans/{}",
- user.belongs_to.read().unwrap().urls.api,
- guild_id,
- user_id
- )
- .as_str(),
- Some(to_string(&schema).unwrap()),
- audit_log_reason.as_deref(),
- None,
- Some(user),
- LimitType::Guild(guild_id),
- );
- request.handle_request_as_result(user).await
- }
-
- /// # Reference
- ///
- pub async fn modify(
- guild_id: Snowflake,
- schema: GuildModifySchema,
- user: &mut ChorusUser,
- ) -> ChorusResult {
- let chorus_request = ChorusRequest {
- request: Client::new()
- .patch(format!(
- "{}/guilds/{}",
- user.belongs_to.read().unwrap().urls.api,
- guild_id,
- ))
- .header("Authorization", user.token())
- .header("Content-Type", "application/json")
- .body(to_string(&schema).unwrap()),
- limit_type: LimitType::Guild(guild_id),
- };
- let response = chorus_request.deserialize_response::(user).await?;
- Ok(response)
- }
-
- /// Returns a guild preview object for the given guild ID. If the user is not in the guild, the guild must be discoverable.
- /// # Reference:
+ /// If the user is not in the guild, the guild must be discoverable.
///
+ /// # Reference:
/// See
pub async fn get_preview(
guild_id: Snowflake,
@@ -270,7 +256,9 @@ impl Guild {
request.deserialize_response::>(user).await
}
- /// Removes a member from a guild. Requires the KICK_MEMBERS permission. Returns a 204 empty response on success.
+ /// Removes a member from a guild.
+ ///
+ /// Requires the [KICK_MEMBERS](crate::types::PermissionFlags::KICK_MEMBERS) permission.
///
/// # Reference
/// See
@@ -383,7 +371,9 @@ impl Guild {
.await
}
- /// Returns a list of ban objects for the guild. Requires the `BAN_MEMBERS` permission.
+ /// Returns a list of ban objects for the guild.
+ ///
+ /// Requires the [BAN_MEMBERS](crate::types::PermissionFlags::BAN_MEMBERS) permission.
///
/// # Reference:
/// See
@@ -413,7 +403,9 @@ impl Guild {
request.deserialize_response::>(user).await
}
- /// Returns a ban object for the given user. Requires the `BAN_MEMBERS` permission.
+ /// Returns a ban object for the given user.
+ ///
+ /// Requires the [BAN_MEMBERS](crate::types::PermissionFlags::BAN_MEMBERS) permission.
///
/// # Reference:
/// See
@@ -441,7 +433,39 @@ impl Guild {
request.deserialize_response::(user).await
}
- /// Removes the ban for a user. Requires the BAN_MEMBERS permissions. Returns a 204 empty response on success.
+ /// Creates a ban from the guild.
+ ///
+ /// Requires the [BAN_MEMBERS](crate::types::PermissionFlags::BAN_MEMBERS) permission.
+ ///
+ pub async fn create_ban(
+ guild_id: Snowflake,
+ user_id: Snowflake,
+ audit_log_reason: Option,
+ schema: GuildBanCreateSchema,
+ user: &mut ChorusUser,
+ ) -> ChorusResult<()> {
+ // FIXME: Return GuildBan instead of (). Requires to be resolved.
+ let request = ChorusRequest::new(
+ http::Method::PUT,
+ format!(
+ "{}/guilds/{}/bans/{}",
+ user.belongs_to.read().unwrap().urls.api,
+ guild_id,
+ user_id
+ )
+ .as_str(),
+ Some(to_string(&schema).unwrap()),
+ audit_log_reason.as_deref(),
+ None,
+ Some(user),
+ LimitType::Guild(guild_id),
+ );
+ request.handle_request_as_result(user).await
+ }
+
+ /// Removes the ban for a user.
+ ///
+ /// Requires the [BAN_MEMBERS](crate::types::PermissionFlags::BAN_MEMBERS) permission.
///
/// # Reference:
/// See
diff --git a/src/api/guilds/member.rs b/src/api/guilds/member.rs
index 885ddf96..0037ec7d 100644
--- a/src/api/guilds/member.rs
+++ b/src/api/guilds/member.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use reqwest::Client;
use crate::{
diff --git a/src/api/guilds/messages.rs b/src/api/guilds/messages.rs
index 60fd4e36..b45f3e37 100644
--- a/src/api/guilds/messages.rs
+++ b/src/api/guilds/messages.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use crate::errors::ChorusResult;
use crate::instance::ChorusUser;
use crate::types::{Guild, Message, MessageSearchQuery, Snowflake};
@@ -9,7 +13,7 @@ impl Guild {
/// permission to be present on the current user.
///
/// If the guild/channel you are searching is not yet indexed, the endpoint will return a 202 accepted response.
- /// In this case, the method will return a [`ChorusError::InvalidResponse`] error.
+ /// In this case, the method will return a [`ChorusError::InvalidResponse`](crate::errors::ChorusError::InvalidResponse) error.
///
/// # Reference:
/// See
diff --git a/src/api/guilds/mod.rs b/src/api/guilds/mod.rs
index f1fa039f..e8304a11 100644
--- a/src/api/guilds/mod.rs
+++ b/src/api/guilds/mod.rs
@@ -1,3 +1,8 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+#![allow(unused_imports)]
pub use guilds::*;
pub use messages::*;
pub use roles::*;
diff --git a/src/api/guilds/roles.rs b/src/api/guilds/roles.rs
index f131367c..6100a48d 100644
--- a/src/api/guilds/roles.rs
+++ b/src/api/guilds/roles.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use reqwest::Client;
use serde_json::to_string;
diff --git a/src/api/invites/mod.rs b/src/api/invites/mod.rs
index 80b47d25..68f14178 100644
--- a/src/api/invites/mod.rs
+++ b/src/api/invites/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use reqwest::Client;
use serde_json::to_string;
diff --git a/src/api/mod.rs b/src/api/mod.rs
index ab3f9b9e..c9ca2792 100644
--- a/src/api/mod.rs
+++ b/src/api/mod.rs
@@ -1,4 +1,10 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
//! All of the API's endpoints.
+
+#![allow(unused_imports)]
pub use channels::messages::*;
pub use guilds::*;
pub use invites::*;
diff --git a/src/api/policies/instance/instance.rs b/src/api/policies/instance/instance.rs
index 4de5fd87..584db338 100644
--- a/src/api/policies/instance/instance.rs
+++ b/src/api/policies/instance/instance.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde_json::from_str;
use crate::errors::{ChorusError, ChorusResult};
diff --git a/src/api/policies/instance/mod.rs b/src/api/policies/instance/mod.rs
index b3a91486..e2b97e17 100644
--- a/src/api/policies/instance/mod.rs
+++ b/src/api/policies/instance/mod.rs
@@ -1,3 +1,8 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+#![allow(unused_imports)]
pub use instance::*;
pub mod instance;
diff --git a/src/api/policies/mod.rs b/src/api/policies/mod.rs
index 1d5ea990..377ec309 100644
--- a/src/api/policies/mod.rs
+++ b/src/api/policies/mod.rs
@@ -1 +1,5 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
pub mod instance;
diff --git a/src/api/users/channels.rs b/src/api/users/channels.rs
index 330b3e3c..44ed8975 100644
--- a/src/api/users/channels.rs
+++ b/src/api/users/channels.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use reqwest::Client;
use serde_json::to_string;
@@ -9,6 +13,26 @@ use crate::{
};
impl ChorusUser {
+ /// Fetches a list of private channels the user is in.
+ ///
+ /// # Reference:
+ /// See
+ pub async fn get_private_channels(&mut self) -> ChorusResult> {
+ let url = format!(
+ "{}/users/@me/channels",
+ self.belongs_to.read().unwrap().urls.api
+ );
+ ChorusRequest {
+ request: Client::new()
+ .get(url)
+ .header("Authorization", self.token())
+ .header("Content-Type", "application/json"),
+ limit_type: LimitType::Global,
+ }
+ .deserialize_response::>(self)
+ .await
+ }
+
/// Creates a DM channel or group DM channel.
///
/// One recipient creates or returns an existing DM channel,
diff --git a/src/api/users/guilds.rs b/src/api/users/guilds.rs
index 6ffcdfc3..aac2f7ad 100644
--- a/src/api/users/guilds.rs
+++ b/src/api/users/guilds.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use reqwest::Client;
use serde_json::to_string;
diff --git a/src/api/users/mod.rs b/src/api/users/mod.rs
index ba789bae..b11772ab 100644
--- a/src/api/users/mod.rs
+++ b/src/api/users/mod.rs
@@ -1,3 +1,8 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+#![allow(unused_imports)]
pub use channels::*;
pub use guilds::*;
pub use relationships::*;
diff --git a/src/api/users/relationships.rs b/src/api/users/relationships.rs
index 4f9602c1..c77db447 100644
--- a/src/api/users/relationships.rs
+++ b/src/api/users/relationships.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use reqwest::Client;
use serde_json::to_string;
@@ -15,7 +19,7 @@ impl ChorusUser {
/// Retrieves a list of mutual friends between the authenticated user and a given user.
///
/// # Reference
- /// See
+ /// See
pub async fn get_mutual_relationships(
&mut self,
user_id: Snowflake,
@@ -37,7 +41,7 @@ impl ChorusUser {
/// Retrieves the user's relationships.
///
/// # Reference
- /// See
+ /// See
pub async fn get_relationships(&mut self) -> ChorusResult> {
let url = format!(
"{}/users/@me/relationships",
@@ -55,7 +59,7 @@ impl ChorusUser {
/// Sends a friend request to a user.
///
/// # Reference
- /// See
+ /// See
pub async fn send_friend_request(
&mut self,
schema: FriendRequestSendSchema,
@@ -132,7 +136,7 @@ impl ChorusUser {
/// Removes the relationship between the authenticated user and a given user.
///
/// # Reference
- /// See
+ /// See
pub async fn remove_relationship(&mut self, user_id: Snowflake) -> ChorusResult<()> {
let url = format!(
"{}/users/@me/relationships/{}",
diff --git a/src/api/users/users.rs b/src/api/users/users.rs
index 0f31d6f7..b80bc1e2 100644
--- a/src/api/users/users.rs
+++ b/src/api/users/users.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use std::sync::{Arc, RwLock};
use reqwest::Client;
@@ -113,7 +117,7 @@ impl User {
/// Gets the user's settings.
///
/// # Reference
- /// See
+ /// See
pub async fn get_settings(
token: &String,
url_api: &String,
diff --git a/src/errors.rs b/src/errors.rs
index c20ac64b..a2f174d8 100644
--- a/src/errors.rs
+++ b/src/errors.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
//! Contains all the errors that can be returned by the library.
use custom_error::custom_error;
@@ -17,7 +21,7 @@ custom_error! {
/// Server did not respond.
NoResponse = "Did not receive a response from the Server.",
/// Reqwest returned an Error instead of a Response object.
- RequestFailed{url:String, error: String} = "An error occured while trying to GET from {url}: {error}",
+ RequestFailed{url:String, error: String} = "An error occurred while trying to GET from {url}: {error}",
/// Response received, however, it was not of the successful responses type. Used when no other, special case applies.
ReceivedErrorCode{error_code: u16, error: String} = "Received the following error code while requesting from the route: {error_code}",
/// Used when there is likely something wrong with the instance, the request was directed to.
@@ -63,7 +67,7 @@ custom_error! {
}
custom_error! {
- /// For errors we receive from the gateway, see https://discord-userdoccers.vercel.app/topics/opcodes-and-status-codes#gateway-close-event-codes;
+ /// For errors we receive from the gateway, see ;
///
/// Supposed to be sent as numbers, though they are sent as string most of the time?
///
@@ -96,3 +100,59 @@ custom_error! {
}
impl WebSocketEvent for GatewayError {}
+
+custom_error! {
+ /// Voice Gateway errors
+ ///
+ /// Similar to [GatewayError].
+ ///
+ /// See ;
+ #[derive(Clone, Default, PartialEq, Eq)]
+ pub VoiceGatewayError
+ // Errors we receive
+ #[default]
+ UnknownOpcode = "You sent an invalid opcode",
+ FailedToDecodePayload = "You sent an invalid payload in your identifying to the (Voice) Gateway",
+ NotAuthenticated = "You sent a payload before identifying with the (Voice) Gateway",
+ AuthenticationFailed = "The token you sent in your identify payload is incorrect",
+ AlreadyAuthenticated = "You sent more than one identify payload",
+ SessionNoLongerValid = "Your session is no longer valid",
+ SessionTimeout = "Your session has timed out",
+ ServerNotFound = "We can't find the server you're trying to connect to",
+ UnknownProtocol = "We didn't recognize the protocol you sent",
+ Disconnected = "Channel was deleted, you were kicked, voice server changed, or the main gateway session was dropped. Should not reconnect.",
+ VoiceServerCrashed = "The server crashed, try resuming",
+ UnknownEncryptionMode = "Server failed to decrypt data",
+
+ // Errors when initiating a gateway connection
+ CannotConnect{error: String} = "Cannot connect due to a tungstenite error: {error}",
+ NonHelloOnInitiate{opcode: u8} = "Received non hello on initial gateway connection ({opcode}), something is definitely wrong",
+
+ // Other misc errors
+ UnexpectedOpcodeReceived{opcode: u8} = "Received an opcode we weren't expecting to receive: {opcode}",
+}
+
+impl WebSocketEvent for VoiceGatewayError {}
+
+custom_error! {
+ /// Voice UDP errors.
+ #[derive(Clone, PartialEq, Eq)]
+ pub VoiceUdpError
+
+ // General errors
+ BrokenSocket{error: String} = "Could not write / read from UDP socket: {error}",
+ NoData = "We have not set received the necessary data to perform this operation.",
+
+ // Encryption errors
+ EncryptionModeNotImplemented{encryption_mode: String} = "Voice encryption mode {encryption_mode} is not yet implemented.",
+ NoKey = "Tried to encrypt / decrypt rtp data, but no key has been received yet",
+ FailedEncryption = "Tried to encrypt rtp data, but failed. Most likely this is an issue chorus' nonce generation. Please open an issue on the chorus github: https://github.com/polyphony-chat/chorus/issues/new",
+ FailedDecryption = "Tried to decrypt rtp data, but failed. Most likely this is an issue chorus' nonce generation. Please open an issue on the chorus github: https://github.com/polyphony-chat/chorus/issues/new",
+ FailedNonceGeneration{error: String} = "Tried to generate nonce, but failed due to error: {error}.",
+
+ // Errors when initiating a socket connection
+ CannotBind{error: String} = "Cannot bind socket due to a UDP error: {error}",
+ CannotConnect{error: String} = "Cannot connect due to a UDP error: {error}",
+}
+
+impl WebSocketEvent for VoiceUdpError {}
diff --git a/src/gateway/backends/mod.rs b/src/gateway/backends/mod.rs
index edb5dc97..6df070da 100644
--- a/src/gateway/backends/mod.rs
+++ b/src/gateway/backends/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
#[cfg(all(not(target_arch = "wasm32"), feature = "client"))]
pub mod tungstenite;
#[cfg(all(not(target_arch = "wasm32"), feature = "client"))]
diff --git a/src/gateway/backends/tungstenite.rs b/src/gateway/backends/tungstenite.rs
index 51843293..a9f9f64f 100644
--- a/src/gateway/backends/tungstenite.rs
+++ b/src/gateway/backends/tungstenite.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use futures_util::{
stream::{SplitSink, SplitStream},
StreamExt,
@@ -23,8 +27,16 @@ impl TungsteniteBackend {
websocket_url: &str,
) -> Result<(TungsteniteSink, TungsteniteStream), crate::errors::GatewayError> {
let mut roots = rustls::RootCertStore::empty();
- for cert in rustls_native_certs::load_native_certs().expect("could not load platform certs")
- {
+ let certs = rustls_native_certs::load_native_certs();
+
+ if let Err(e) = certs {
+ log::error!("Failed to load platform native certs! {:?}", e);
+ return Err(GatewayError::CannotConnect {
+ error: format!("{:?}", e),
+ });
+ }
+
+ for cert in certs.unwrap() {
roots.add(&rustls::Certificate(cert.0)).unwrap();
}
let (websocket_stream, _) = match connect_async_tls_with_config(
diff --git a/src/gateway/backends/wasm.rs b/src/gateway/backends/wasm.rs
index e9927ac4..83f4b37b 100644
--- a/src/gateway/backends/wasm.rs
+++ b/src/gateway/backends/wasm.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use futures_util::{
stream::{SplitSink, SplitStream},
StreamExt,
diff --git a/src/gateway/events.rs b/src/gateway/events.rs
index fdb7b253..8d38cca1 100644
--- a/src/gateway/events.rs
+++ b/src/gateway/events.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use super::*;
use crate::types;
@@ -42,6 +46,8 @@ pub struct Session {
pub ready: GatewayEvent,
pub ready_supplemental: GatewayEvent,
pub replace: GatewayEvent,
+ pub reconnect: GatewayEvent,
+ pub invalid: GatewayEvent,
}
#[derive(Default, Debug)]
diff --git a/src/gateway/gateway.rs b/src/gateway/gateway.rs
index 684b9d21..dabfeb67 100644
--- a/src/gateway/gateway.rs
+++ b/src/gateway/gateway.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use std::time::Duration;
use futures_util::{SinkExt, StreamExt};
@@ -5,13 +9,14 @@ use log::*;
#[cfg(not(target_arch = "wasm32"))]
use tokio::task;
-use self::event::Events;
+use super::events::Events;
use super::*;
use super::{Sink, Stream};
use crate::types::{
self, AutoModerationRule, AutoModerationRuleUpdate, Channel, ChannelCreate, ChannelDelete,
- ChannelUpdate, Guild, GuildRoleCreate, GuildRoleUpdate, JsonField, RoleObject, SourceUrlField,
- ThreadUpdate, UpdateMessage, WebSocketEvent,
+ ChannelUpdate, GatewayInvalidSession, GatewayReconnect, Guild, GuildRoleCreate,
+ GuildRoleUpdate, JsonField, RoleObject, SourceUrlField, ThreadUpdate, UpdateMessage,
+ WebSocketEvent,
};
#[derive(Debug)]
@@ -21,6 +26,7 @@ pub struct Gateway {
websocket_send: Arc>,
websocket_receive: Stream,
kill_send: tokio::sync::broadcast::Sender<()>,
+ kill_receive: tokio::sync::broadcast::Receiver<()>,
store: Arc>>>>,
url: String,
}
@@ -70,6 +76,7 @@ impl Gateway {
websocket_send: shared_websocket_send.clone(),
websocket_receive,
kill_send: kill_send.clone(),
+ kill_receive: kill_send.subscribe(),
store: store.clone(),
url: websocket_url.clone(),
};
@@ -94,14 +101,21 @@ impl Gateway {
}
/// The main gateway listener task;
- ///
- /// Can only be stopped by closing the websocket, cannot be made to listen for kill
pub async fn gateway_listen_task(&mut self) {
loop {
- let msg = self.websocket_receive.next().await;
+ let msg;
+
+ tokio::select! {
+ Ok(_) = self.kill_receive.recv() => {
+ log::trace!("GW: Closing listener task");
+ break;
+ }
+ message = self.websocket_receive.next() => {
+ msg = message;
+ }
+ }
// PRETTYFYME: Remove inline conditional compiling
- // This if chain can be much better but if let is unstable on stable rust
#[cfg(not(target_arch = "wasm32"))]
if let Some(Ok(message)) = msg {
self.handle_message(message.into()).await;
@@ -335,10 +349,42 @@ impl Gateway {
.unwrap();
}
GATEWAY_RECONNECT => {
- todo!()
+ trace!("GW: Received Reconnect");
+
+ let reconnect = GatewayReconnect {};
+
+ self.events
+ .lock()
+ .await
+ .session
+ .reconnect
+ .notify(reconnect)
+ .await;
}
GATEWAY_INVALID_SESSION => {
- todo!()
+ trace!("GW: Received Invalid Session");
+
+ let mut resumable: bool = false;
+
+ if let Some(raw_value) = gateway_payload.event_data {
+ if let Ok(deserialized) = serde_json::from_str(raw_value.get()) {
+ resumable = deserialized;
+ } else {
+ warn!("Failed to parse part of INVALID_SESSION ('{}' as bool), assuming non-resumable", raw_value.get());
+ }
+ } else {
+ warn!("Failed to parse part of INVALID_SESSION ('d' missing), assuming non-resumable");
+ }
+
+ let invalid_session = GatewayInvalidSession { resumable };
+
+ self.events
+ .lock()
+ .await
+ .session
+ .invalid
+ .notify(invalid_session)
+ .await;
}
// Starts our heartbeat
// We should have already handled this in gateway init
@@ -394,165 +440,3 @@ impl Gateway {
}
}
}
-
-pub mod event {
- use super::*;
-
- #[derive(Default, Debug)]
- pub struct Events {
- pub application: Application,
- pub auto_moderation: AutoModeration,
- pub session: Session,
- pub message: Message,
- pub user: User,
- pub relationship: Relationship,
- pub channel: Channel,
- pub thread: Thread,
- pub guild: Guild,
- pub invite: Invite,
- pub integration: Integration,
- pub interaction: Interaction,
- pub stage_instance: StageInstance,
- pub call: Call,
- pub voice: Voice,
- pub webhooks: Webhooks,
- pub gateway_identify_payload: GatewayEvent,
- pub gateway_resume: GatewayEvent,
- pub error: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Application {
- pub command_permissions_update: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct AutoModeration {
- pub rule_create: GatewayEvent,
- pub rule_update: GatewayEvent,
- pub rule_delete: GatewayEvent,
- pub action_execution: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Session {
- pub ready: GatewayEvent,
- pub ready_supplemental: GatewayEvent,
- pub replace: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct StageInstance {
- pub create: GatewayEvent,
- pub update: GatewayEvent,
- pub delete: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Message {
- pub create: GatewayEvent,
- pub update: GatewayEvent,
- pub delete: GatewayEvent,
- pub delete_bulk: GatewayEvent,
- pub reaction_add: GatewayEvent,
- pub reaction_remove: GatewayEvent,
- pub reaction_remove_all: GatewayEvent,
- pub reaction_remove_emoji: GatewayEvent,
- pub ack: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct User {
- pub update: GatewayEvent,
- pub guild_settings_update: GatewayEvent,
- pub presence_update: GatewayEvent,
- pub typing_start: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Relationship {
- pub add: GatewayEvent,
- pub remove: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Channel {
- pub create: GatewayEvent,
- pub update: GatewayEvent,
- pub unread_update: GatewayEvent,
- pub delete: GatewayEvent,
- pub pins_update: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Thread {
- pub create: GatewayEvent,
- pub update: GatewayEvent,
- pub delete: GatewayEvent,
- pub list_sync: GatewayEvent,
- pub member_update: GatewayEvent,
- pub members_update: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Guild {
- pub create: GatewayEvent,
- pub update: GatewayEvent,
- pub delete: GatewayEvent,
- pub audit_log_entry_create: GatewayEvent,
- pub ban_add: GatewayEvent,
- pub ban_remove: GatewayEvent,
- pub emojis_update: GatewayEvent,
- pub stickers_update: GatewayEvent,
- pub integrations_update: GatewayEvent,
- pub member_add: GatewayEvent,
- pub member_remove: GatewayEvent,
- pub member_update: GatewayEvent,
- pub members_chunk: GatewayEvent,
- pub role_create: GatewayEvent,
- pub role_update: GatewayEvent,
- pub role_delete: GatewayEvent,
- pub role_scheduled_event_create: GatewayEvent,
- pub role_scheduled_event_update: GatewayEvent,
- pub role_scheduled_event_delete: GatewayEvent,
- pub role_scheduled_event_user_add: GatewayEvent,
- pub role_scheduled_event_user_remove: GatewayEvent,
- pub passive_update_v1: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Invite {
- pub create: GatewayEvent,
- pub delete: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Integration {
- pub create: GatewayEvent,
- pub update: GatewayEvent,
- pub delete: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Interaction {
- pub create: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Call {
- pub create: GatewayEvent,
- pub update: GatewayEvent,
- pub delete: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Voice {
- pub state_update: GatewayEvent,
- pub server_update: GatewayEvent,
- }
-
- #[derive(Default, Debug)]
- pub struct Webhooks {
- pub update: GatewayEvent,
- }
-}
diff --git a/src/gateway/handle.rs b/src/gateway/handle.rs
index 620faba8..6af5f0d2 100644
--- a/src/gateway/handle.rs
+++ b/src/gateway/handle.rs
@@ -1,9 +1,13 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use futures_util::SinkExt;
use log::*;
use std::fmt::Debug;
-use super::{event::Events, *};
+use super::{events::Events, *};
use crate::types::{self, Composite};
/// Represents a handle to a Gateway connection. A Gateway connection will create observable
@@ -40,10 +44,19 @@ impl GatewayHandle {
.unwrap();
}
+ /// Recursively observes a [`Shared`] object, by making sure all [`Composite `] fields within
+ /// that object and its children are being watched.
+ ///
+ /// Observing means, that if new information arrives about the observed object or its children,
+ /// the object automatically gets updated, without you needing to request new information about
+ /// the object in question from the API, which is expensive and can lead to rate limiting.
+ ///
+ /// The [`Shared`] object returned by this method points to a different object than the one
+ /// being supplied as a &self function argument.
pub async fn observe>(
&self,
- object: Arc>,
- ) -> Arc> {
+ object: Shared,
+ ) -> Shared {
let mut store = self.store.lock().await;
let id = object.read().unwrap().id();
if let Some(channel) = store.get(&id) {
@@ -84,7 +97,7 @@ impl GatewayHandle {
/// with all of its observable fields being observed.
pub async fn observe_and_into_inner>(
&self,
- object: Arc>,
+ object: Shared,
) -> T {
let channel = self.observe(object.clone()).await;
let object = channel.read().unwrap().clone();
@@ -160,7 +173,7 @@ impl GatewayHandle {
/// Closes the websocket connection and stops all gateway tasks;
///
- /// Esentially pulls the plug on the gateway, leaving it possible to resume;
+ /// Essentially pulls the plug on the gateway, leaving it possible to resume;
pub async fn close(&self) {
self.kill_send.send(()).unwrap();
self.websocket_send.lock().await.close().await.unwrap();
diff --git a/src/gateway/heartbeat.rs b/src/gateway/heartbeat.rs
index 8e376976..5dcc98db 100644
--- a/src/gateway/heartbeat.rs
+++ b/src/gateway/heartbeat.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use futures_util::SinkExt;
use log::*;
@@ -22,7 +26,7 @@ use super::*;
use crate::types;
/// The amount of time we wait for a heartbeat ack before resending our heartbeat in ms
-const HEARTBEAT_ACK_TIMEOUT: u64 = 2000;
+pub const HEARTBEAT_ACK_TIMEOUT: u64 = 2000;
/// Handles sending heartbeats to the gateway in another thread
#[allow(dead_code)] // FIXME: Remove this, once HeartbeatHandler is used
@@ -71,13 +75,8 @@ impl HeartbeatHandler {
let mut last_heartbeat_timestamp: Instant = Instant::now();
let mut last_heartbeat_acknowledged = true;
let mut last_seq_number: Option = None;
-
- loop {
- if kill_receive.try_recv().is_ok() {
- trace!("GW: Closing heartbeat task");
- break;
- }
+ loop {
let timeout = if last_heartbeat_acknowledged {
heartbeat_interval
} else {
@@ -111,6 +110,10 @@ impl HeartbeatHandler {
}
}
}
+ Ok(_) = kill_receive.recv() => {
+ log::trace!("GW: Closing heartbeat task");
+ break;
+ }
}
if should_send {
@@ -128,7 +131,7 @@ impl HeartbeatHandler {
let send_result = websocket_tx.lock().await.send(msg.into()).await;
if send_result.is_err() {
// We couldn't send, the websocket is broken
- warn!("GW: Couldnt send heartbeat, websocket seems broken");
+ warn!("GW: Couldn't send heartbeat, websocket seems broken");
break;
}
diff --git a/src/gateway/message.rs b/src/gateway/message.rs
index 2c12e486..44d912ef 100644
--- a/src/gateway/message.rs
+++ b/src/gateway/message.rs
@@ -1,15 +1,19 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use crate::types;
use super::*;
-/// Represents a messsage received from the gateway. This will be either a [types::GatewayReceivePayload], containing events, or a [GatewayError].
+/// Represents a message received from the gateway. This will be either a [types::GatewayReceivePayload], containing events, or a [GatewayError].
/// This struct is used internally when handling messages.
#[derive(Clone, Debug)]
pub struct GatewayMessage(pub String);
impl GatewayMessage {
/// Parses the message as an error;
- /// Returns the error if succesfully parsed, None if the message isn't an error
+ /// Returns the error if successfully parsed, None if the message isn't an error
pub fn error(&self) -> Option {
// Some error strings have dots on the end, which we don't care about
let processed_content = self.0.to_lowercase().replace('.', "");
diff --git a/src/gateway/mod.rs b/src/gateway/mod.rs
index 076ed541..5a5881ad 100644
--- a/src/gateway/mod.rs
+++ b/src/gateway/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use async_trait::async_trait;
pub mod backends;
@@ -94,6 +98,12 @@ pub struct GatewayEvent {
}
impl GatewayEvent {
+ pub fn new() -> Self {
+ Self {
+ observers: Vec::new(),
+ }
+ }
+
/// Returns true if the GatewayEvent is observed by at least one Observer.
pub fn is_observed(&self) -> bool {
!self.observers.is_empty()
@@ -116,9 +126,17 @@ impl GatewayEvent {
}
/// Notifies the observers of the GatewayEvent.
- async fn notify(&self, new_event_data: T) {
+ pub(crate) async fn notify(&self, new_event_data: T) {
for observer in &self.observers {
observer.update(&new_event_data).await;
}
}
}
+
+/// A type alias for [`Arc>`], used to make the public facing API concerned with
+/// Composite structs more ergonomic.
+/// ## Note
+///
+/// While `T` does not have to implement `Composite` to be used with `Shared`,
+/// the primary use of `Shared` is with types that implement `Composite`.
+pub type Shared = Arc>;
diff --git a/src/instance.rs b/src/instance.rs
index a4967e88..16610429 100644
--- a/src/instance.rs
+++ b/src/instance.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
//! Instance and ChorusUser objects.
use std::collections::HashMap;
@@ -9,7 +13,7 @@ use reqwest::Client;
use serde::{Deserialize, Serialize};
use crate::errors::ChorusResult;
-use crate::gateway::{Gateway, GatewayHandle};
+use crate::gateway::{Gateway, GatewayHandle, Shared};
use crate::ratelimiter::ChorusRequest;
use crate::types::types::subconfigs::limits::rates::RateLimits;
use crate::types::{
@@ -19,6 +23,7 @@ use crate::UrlBundle;
#[derive(Debug, Clone, Default, Serialize, Deserialize)]
/// The [`Instance`]; what you will be using to perform all sorts of actions on the Spacebar server.
+///
/// If `limits_information` is `None`, then the instance will not be rate limited.
pub struct Instance {
pub urls: UrlBundle,
@@ -36,8 +41,6 @@ impl PartialEq for Instance {
}
}
-impl Eq for Instance {}
-
impl std::hash::Hash for Instance {
fn hash(&self, state: &mut H) {
self.urls.hash(state);
@@ -72,8 +75,17 @@ impl PartialEq for LimitsInformation {
}
impl Instance {
- /// Creates a new [`Instance`] from the [relevant instance urls](UrlBundle). To create an Instance from one singular url, use [`Instance::from_root_url()`].
- async fn from_url_bundle(urls: UrlBundle) -> ChorusResult {
+ pub(crate) fn clone_limits_if_some(&self) -> Option> {
+ if self.limits_information.is_some() {
+ return Some(self.limits_information.as_ref().unwrap().ratelimits.clone());
+ }
+ None
+ }
+
+ /// Creates a new [`Instance`] from the [relevant instance urls](UrlBundle).
+ ///
+ /// To create an Instance from one singular url, use [`Instance::new()`].
+ pub async fn from_url_bundle(urls: UrlBundle) -> ChorusResult {
let is_limited: Option = Instance::is_limited(&urls.api).await?;
let limit_information;
@@ -103,17 +115,9 @@ impl Instance {
Ok(instance)
}
- pub(crate) fn clone_limits_if_some(&self) -> Option> {
- if self.limits_information.is_some() {
- return Some(self.limits_information.as_ref().unwrap().ratelimits.clone());
- }
- None
- }
-
/// Creates a new [`Instance`] by trying to get the [relevant instance urls](UrlBundle) from a root url.
- /// Shorthand for `Instance::new(UrlBundle::from_root_domain(root_domain).await?)`.
///
- /// If `limited` is `true`, then Chorus will track and enforce rate limits for this instance.
+ /// Shorthand for `Instance::from_url_bundle(UrlBundle::from_root_domain(root_domain).await?)`.
pub async fn new(root_url: &str) -> ChorusResult {
let urls = UrlBundle::from_root_url(root_url).await?;
Instance::from_url_bundle(urls).await
@@ -153,11 +157,11 @@ impl fmt::Display for Token {
/// It is used for most authenticated actions on a Spacebar server.
/// It also has its own [Gateway] connection.
pub struct ChorusUser {
- pub belongs_to: Arc>,
+ pub belongs_to: Shared,
pub token: String,
pub limits: Option>,
- pub settings: Arc>,
- pub object: Arc>,
+ pub settings: Shared,
+ pub object: Shared,
pub gateway: GatewayHandle,
}
@@ -169,8 +173,6 @@ impl PartialEq for ChorusUser {
}
}
-impl Eq for ChorusUser {}
-
impl ChorusUser {
pub fn token(&self) -> String {
self.token.clone()
@@ -183,14 +185,14 @@ impl ChorusUser {
/// Creates a new [ChorusUser] from existing data.
///
/// # Notes
- /// This isn't the prefered way to create a ChorusUser.
+ /// This isn't the preferred way to create a ChorusUser.
/// See [Instance::login_account] and [Instance::register_account] instead.
pub fn new(
- belongs_to: Arc>,
+ belongs_to: Shared,
token: String,
limits: Option>,
- settings: Arc>,
- object: Arc>,
+ settings: Shared,
+ object: Shared,
gateway: GatewayHandle,
) -> ChorusUser {
ChorusUser {
@@ -208,7 +210,7 @@ impl ChorusUser {
/// registering or logging in to the Instance, where you do not yet have a User object, but still
/// need to make a RateLimited request. To use the [`GatewayHandle`], you will have to identify
/// first.
- pub(crate) async fn shell(instance: Arc>, token: String) -> ChorusUser {
+ pub(crate) async fn shell(instance: Shared, token: String) -> ChorusUser {
let settings = Arc::new(RwLock::new(UserSettings::default()));
let object = Arc::new(RwLock::new(User::default()));
let wss_url = instance.read().unwrap().urls.wss.clone();
diff --git a/src/lib.rs b/src/lib.rs
index 1bbeeef6..6ffd8b84 100644
--- a/src/lib.rs
+++ b/src/lib.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
/*!
Chorus combines all the required functionalities of a user-centric Spacebar library into one package.
The library handles various aspects on your behalf, such as rate limiting, authentication and maintaining
@@ -44,7 +48,7 @@ let login_schema = LoginSchema {
password: "Correct-Horse-Battery-Staple".to_string(),
..Default::default()
};
-// Each user connects to the Gateway. The Gateway connection lives on a seperate thread. Depending on
+// Each user connects to the Gateway. The Gateway connection lives on a separate thread. Depending on
// the runtime feature you choose, this can potentially take advantage of all of your computers' threads.
let user = instance
.login_account(login_schema)
@@ -128,7 +132,10 @@ pub mod instance;
#[cfg(feature = "client")]
pub mod ratelimiter;
pub mod types;
-#[cfg(feature = "client")]
+#[cfg(all(
+ feature = "client",
+ any(feature = "voice_udp", feature = "voice_gateway")
+))]
pub mod voice;
#[derive(Clone, Default, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)]
diff --git a/src/ratelimiter.rs b/src/ratelimiter.rs
index f6a7c263..5e69d950 100644
--- a/src/ratelimiter.rs
+++ b/src/ratelimiter.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
//! Ratelimiter and request handling functionality.
use std::collections::HashMap;
diff --git a/src/types/config/mod.rs b/src/types/config/mod.rs
index 4a11c718..4419208f 100644
--- a/src/types/config/mod.rs
+++ b/src/types/config/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use serde_json::{Map, Value};
diff --git a/src/types/config/types/api_configuration.rs b/src/types/config/types/api_configuration.rs
index 2d617fe9..2543d9ce 100644
--- a/src/types/config/types/api_configuration.rs
+++ b/src/types/config/types/api_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Eq)]
diff --git a/src/types/config/types/cdn_configuration.rs b/src/types/config/types/cdn_configuration.rs
index 5c762739..c0045b26 100644
--- a/src/types/config/types/cdn_configuration.rs
+++ b/src/types/config/types/cdn_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/defaults_configuration.rs b/src/types/config/types/defaults_configuration.rs
index c2b67c22..e6e08671 100644
--- a/src/types/config/types/defaults_configuration.rs
+++ b/src/types/config/types/defaults_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::config::types::subconfigs::defaults::{guild::GuildDefaults, user::UserDefaults};
diff --git a/src/types/config/types/domains_configuration.rs b/src/types/config/types/domains_configuration.rs
index 297b8279..6e9218ef 100644
--- a/src/types/config/types/domains_configuration.rs
+++ b/src/types/config/types/domains_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Deserialize, Serialize, Eq, PartialEq, Hash, Clone, Debug)]
diff --git a/src/types/config/types/email_configuration.rs b/src/types/config/types/email_configuration.rs
index 954f4de4..ffeb7269 100644
--- a/src/types/config/types/email_configuration.rs
+++ b/src/types/config/types/email_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::config::types::subconfigs::email::{
diff --git a/src/types/config/types/endpoint_configuration.rs b/src/types/config/types/endpoint_configuration.rs
index b484791b..0119cb2e 100644
--- a/src/types/config/types/endpoint_configuration.rs
+++ b/src/types/config/types/endpoint_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/external_tokens_configuration.rs b/src/types/config/types/external_tokens_configuration.rs
index f417b2ff..44e13131 100644
--- a/src/types/config/types/external_tokens_configuration.rs
+++ b/src/types/config/types/external_tokens_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/general_configuration.rs b/src/types/config/types/general_configuration.rs
index 450fd526..d49a6f85 100644
--- a/src/types/config/types/general_configuration.rs
+++ b/src/types/config/types/general_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::utils::Snowflake;
diff --git a/src/types/config/types/gif_configuration.rs b/src/types/config/types/gif_configuration.rs
index 8644fb4b..f0e1358b 100644
--- a/src/types/config/types/gif_configuration.rs
+++ b/src/types/config/types/gif_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Copy, Default, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/guild_configuration.rs b/src/types/config/types/guild_configuration.rs
index 65897ea3..af40b30f 100644
--- a/src/types/config/types/guild_configuration.rs
+++ b/src/types/config/types/guild_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use std::fmt::{Display, Formatter};
#[cfg(feature = "sqlx")]
use std::io::Write;
diff --git a/src/types/config/types/kafka_configuration.rs b/src/types/config/types/kafka_configuration.rs
index 46d10b61..909dc032 100644
--- a/src/types/config/types/kafka_configuration.rs
+++ b/src/types/config/types/kafka_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::config::types::subconfigs::kafka::KafkaBroker;
diff --git a/src/types/config/types/limit_configuration.rs b/src/types/config/types/limit_configuration.rs
index 44f888a0..f322a30a 100644
--- a/src/types/config/types/limit_configuration.rs
+++ b/src/types/config/types/limit_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::config::types::subconfigs::limits::{
diff --git a/src/types/config/types/login_configuration.rs b/src/types/config/types/login_configuration.rs
index a2b1039b..83125e05 100644
--- a/src/types/config/types/login_configuration.rs
+++ b/src/types/config/types/login_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/metrics_configuration.rs b/src/types/config/types/metrics_configuration.rs
index 336ac842..98d35369 100644
--- a/src/types/config/types/metrics_configuration.rs
+++ b/src/types/config/types/metrics_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/mod.rs b/src/types/config/types/mod.rs
index 6ea2c033..a35b305f 100644
--- a/src/types/config/types/mod.rs
+++ b/src/types/config/types/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
pub mod api_configuration;
pub mod cdn_configuration;
pub mod defaults_configuration;
diff --git a/src/types/config/types/password_reset_configuration.rs b/src/types/config/types/password_reset_configuration.rs
index 4dddae96..d1c730e9 100644
--- a/src/types/config/types/password_reset_configuration.rs
+++ b/src/types/config/types/password_reset_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/rabbit_mq_configuration.rs b/src/types/config/types/rabbit_mq_configuration.rs
index 24370551..c073fa00 100644
--- a/src/types/config/types/rabbit_mq_configuration.rs
+++ b/src/types/config/types/rabbit_mq_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/region_configuration.rs b/src/types/config/types/region_configuration.rs
index 078fa033..9e933a2c 100644
--- a/src/types/config/types/region_configuration.rs
+++ b/src/types/config/types/region_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::config::types::subconfigs::region::Region;
diff --git a/src/types/config/types/register_configuration.rs b/src/types/config/types/register_configuration.rs
index 4a208240..a4573bfd 100644
--- a/src/types/config/types/register_configuration.rs
+++ b/src/types/config/types/register_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::config::types::subconfigs::register::{
diff --git a/src/types/config/types/security_configuration.rs b/src/types/config/types/security_configuration.rs
index caeb72ce..0bd190fd 100644
--- a/src/types/config/types/security_configuration.rs
+++ b/src/types/config/types/security_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use base64::Engine;
use rand::Fill;
use serde::{Deserialize, Serialize};
diff --git a/src/types/config/types/sentry_configuration.rs b/src/types/config/types/sentry_configuration.rs
index 99de4ffd..e6e15df3 100644
--- a/src/types/config/types/sentry_configuration.rs
+++ b/src/types/config/types/sentry_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use std::ffi::OsString;
use serde::{Deserialize, Serialize};
diff --git a/src/types/config/types/subconfigs/client/mod.rs b/src/types/config/types/subconfigs/client/mod.rs
index 5d3d304f..95aefdcb 100644
--- a/src/types/config/types/subconfigs/client/mod.rs
+++ b/src/types/config/types/subconfigs/client/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/defaults/guild.rs b/src/types/config/types/subconfigs/defaults/guild.rs
index a709f824..8509fe57 100644
--- a/src/types/config/types/subconfigs/defaults/guild.rs
+++ b/src/types/config/types/subconfigs/defaults/guild.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::{ExplicitContentFilterLevel, MessageNotificationLevel};
diff --git a/src/types/config/types/subconfigs/defaults/mod.rs b/src/types/config/types/subconfigs/defaults/mod.rs
index 56d877fa..250738ff 100644
--- a/src/types/config/types/subconfigs/defaults/mod.rs
+++ b/src/types/config/types/subconfigs/defaults/mod.rs
@@ -1,2 +1,6 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
pub mod guild;
pub mod user;
diff --git a/src/types/config/types/subconfigs/defaults/user.rs b/src/types/config/types/subconfigs/defaults/user.rs
index 635d6d45..d7dc7b3e 100644
--- a/src/types/config/types/subconfigs/defaults/user.rs
+++ b/src/types/config/types/subconfigs/defaults/user.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/email/mailgun.rs b/src/types/config/types/subconfigs/email/mailgun.rs
index 636e462e..9f803ad1 100644
--- a/src/types/config/types/subconfigs/email/mailgun.rs
+++ b/src/types/config/types/subconfigs/email/mailgun.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/email/mailjet.rs b/src/types/config/types/subconfigs/email/mailjet.rs
index 4e505c1e..b1c9b42b 100644
--- a/src/types/config/types/subconfigs/email/mailjet.rs
+++ b/src/types/config/types/subconfigs/email/mailjet.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/email/mod.rs b/src/types/config/types/subconfigs/email/mod.rs
index 21253fd4..f3cbbecc 100644
--- a/src/types/config/types/subconfigs/email/mod.rs
+++ b/src/types/config/types/subconfigs/email/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
pub mod mailgun;
pub mod mailjet;
pub mod sendgrid;
diff --git a/src/types/config/types/subconfigs/email/sendgrid.rs b/src/types/config/types/subconfigs/email/sendgrid.rs
index 879c7195..993a917d 100644
--- a/src/types/config/types/subconfigs/email/sendgrid.rs
+++ b/src/types/config/types/subconfigs/email/sendgrid.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/email/smtp.rs b/src/types/config/types/subconfigs/email/smtp.rs
index a02c66f3..38ade701 100644
--- a/src/types/config/types/subconfigs/email/smtp.rs
+++ b/src/types/config/types/subconfigs/email/smtp.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/guild/autojoin.rs b/src/types/config/types/subconfigs/guild/autojoin.rs
index fe72c6c9..2cb5cff6 100644
--- a/src/types/config/types/subconfigs/guild/autojoin.rs
+++ b/src/types/config/types/subconfigs/guild/autojoin.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::utils::Snowflake;
diff --git a/src/types/config/types/subconfigs/guild/discovery.rs b/src/types/config/types/subconfigs/guild/discovery.rs
index 1e283b0f..50738f13 100644
--- a/src/types/config/types/subconfigs/guild/discovery.rs
+++ b/src/types/config/types/subconfigs/guild/discovery.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/guild/mod.rs b/src/types/config/types/subconfigs/guild/mod.rs
index e4d7dcf1..f1803568 100644
--- a/src/types/config/types/subconfigs/guild/mod.rs
+++ b/src/types/config/types/subconfigs/guild/mod.rs
@@ -1,2 +1,6 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
pub mod autojoin;
pub mod discovery;
diff --git a/src/types/config/types/subconfigs/kafka/mod.rs b/src/types/config/types/subconfigs/kafka/mod.rs
index 1ee4015f..459cfd8b 100644
--- a/src/types/config/types/subconfigs/kafka/mod.rs
+++ b/src/types/config/types/subconfigs/kafka/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/limits/channel.rs b/src/types/config/types/subconfigs/limits/channel.rs
index 03e46e5f..24157264 100644
--- a/src/types/config/types/subconfigs/limits/channel.rs
+++ b/src/types/config/types/subconfigs/limits/channel.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/limits/global.rs b/src/types/config/types/subconfigs/limits/global.rs
index 87f9e1c0..0140447b 100644
--- a/src/types/config/types/subconfigs/limits/global.rs
+++ b/src/types/config/types/subconfigs/limits/global.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/limits/guild.rs b/src/types/config/types/subconfigs/limits/guild.rs
index 6def5a00..9ef8f906 100644
--- a/src/types/config/types/subconfigs/limits/guild.rs
+++ b/src/types/config/types/subconfigs/limits/guild.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/limits/message.rs b/src/types/config/types/subconfigs/limits/message.rs
index 9d368b9b..3beb76e2 100644
--- a/src/types/config/types/subconfigs/limits/message.rs
+++ b/src/types/config/types/subconfigs/limits/message.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/limits/mod.rs b/src/types/config/types/subconfigs/limits/mod.rs
index 4dbc2faf..549e9095 100644
--- a/src/types/config/types/subconfigs/limits/mod.rs
+++ b/src/types/config/types/subconfigs/limits/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
pub mod channel;
pub mod global;
pub mod guild;
diff --git a/src/types/config/types/subconfigs/limits/ratelimits/auth.rs b/src/types/config/types/subconfigs/limits/ratelimits/auth.rs
index 9815a5d2..78f1908a 100644
--- a/src/types/config/types/subconfigs/limits/ratelimits/auth.rs
+++ b/src/types/config/types/subconfigs/limits/ratelimits/auth.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::config::types::subconfigs::limits::ratelimits::RateLimitOptions;
diff --git a/src/types/config/types/subconfigs/limits/ratelimits/mod.rs b/src/types/config/types/subconfigs/limits/ratelimits/mod.rs
index 934b9228..501afe90 100644
--- a/src/types/config/types/subconfigs/limits/ratelimits/mod.rs
+++ b/src/types/config/types/subconfigs/limits/ratelimits/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
pub mod auth;
diff --git a/src/types/config/types/subconfigs/limits/ratelimits/route.rs b/src/types/config/types/subconfigs/limits/ratelimits/route.rs
index 1aa0be2a..7c70fdc1 100644
--- a/src/types/config/types/subconfigs/limits/ratelimits/route.rs
+++ b/src/types/config/types/subconfigs/limits/ratelimits/route.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::config::types::subconfigs::limits::ratelimits::{
diff --git a/src/types/config/types/subconfigs/limits/rates.rs b/src/types/config/types/subconfigs/limits/rates.rs
index 642dcc80..4c9b9a15 100644
--- a/src/types/config/types/subconfigs/limits/rates.rs
+++ b/src/types/config/types/subconfigs/limits/rates.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use std::collections::HashMap;
use serde::{Deserialize, Serialize};
diff --git a/src/types/config/types/subconfigs/limits/user.rs b/src/types/config/types/subconfigs/limits/user.rs
index e43b746c..473535ad 100644
--- a/src/types/config/types/subconfigs/limits/user.rs
+++ b/src/types/config/types/subconfigs/limits/user.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/mod.rs b/src/types/config/types/subconfigs/mod.rs
index 4c850968..4366dc4e 100644
--- a/src/types/config/types/subconfigs/mod.rs
+++ b/src/types/config/types/subconfigs/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
pub mod client;
pub mod defaults;
pub mod email;
diff --git a/src/types/config/types/subconfigs/region/mod.rs b/src/types/config/types/subconfigs/region/mod.rs
index 08c3b73d..1661c099 100644
--- a/src/types/config/types/subconfigs/region/mod.rs
+++ b/src/types/config/types/subconfigs/region/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/register/date_of_birth.rs b/src/types/config/types/subconfigs/register/date_of_birth.rs
index 9c1bec11..66892973 100644
--- a/src/types/config/types/subconfigs/register/date_of_birth.rs
+++ b/src/types/config/types/subconfigs/register/date_of_birth.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/register/email.rs b/src/types/config/types/subconfigs/register/email.rs
index 8688cd22..f2aea638 100644
--- a/src/types/config/types/subconfigs/register/email.rs
+++ b/src/types/config/types/subconfigs/register/email.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/register/mod.rs b/src/types/config/types/subconfigs/register/mod.rs
index a2d714e4..9a873ae1 100644
--- a/src/types/config/types/subconfigs/register/mod.rs
+++ b/src/types/config/types/subconfigs/register/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
pub use date_of_birth::DateOfBirthConfiguration;
pub use email::RegistrationEmailConfiguration;
pub use password::PasswordConfiguration;
diff --git a/src/types/config/types/subconfigs/register/password.rs b/src/types/config/types/subconfigs/register/password.rs
index 9247f7d0..1d380bab 100644
--- a/src/types/config/types/subconfigs/register/password.rs
+++ b/src/types/config/types/subconfigs/register/password.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/security/captcha.rs b/src/types/config/types/subconfigs/security/captcha.rs
index 82bb5171..c97b6e29 100644
--- a/src/types/config/types/subconfigs/security/captcha.rs
+++ b/src/types/config/types/subconfigs/security/captcha.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/subconfigs/security/mod.rs b/src/types/config/types/subconfigs/security/mod.rs
index 84550003..26146cdd 100644
--- a/src/types/config/types/subconfigs/security/mod.rs
+++ b/src/types/config/types/subconfigs/security/mod.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
pub use captcha::{CaptchaConfiguration, CaptchaService};
pub use twofactor::TwoFactorConfiguration;
diff --git a/src/types/config/types/subconfigs/security/twofactor.rs b/src/types/config/types/subconfigs/security/twofactor.rs
index 39a03730..1182a458 100644
--- a/src/types/config/types/subconfigs/security/twofactor.rs
+++ b/src/types/config/types/subconfigs/security/twofactor.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/config/types/template_configuration.rs b/src/types/config/types/template_configuration.rs
index 932670e5..9f370b47 100644
--- a/src/types/config/types/template_configuration.rs
+++ b/src/types/config/types/template_configuration.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
diff --git a/src/types/entities/application.rs b/src/types/entities/application.rs
index 0b55626b..4014e557 100644
--- a/src/types/entities/application.rs
+++ b/src/types/entities/application.rs
@@ -1,10 +1,13 @@
-use std::sync::{Arc, RwLock};
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
use bitflags::bitflags;
use serde::{Deserialize, Serialize};
use serde_json::Value;
use serde_repr::{Deserialize_repr, Serialize_repr};
+use crate::gateway::Shared;
use crate::types::utils::Snowflake;
use crate::types::{Team, User};
@@ -27,7 +30,7 @@ pub struct Application {
pub bot_require_code_grant: bool,
pub verify_key: String,
#[cfg_attr(feature = "sqlx", sqlx(skip))]
- pub owner: Arc>,
+ pub owner: Shared,
pub flags: u64,
#[cfg(feature = "sqlx")]
pub redirect_uris: Option>>,
@@ -49,7 +52,7 @@ pub struct Application {
#[cfg(feature = "sqlx")]
pub install_params: Option>,
#[cfg(not(feature = "sqlx"))]
- pub install_params: Option>>,
+ pub install_params: Option>,
pub terms_of_service_url: Option,
pub privacy_policy_url: Option,
#[cfg_attr(feature = "sqlx", sqlx(skip))]
@@ -142,7 +145,7 @@ pub struct ApplicationCommand {
pub application_id: Snowflake,
pub name: String,
pub description: String,
- pub options: Vec>>,
+ pub options: Vec>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
@@ -154,7 +157,7 @@ pub struct ApplicationCommandOption {
pub description: String,
pub required: bool,
pub choices: Vec,
- pub options: Arc>>,
+ pub options: Shared>,
}
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
@@ -190,14 +193,14 @@ pub enum ApplicationCommandOptionType {
pub struct ApplicationCommandInteractionData {
pub id: Snowflake,
pub name: String,
- pub options: Vec>>,
+ pub options: Vec>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct ApplicationCommandInteractionDataOption {
pub name: String,
pub value: Value,
- pub options: Vec>>,
+ pub options: Vec>,
}
#[derive(Debug, Default, Clone, Serialize, Deserialize)]
@@ -206,7 +209,7 @@ pub struct GuildApplicationCommandPermissions {
pub id: Snowflake,
pub application_id: Snowflake,
pub guild_id: Snowflake,
- pub permissions: Vec>>,
+ pub permissions: Vec>,
}
#[derive(Debug, Default, Clone, PartialEq, Serialize, Deserialize)]
diff --git a/src/types/entities/attachment.rs b/src/types/entities/attachment.rs
index ffbc5209..f2e221d1 100644
--- a/src/types/entities/attachment.rs
+++ b/src/types/entities/attachment.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use crate::types::utils::Snowflake;
@@ -55,73 +59,3 @@ pub struct PartialDiscordFileAttachment {
#[serde(skip_serializing)]
pub content: Vec,
}
-
-impl PartialDiscordFileAttachment {
- /// Moves `self.content` out of `self` and returns it.
- pub fn move_content(self) -> (Vec, PartialDiscordFileAttachment) {
- let content = self.content;
- let updated_struct = PartialDiscordFileAttachment {
- id: self.id,
- filename: self.filename,
- description: self.description,
- content_type: self.content_type,
- size: self.size,
- url: self.url,
- proxy_url: self.proxy_url,
- height: self.height,
- width: self.width,
- ephemeral: self.ephemeral,
- duration_secs: self.duration_secs,
- waveform: self.waveform,
- content: Vec::new(),
- };
- (content, updated_struct)
- }
-
- /// Moves `self.filename` out of `self` and returns it.
- pub fn move_filename(self) -> (String, PartialDiscordFileAttachment) {
- let filename = self.filename;
- let updated_struct = PartialDiscordFileAttachment {
- id: self.id,
- filename: String::new(),
- description: self.description,
- content_type: self.content_type,
- size: self.size,
- url: self.url,
- proxy_url: self.proxy_url,
- height: self.height,
- width: self.width,
-
- ephemeral: self.ephemeral,
- duration_secs: self.duration_secs,
- waveform: self.waveform,
- content: self.content,
- };
- (filename, updated_struct)
- }
-
- /// Moves `self.content_type` out of `self` and returns it.
- pub fn move_content_type(self) -> (Option, PartialDiscordFileAttachment) {
- let content_type = self.content_type;
- let updated_struct = PartialDiscordFileAttachment {
- id: self.id,
- filename: self.filename,
- description: self.description,
- content_type: None,
- size: self.size,
- url: self.url,
- proxy_url: self.proxy_url,
- height: self.height,
- width: self.width,
- ephemeral: self.ephemeral,
- duration_secs: self.duration_secs,
- waveform: self.waveform,
- content: self.content,
- };
- (content_type, updated_struct)
- }
-
- pub fn set_id(&mut self, id: i16) {
- self.id = Some(id);
- }
-}
diff --git a/src/types/entities/audit_log.rs b/src/types/entities/audit_log.rs
index be14f0f9..477fb20d 100644
--- a/src/types/entities/audit_log.rs
+++ b/src/types/entities/audit_log.rs
@@ -1,14 +1,17 @@
-use std::sync::{Arc, RwLock};
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
use serde::{Deserialize, Serialize};
+use crate::gateway::Shared;
use crate::types::utils::Snowflake;
#[derive(Serialize, Deserialize, Debug, Default, Clone)]
/// See
pub struct AuditLogEntry {
pub target_id: Option,
- pub changes: Option>>>,
+ pub changes: Option>>,
pub user_id: Option,
pub id: Snowflake,
// to:do implement an enum for these types
diff --git a/src/types/entities/auto_moderation.rs b/src/types/entities/auto_moderation.rs
index a8910b18..cd69bf26 100644
--- a/src/types/entities/auto_moderation.rs
+++ b/src/types/entities/auto_moderation.rs
@@ -1,5 +1,8 @@
-use std::sync::{Arc, RwLock};
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+use crate::gateway::Shared;
#[cfg(feature = "client")]
use crate::gateway::Updateable;
@@ -21,8 +24,8 @@ pub struct AutoModerationRule {
pub creator_id: Snowflake,
pub event_type: AutoModerationRuleEventType,
pub trigger_type: AutoModerationRuleTriggerType,
- pub trigger_metadata: Arc>,
- pub actions: Vec>>,
+ pub trigger_metadata: Shared,
+ pub actions: Vec>,
pub enabled: bool,
pub exempt_roles: Vec,
pub exempt_channels: Vec,
@@ -99,7 +102,7 @@ pub enum AutoModerationRuleKeywordPresetType {
pub struct AutoModerationAction {
#[serde(rename = "type")]
pub action_type: AutoModerationActionType,
- pub metadata: Option>>,
+ pub metadata: Option>,
}
#[derive(Serialize_repr, Deserialize_repr, Debug, Clone, Default)]
diff --git a/src/types/entities/channel.rs b/src/types/entities/channel.rs
index 5acf2ae5..7d000dcd 100644
--- a/src/types/entities/channel.rs
+++ b/src/types/entities/channel.rs
@@ -1,4 +1,6 @@
-use std::sync::{Arc, RwLock};
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
@@ -6,6 +8,7 @@ use serde_aux::prelude::deserialize_string_from_number;
use serde_repr::{Deserialize_repr, Serialize_repr};
use std::fmt::Debug;
+use crate::gateway::Shared;
use crate::types::{
entities::{GuildMember, User},
utils::Snowflake,
@@ -71,13 +74,13 @@ pub struct Channel {
pub permission_overwrites: Option>>,
#[cfg(not(feature = "sqlx"))]
#[cfg_attr(feature = "client", observe_option_vec)]
- pub permission_overwrites: Option>>>,
+ pub permission_overwrites: Option>>,
pub permissions: Option,
pub position: Option,
pub rate_limit_per_user: Option,
#[cfg_attr(feature = "sqlx", sqlx(skip))]
#[cfg_attr(feature = "client", observe_option_vec)]
- pub recipients: Option>>>,
+ pub recipients: Option>>,
pub rtc_region: Option,
#[cfg_attr(feature = "sqlx", sqlx(skip))]
pub thread_metadata: Option,
@@ -171,7 +174,7 @@ pub struct ThreadMember {
pub user_id: Option,
pub join_timestamp: Option,
pub flags: Option,
- pub member: Option>>,
+ pub member: Option>,
}
#[derive(Debug, Deserialize, Serialize, Clone, PartialEq, Eq)]
diff --git a/src/types/entities/config.rs b/src/types/entities/config.rs
index 25b1ef16..aca828a7 100644
--- a/src/types/entities/config.rs
+++ b/src/types/entities/config.rs
@@ -1,3 +1,7 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use serde::{Deserialize, Serialize};
use serde_json::Value;
#[cfg(feature = "sqlx")]
@@ -12,23 +16,20 @@ pub struct ConfigEntity {
impl ConfigEntity {
pub fn as_string(&self) -> Option {
- let Some(v) = self.value.as_ref() else {
- return None;
- };
- Some(v.as_str().expect("value is not a string").to_string())
+ let v = self.value.as_ref()?;
+ let v = v.as_str()?;
+ Some(v.to_string())
}
pub fn as_bool(&self) -> Option {
- let Some(v) = self.value.as_ref() else {
- return None;
- };
- Some(v.as_bool().expect("value is not a boolean"))
+ let v = self.value.as_ref()?;
+ let v = v.as_bool()?;
+ Some(v)
}
pub fn as_int(&self) -> Option {
- let Some(v) = self.value.as_ref() else {
- return None;
- };
- Some(v.as_i64().expect("value is not a number"))
+ let v = self.value.as_ref()?;
+ let v = v.as_i64()?;
+ Some(v)
}
}
diff --git a/src/types/entities/emoji.rs b/src/types/entities/emoji.rs
index b3916e20..e84b025c 100644
--- a/src/types/entities/emoji.rs
+++ b/src/types/entities/emoji.rs
@@ -1,8 +1,12 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use std::fmt::Debug;
-use std::sync::{Arc, RwLock};
use serde::{Deserialize, Serialize};
+use crate::gateway::Shared;
use crate::types::entities::User;
use crate::types::Snowflake;
@@ -31,7 +35,7 @@ pub struct Emoji {
#[cfg(not(feature = "sqlx"))]
pub roles: Option>,
#[cfg_attr(feature = "sqlx", sqlx(skip))]
- pub user: Option>>,
+ pub user: Option>,
pub require_colons: Option,
pub managed: Option,
pub animated: Option,
@@ -62,37 +66,3 @@ impl PartialEq for Emoji {
|| self.available != other.available)
}
}
-
-impl PartialOrd for Emoji {
- fn partial_cmp(&self, other: &Self) -> Option {
- match self.id.partial_cmp(&other.id) {
- Some(core::cmp::Ordering::Equal) => {}
- ord => return ord,
- }
- match self.name.partial_cmp(&other.name) {
- Some(core::cmp::Ordering::Equal) => {}
- ord => return ord,
- }
- match self.roles.partial_cmp(&other.roles) {
- Some(core::cmp::Ordering::Equal) => {}
- ord => return ord,
- }
- match self.roles.partial_cmp(&other.roles) {
- Some(core::cmp::Ordering::Equal) => {}
- ord => return ord,
- }
- match self.require_colons.partial_cmp(&other.require_colons) {
- Some(core::cmp::Ordering::Equal) => {}
- ord => return ord,
- }
- match self.managed.partial_cmp(&other.managed) {
- Some(core::cmp::Ordering::Equal) => {}
- ord => return ord,
- }
- match self.animated.partial_cmp(&other.animated) {
- Some(core::cmp::Ordering::Equal) => {}
- ord => return ord,
- }
- self.available.partial_cmp(&other.available)
- }
-}
diff --git a/src/types/entities/guild.rs b/src/types/entities/guild.rs
index 1fe235b2..52ec5e50 100644
--- a/src/types/entities/guild.rs
+++ b/src/types/entities/guild.rs
@@ -1,11 +1,15 @@
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
use std::fmt::Debug;
-use std::sync::{Arc, RwLock};
use bitflags::bitflags;
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use serde_repr::{Deserialize_repr, Serialize_repr};
+use crate::gateway::Shared;
use crate::types::types::guild_configuration::GuildFeaturesList;
use crate::types::{
entities::{Channel, Emoji, RoleObject, Sticker, User, VoiceState, Webhook},
@@ -45,14 +49,14 @@ pub struct Guild {
pub bans: Option>,
#[cfg_attr(feature = "sqlx", sqlx(skip))]
#[cfg_attr(feature = "client", observe_option_vec)]
- pub channels: Option>>>,
+ pub channels: Option>>,
pub default_message_notifications: Option,
pub description: Option,
pub discovery_splash: Option,
#[cfg_attr(feature = "sqlx", sqlx(skip))]
#[cfg_attr(feature = "client", observe_vec)]
#[serde(default)]
- pub emojis: Vec>>,
+ pub emojis: Vec>,
pub explicit_content_filter: Option,
//#[cfg_attr(feature = "sqlx", sqlx(try_from = "String"))]
pub features: Option,
@@ -88,7 +92,7 @@ pub struct Guild {
pub region: Option