From 96b16c62d95552884ac665b3cb5e79f0f21aa481 Mon Sep 17 00:00:00 2001 From: "D. Scott Boggs" Date: Sun, 27 Nov 2022 09:44:43 -0500 Subject: [PATCH] initial state; elefren 0.22.0 --- .cargo-ok | 1 + .cargo_vcs_info.json | 5 + .clog.toml | 3 + .env.sample | 5 + .gitignore | 5 + .travis.yml | 42 + .vscode/settings.json | 22 + CHANGELOG.md | 190 ++++ Cargo.toml | 99 ++ LICENCE-APACHE | 13 + LICENCE-MIT | 21 + Makefile | 8 + README.md | 100 ++ appveyor.yml | 29 + build.rs | 20 + docs/book.toml | 8 + docs/guide/.nojekyll | 1 + docs/guide/FontAwesome/css/font-awesome.css | 4 + docs/guide/FontAwesome/fonts/FontAwesome.ttf | Bin 0 -> 138204 bytes .../FontAwesome/fonts/fontawesome-webfont.eot | Bin 0 -> 68875 bytes .../FontAwesome/fonts/fontawesome-webfont.svg | 640 +++++++++++++ .../FontAwesome/fonts/fontawesome-webfont.ttf | Bin 0 -> 138204 bytes .../fonts/fontawesome-webfont.woff | Bin 0 -> 81284 bytes .../fonts/fontawesome-webfont.woff2 | Bin 0 -> 64464 bytes docs/guide/ayu-highlight.css | 71 ++ docs/guide/book.js | 600 ++++++++++++ docs/guide/chapter_1.html | 184 ++++ docs/guide/clipboard.min.js | 7 + docs/guide/css/chrome.css | 417 +++++++++ docs/guide/css/general.css | 144 +++ docs/guide/css/print.css | 54 ++ docs/guide/css/variables.css | 210 +++++ docs/guide/elasticlunr.min.js | 10 + docs/guide/favicon.png | Bin 0 -> 5679 bytes docs/guide/highlight.css | 69 ++ docs/guide/highlight.js | 2 + docs/guide/index.html | 184 ++++ docs/guide/mark.min.js | 7 + docs/guide/print.html | 192 ++++ docs/guide/searcher.js | 477 ++++++++++ docs/guide/searchindex.js | 1 + docs/guide/searchindex.json | 1 + docs/guide/tomorrow-night.css | 96 ++ docs/src/SUMMARY.md | 3 + docs/src/chapter_1.md | 1 + examples/follow_profile.rs | 27 + examples/follows_me.rs | 27 + examples/home_timeline.rs | 27 + examples/print_your_profile.rs | 27 + examples/register/mod.rs | 56 ++ examples/search.rs | 28 + examples/upload_photo.rs | 27 + rustfmt.toml | 16 + src/apps.rs | 233 +++++ src/data.rs | 17 + src/entities/account.rs | 144 +++ src/entities/attachment.rs | 63 ++ src/entities/card.rs | 28 + src/entities/context.rs | 13 + src/entities/event.rs | 14 + src/entities/filter.rs | 27 + src/entities/instance.rs | 45 + src/entities/itemsiter.rs | 95 ++ src/entities/list.rs | 6 + src/entities/mention.rs | 12 + src/entities/mod.rs | 59 ++ src/entities/notification.rs | 34 + src/entities/push.rs | 67 ++ src/entities/relationship.rs | 31 + src/entities/report.rs | 10 + src/entities/search_result.rs | 29 + src/entities/status.rs | 107 +++ src/errors.rs | 245 +++++ src/helpers/cli.rs | 27 + src/helpers/env.rs | 81 ++ src/helpers/json.rs | 231 +++++ src/helpers/mod.rs | 38 + src/helpers/toml.rs | 229 +++++ src/http_send.rs | 26 + src/lib.rs | 858 ++++++++++++++++++ src/macros.rs | 364 ++++++++ src/mastodon_client.rs | 388 ++++++++ src/page.rs | 226 +++++ src/registration.rs | 430 +++++++++ src/requests/filter.rs | 154 ++++ src/requests/mod.rs | 13 + src/requests/push.rs | 633 +++++++++++++ src/requests/statuses.rs | 614 +++++++++++++ src/requests/update_credentials.rs | 327 +++++++ src/scopes.rs | 818 +++++++++++++++++ src/status_builder.rs | 366 ++++++++ tests/skeptic.rs | 1 + 92 files changed, 11284 insertions(+) create mode 100644 .cargo-ok create mode 100644 .cargo_vcs_info.json create mode 100644 .clog.toml create mode 100644 .env.sample create mode 100644 .gitignore create mode 100644 .travis.yml create mode 100644 .vscode/settings.json create mode 100644 CHANGELOG.md create mode 100644 Cargo.toml create mode 100644 LICENCE-APACHE create mode 100644 LICENCE-MIT create mode 100644 Makefile create mode 100644 README.md create mode 100644 appveyor.yml create mode 100644 build.rs create mode 100644 docs/book.toml create mode 100644 docs/guide/.nojekyll create mode 100644 docs/guide/FontAwesome/css/font-awesome.css create mode 100644 docs/guide/FontAwesome/fonts/FontAwesome.ttf create mode 100644 docs/guide/FontAwesome/fonts/fontawesome-webfont.eot create mode 100644 docs/guide/FontAwesome/fonts/fontawesome-webfont.svg create mode 100644 docs/guide/FontAwesome/fonts/fontawesome-webfont.ttf create mode 100644 docs/guide/FontAwesome/fonts/fontawesome-webfont.woff create mode 100644 docs/guide/FontAwesome/fonts/fontawesome-webfont.woff2 create mode 100644 docs/guide/ayu-highlight.css create mode 100644 docs/guide/book.js create mode 100644 docs/guide/chapter_1.html create mode 100644 docs/guide/clipboard.min.js create mode 100644 docs/guide/css/chrome.css create mode 100644 docs/guide/css/general.css create mode 100644 docs/guide/css/print.css create mode 100644 docs/guide/css/variables.css create mode 100644 docs/guide/elasticlunr.min.js create mode 100644 docs/guide/favicon.png create mode 100644 docs/guide/highlight.css create mode 100644 docs/guide/highlight.js create mode 100644 docs/guide/index.html create mode 100644 docs/guide/mark.min.js create mode 100644 docs/guide/print.html create mode 100644 docs/guide/searcher.js create mode 100644 docs/guide/searchindex.js create mode 100644 docs/guide/searchindex.json create mode 100644 docs/guide/tomorrow-night.css create mode 100644 docs/src/SUMMARY.md create mode 100644 docs/src/chapter_1.md create mode 100644 examples/follow_profile.rs create mode 100644 examples/follows_me.rs create mode 100644 examples/home_timeline.rs create mode 100644 examples/print_your_profile.rs create mode 100644 examples/register/mod.rs create mode 100644 examples/search.rs create mode 100644 examples/upload_photo.rs create mode 100644 rustfmt.toml create mode 100644 src/apps.rs create mode 100644 src/data.rs create mode 100644 src/entities/account.rs create mode 100644 src/entities/attachment.rs create mode 100644 src/entities/card.rs create mode 100644 src/entities/context.rs create mode 100644 src/entities/event.rs create mode 100644 src/entities/filter.rs create mode 100644 src/entities/instance.rs create mode 100644 src/entities/itemsiter.rs create mode 100644 src/entities/list.rs create mode 100644 src/entities/mention.rs create mode 100644 src/entities/mod.rs create mode 100644 src/entities/notification.rs create mode 100644 src/entities/push.rs create mode 100644 src/entities/relationship.rs create mode 100644 src/entities/report.rs create mode 100644 src/entities/search_result.rs create mode 100644 src/entities/status.rs create mode 100644 src/errors.rs create mode 100644 src/helpers/cli.rs create mode 100644 src/helpers/env.rs create mode 100644 src/helpers/json.rs create mode 100644 src/helpers/mod.rs create mode 100644 src/helpers/toml.rs create mode 100644 src/http_send.rs create mode 100644 src/lib.rs create mode 100644 src/macros.rs create mode 100644 src/mastodon_client.rs create mode 100644 src/page.rs create mode 100644 src/registration.rs create mode 100644 src/requests/filter.rs create mode 100644 src/requests/mod.rs create mode 100644 src/requests/push.rs create mode 100644 src/requests/statuses.rs create mode 100644 src/requests/update_credentials.rs create mode 100644 src/scopes.rs create mode 100644 src/status_builder.rs create mode 100644 tests/skeptic.rs diff --git a/.cargo-ok b/.cargo-ok new file mode 100644 index 0000000..b5754e2 --- /dev/null +++ b/.cargo-ok @@ -0,0 +1 @@ +ok \ No newline at end of file diff --git a/.cargo_vcs_info.json b/.cargo_vcs_info.json new file mode 100644 index 0000000..8dcfab9 --- /dev/null +++ b/.cargo_vcs_info.json @@ -0,0 +1,5 @@ +{ + "git": { + "sha1": "17c727f5c615eed921d0095d1b6f49e44024025d" + } +} diff --git a/.clog.toml b/.clog.toml new file mode 100644 index 0000000..a853fbd --- /dev/null +++ b/.clog.toml @@ -0,0 +1,3 @@ +[clog] +changelog = "CHANGELOG.md" + diff --git a/.env.sample b/.env.sample new file mode 100644 index 0000000..693aa30 --- /dev/null +++ b/.env.sample @@ -0,0 +1,5 @@ +export TOKEN='snakeoil' +export CLIENT_ID='' +export CLIENT_SECRET='' +export REDIRECT='' +export BASE='https://mastodon.social' diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..7111969 --- /dev/null +++ b/.gitignore @@ -0,0 +1,5 @@ +target +Cargo.lock +.env +mastodon-data.toml* +libtest.rmeta diff --git a/.travis.yml b/.travis.yml new file mode 100644 index 0000000..8d8f57e --- /dev/null +++ b/.travis.yml @@ -0,0 +1,42 @@ +sudo: required +language: rust +dist: trusty +addons: + apt: + packages: + - libssl-dev +cache: + directories: + - $HOME/.cargo +env: + - ELEFREN_FEATURES="--features toml" + - ELEFREN_FEATURES="--features json" + - ELEFREN_FEATURES="--features all" + - ELEFREN_FEATURES="" +os: + - linux + - osx +rust: + - stable + - beta + - nightly +script: + - cargo test $ELEFREN_FEATURES +matrix: + include: + - name: "Coverage" + rust: nightly + before_cache: RUSTFLAGS="--cfg procmacro2_semver_exempt" cargo install -f cargo-tarpaulin + script: cargo build --features all + after_success: cargo tarpaulin --ciserver travis-ci --coveralls $TRAVIS_JOB_ID --features all + + - name: "rustfmt" + rust: nightly + before_script: rustup component add rustfmt-preview + script: cargo fmt --all -- --check + + allow_failures: + - rust: nightly +notifications: + email: + on_success: never diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..2c859eb --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,22 @@ +{ + "cSpell.words": [ + "Creds", + "deserialise", + "Elefren", + "favourite", + "favourited", + "indoc", + "isolang", + "querystring", + "reblog", + "reqwest", + "tomlcrate", + "tungstenite", + "Unauth", + "unendorse", + "unfavourite", + "unfollow", + "unmute", + "unreblog" + ] +} \ No newline at end of file diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 0000000..9bfe7b4 --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,190 @@ + +## v0.18.0 (2019-01-05) + +#### Breaking Changes + +* Change `Registered::complete` to take the receiver by reference + +#### Features + +* Add `Registered::into_parts` and `Registered::from_parts` +* Add `FromStr` implementation for `Scopes` + + +## v0.17.0 (2018-12-12) + +#### Bug Fixes + +* Fixes `follow` to return a `Relationship` [6f63fef](6f63fef0b2414099c55e36107aab785f689d19b6) + +#### Features + +* Add `follows_me` and `followed_by_me` shortcut methods [4dd4042](4dd40422b3de11aaadcfc600317c0df3556b38f5) + +#### Breaking Changes + +* Change `u64` ids to `&str` [9a08058](9a080583f0577d3f77eaf7be55758835c56ea61e) +* Fixes `follow` to return a `Relationship` [6f63fef](6f63fef0b2414099c55e36107aab785f689d19b6) + + +## v0.16.0 (2018-10-03) + +This release upgrades reqwest, our http client dependency, from 0.8.8 -> +0.9.0 + + +## v0.15.2 + +This is a bugfix release + + +## v0.15.1 + +This is a bugfix release + + +## v0.15.0 (2018-09-15) + +This release adds support for all new API endpoints introduced in Mastodon 2.4.* and 2.5.0 + +#### Features + +* **scopes:** Implement granular OAuth scopes ([e284894d](e284894d), closes [#44](44)) +* **helpers:** cli::authenticate ([034bd4e6](034bd4e6)) +* **client:** Implement client.update\_credentials ([a57c7e2f](a57c7e2f)) +* **status:** Add language code to status builer ([989d9a59](989d9a59), closes [#55](55)) +* **client:** Implement profile metadata update ([0ad1e374](0ad1e374), closes [#54](54)) +* **search:** Implement `GET /api/v2/search` ([28192e11](28192e11)) +* **client:** Implement push notifciations endpoints ([690b029d](690b029d), closes [#53](53)) +* **client:** Add `replies_count` property to `Status` entities ([7d752a9f](7d752a9f), closes [#73](73)) +* **client:** Implement Keyword filtering API ([7d164cb8](7d164cb8), closes [#71](71)) +* **client:** Implement the Follow Suggestions API ([7de1bdc0](7de1bdc0), closes [#72](72)) +* **client:** Implement the Endorsements API ([1f0ba184](1f0ba184), closes [#74](74)) + +#### Breaking Changes + +* **scopes:** Implement granular OAuth scopes ([e284894d](e284894d)) (closes [#44](44)) + +#### Bug Fixes + +* **client:** change return value of client.unfollow to `Relationship` ([20a9c69a](20a9c69a)) +* **client:** change `POST /search` to `GET /search` ([34e2c008](34e2c008)) + + + +## v0.14.0 (2018-08-29) + +#### Features + +* **helpers:** add json helper module ([46871da4](46871da4)) + +#### Breaking Changes + +* **client:** fix broken paged\_routes\_with\_id methods ([c66c305d](c66c305d), closes [#42](42)) + +#### Bug Fixes + +* **client:** fix broken paged\_routes\_with\_id methods ([c66c305d](c66c305d), closes [#42](42)) + + +# 0.13 (2018/08/27) + +## Features + +- `Registration` now duplicates the `AppBuilder` API, so you can + replace: + + ``` + let app = App::builder(); + app.client_name("test-client"); + + let registration = Registration::new("http://example.com") + .register(app)?; + ``` + + with this: + + ``` + let registration = Registration::new("http://example.com") + .client_name("test-client") + .build()?; + ``` + + You can still call use the `Registration` & `AppBuilder` APIs like + before, but any App passed to `.register` will supercede anything app + config set on the `Registration` object itself. + + In future releases, this will become a hard error. + +- `elefren::status_builder::StatusBuilder::new()` now takes anything + that implements `Display` instead of specifically an owned `String` + +## Breaking Changes + +- The `elefren::data::toml` module has been moved to + `elefren::helpers::toml` +- Because of the changes to `Registration`, the `elefren::apps::prelude` + module has been removed. The types that are still necessary from that + prelude have been moved to `elefren::prelude`, but + `elefren::apps::App` will have to be imported separately +- `elefren::entities::account::CredientialsBuilder` has been moved to + `elefren::entities::account::CredentialsBuilder` (note the spelling + difference) +- `Registered::complete` now takes a `&str` instead of a `String` + +## Compatibility + +- `elefren::entities::instance::Instance` now has the `max_toot_chars` + property, for use with the Pleroma and Glitch-soc APIs + +# 0.12 (2018/08/23) + +## Features + +- `Page::items_iter` added, abstracts over "next_page" logic +- `elefen::prelude` and `elefen::apps::prelude` modules added to more + easily import all necessary types +- Helpers added to allow serialization & deseriasization of `Data` types + to `toml` + +## Breaking Changes + +- Combined all parameters to `mastodon.statuses` (except `id`) to their + own type, `StatusesRequest` +- All API methods on `Mastodon` type moved into a trait, + `MastodonClient`, to enable better mocking during tests +- `Mastodon::from_data(Data)` was changed to `Mastodon::from(Data)` +- `AppBuilder` was changed, use `App::builder()` instead +- `Registration` was broken up to enable better registration flow + +## Documentation + +- All API methods now have doc comments +- All docs were updated for the new breaking changes + +## Compatibility + +- Login to pleroma instances was fixed + +# 0.11 +- Added more examples to `examples` directory. +- Fixed `follow` and `unfollow` routes. +- Updated `moved` field to be `Box`. + +# 0.10 + +- Added the ability to handle paged entities like favourites and such.(Only favourites in prerelease) +- Added optional `source` and `moved` fields to `Account`. +- Added `Source` struct to match with the `Account.source` field. +- Added `CredientialsBuilder` struct for updating profile using + `verify_credientials`. +- Attachment now handles being sent an empty object, which is converted + to `None`. +- Added ombed data fields to `Card`. +- Added `version` and `urls` fields to `Instance`. +- Added `id`, `muting_notifications`, and `domain_blocking` to `Relationship`. +- Added `emojis`, `language`, and `pinned` fields to `Status` +- Added `Emoji` struct. +- Added `List` and `Mention` structs(matching routes not added yet). +- Added example that prints your profile. +- Updated dependencies diff --git a/Cargo.toml b/Cargo.toml new file mode 100644 index 0000000..cb857fa --- /dev/null +++ b/Cargo.toml @@ -0,0 +1,99 @@ +# THIS FILE IS AUTOMATICALLY GENERATED BY CARGO +# +# When uploading crates to the registry Cargo will automatically +# "normalize" Cargo.toml files for maximal compatibility +# with all versions of Cargo and also rewrite `path` dependencies +# to registry (e.g., crates.io) dependencies +# +# If you believe there's an error in this file please file an +# issue against the rust-lang/cargo repository. If you're +# editing this file be aware that the upstream Cargo.toml +# will likely look very different (and much more reasonable) + +[package] +name = "elefren" +version = "0.22.0" +authors = ["Aaron Power ", "Paul Woolcock "] +description = "A wrapper around the Mastodon API." +readme = "README.md" +keywords = ["api", "web", "social", "mastodon", "wrapper"] +categories = ["web-programming", "web-programming::http-client", "api-bindings"] +license = "MIT/Apache-2.0" +repository = "https://github.com/pwoolcoc/elefren.git" +[package.metadata.docs.rs] +features = ["all"] +[dependencies.chrono] +version = "0.4" +features = ["serde"] + +[dependencies.doc-comment] +version = "0.3" + +[dependencies.envy] +version = "0.4.0" +optional = true + +[dependencies.hyper-old-types] +version = "0.11.0" + +[dependencies.isolang] +version = "1.0" +features = ["serde_serialize"] + +[dependencies.log] +version = "0.4" + +[dependencies.reqwest] +version = "0.9" +default-features = false + +[dependencies.serde] +version = "1" + +[dependencies.serde_derive] +version = "1" + +[dependencies.serde_json] +version = "1" + +[dependencies.serde_qs] +version = "0.4.5" + +[dependencies.serde_urlencoded] +version = "0.6.1" + +[dependencies.tap-reader] +version = "1" + +[dependencies.toml] +version = "0.5.0" +optional = true + +[dependencies.try_from] +version = "0.3.2" + +[dependencies.tungstenite] +version = "0.10.1" + +[dependencies.url] +version = "1" +[dev-dependencies.indoc] +version = "0.3.1" + +[dev-dependencies.pretty_env_logger] +version = "0.3.0" + +[dev-dependencies.skeptic] +version = "0.13.3" + +[dev-dependencies.tempfile] +version = "3.0.3" +[build-dependencies.skeptic] +version = "0.13.3" + +[features] +all = ["toml", "json", "env"] +default = ["reqwest/default-tls"] +env = ["envy"] +json = [] +rustls-tls = ["reqwest/rustls-tls"] diff --git a/LICENCE-APACHE b/LICENCE-APACHE new file mode 100644 index 0000000..97fdf99 --- /dev/null +++ b/LICENCE-APACHE @@ -0,0 +1,13 @@ +Copyright 2016 Aaron Power + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. \ No newline at end of file diff --git a/LICENCE-MIT b/LICENCE-MIT new file mode 100644 index 0000000..0e95451 --- /dev/null +++ b/LICENCE-MIT @@ -0,0 +1,21 @@ +MIT License (MIT) + +Copyright (c) 2016 Aaron Power + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. diff --git a/Makefile b/Makefile new file mode 100644 index 0000000..1c9c7fb --- /dev/null +++ b/Makefile @@ -0,0 +1,8 @@ +DOCS := $(PWD)/docs + +book: $(DOCS)/guide/index.html + +$(DOCS)/guide/index.html: $(DOCS)/src/*.md + cd docs && mdbook build + +.PHONY: book diff --git a/README.md b/README.md new file mode 100644 index 0000000..55bf18f --- /dev/null +++ b/README.md @@ -0,0 +1,100 @@ +# Elefren + +## A Wrapper for the Mastodon API. + +[![Build Status](https://travis-ci.org/pwoolcoc/elefren.svg?branch=master)](https://travis-ci.org/pwoolcoc/elefren) +[![Build Status](https://ci.appveyor.com/api/projects/status/qeigk3nmmps52wxv?svg=true)](https://ci.appveyor.com/project/pwoolcoc/elefren) +[![Coverage Status](https://coveralls.io/repos/github/pwoolcoc/elefren/badge.svg?branch=master&service=github)](https://coveralls.io/github/pwoolcoc/elefren?branch=master) +[![crates.io](https://img.shields.io/crates/v/elefren.svg)](https://crates.io/crates/elefren) +[![Docs](https://docs.rs/elefren/badge.svg)](https://docs.rs/elefren) +[![MIT/APACHE-2.0](https://img.shields.io/crates/l/elefren.svg)](https://crates.io/crates/elefren) + +[Documentation](https://docs.rs/elefren/) + +A wrapper around the [API](https://github.com/tootsuite/documentation/blob/master/docs/Using-the-API/API.md#tag) for [Mastodon](https://mastodon.social/) + +## Installation + +To add `elefren` to your project, add the following to the +`[dependencies]` section of your `Cargo.toml` + +```toml +elefren = "0.22" +``` + +## Example + +In your `Cargo.toml`, make sure you enable the `toml` feature: + +```toml +[dependencies] +elefren = { version = "0.22", features = ["toml"] } +``` + +```rust,no_run +// src/main.rs +extern crate elefren; + +use std::error::Error; + +use elefren::prelude::*; +use elefren::helpers::toml; // requires `features = ["toml"]` +use elefren::helpers::cli; + +fn main() -> Result<(), Box> { + let mastodon = if let Ok(data) = toml::from_file("mastodon-data.toml") { + Mastodon::from(data) + } else { + register()? + }; + + let you = mastodon.verify_credentials()?; + + println!("{:#?}", you); + + Ok(()) +} + +fn register() -> Result> { + let registration = Registration::new("https://mastodon.social") + .client_name("elefren-examples") + .build()?; + let mastodon = cli::authenticate(registration)?; + + // Save app data for using on the next run. + toml::to_file(&*mastodon, "mastodon-data.toml")?; + + Ok(mastodon) +} +``` + +It also supports the [Streaming API](https://docs.joinmastodon.org/api/streaming): + +```no_run +use elefren::prelude::*; +use elefren::entities::event::Event; + +use std::error::Error; + +fn main() -> Result<(), Box> { + let data = Data { + base: "".into(), + client_id: "".into(), + client_secret: "".into(), + redirect: "".into(), + token: "".into(), + }; + + let client = Mastodon::from(data); + + for event in client.streaming_user()? { + match event { + Event::Update(ref status) => { /* .. */ }, + Event::Notification(ref notification) => { /* .. */ }, + Event::Delete(ref id) => { /* .. */ }, + Event::FiltersChanged => { /* .. */ }, + } + } + Ok(()) +} +``` diff --git a/appveyor.yml b/appveyor.yml new file mode 100644 index 0000000..6087033 --- /dev/null +++ b/appveyor.yml @@ -0,0 +1,29 @@ +environment: + matrix: + - TARGET: x86_64-pc-windows-msvc + OTHER_TARGET: i686-pc-windows-msvc + MAKE_TARGETS: test-unit-x86_64-pc-windows-msvc + TOOLCHAIN: stable + - TARGET: x86_64-pc-windows-msvc + OTHER_TARGET: i686-pc-windows-msvc + MAKE_TARGETS: test-unit-x86_64-pc-windows-msvc + TOOLCHAIN: beta + - TARGET: x86_64-pc-windows-msvc + OTHER_TARGET: i686-pc-windows-msvc + MAKE_TARGETS: test-unit-x86_64-pc-windows-msvc + TOOLCHAIN: nightly + +install: + - appveyor-retry appveyor DownloadFile https://win.rustup.rs/ -FileName rustup-init.exe + - rustup-init.exe -y --default-host x86_64-pc-windows-msvc --default-toolchain %TOOLCHAIN% + - set PATH=%PATH%;C:\Users\appveyor\.cargo\bin + - rustup target add %OTHER_TARGET% + - rustc -V + - cargo -V + +clone_depth: 1 + +build: false + +test_script: + - cargo test --features all diff --git a/build.rs b/build.rs new file mode 100644 index 0000000..34ca6e2 --- /dev/null +++ b/build.rs @@ -0,0 +1,20 @@ +#[cfg(feature = "toml")] +extern crate skeptic; + +#[cfg(feature = "toml")] +fn main() { + skeptic::generate_doc_tests(&["README.md"]); +} + +#[cfg(not(feature = "toml"))] +fn main() { + // tests/skeptic.rs still expects a file to be at OUT_DIR/skeptic-tests.rs, so + // make a dummy one + use std::{env, fs::OpenOptions, path::Path}; + let out_dir = Path::new(&env::var("OUT_DIR").expect("no out_dir set")).join("skeptic-tests.rs"); + OpenOptions::new() + .create(true) + .write(true) + .open(out_dir) + .expect("couldn't write to file"); +} diff --git a/docs/book.toml b/docs/book.toml new file mode 100644 index 0000000..a711481 --- /dev/null +++ b/docs/book.toml @@ -0,0 +1,8 @@ +[book] +authors = ["Paul Woolcock"] +multilingual = false +src = "src" +title = "Elefren Guide" + +[build] +build-dir = "guide" diff --git a/docs/guide/.nojekyll b/docs/guide/.nojekyll new file mode 100644 index 0000000..8631215 --- /dev/null +++ b/docs/guide/.nojekyll @@ -0,0 +1 @@ +This file makes sure that Github Pages doesn't process mdBook's output. \ No newline at end of file diff --git a/docs/guide/FontAwesome/css/font-awesome.css b/docs/guide/FontAwesome/css/font-awesome.css new file mode 100644 index 0000000..ee4e978 --- /dev/null +++ b/docs/guide/FontAwesome/css/font-awesome.css @@ -0,0 +1,4 @@ +/*! + * Font Awesome 4.4.0 by @davegandy - http://fontawesome.io - @fontawesome + * License - http://fontawesome.io/license (Font: SIL OFL 1.1, CSS: MIT License) + */@font-face{font-family:'FontAwesome';src:url('../fonts/fontawesome-webfont.eot?v=4.4.0');src:url('../fonts/fontawesome-webfont.eot?#iefix&v=4.4.0') format('embedded-opentype'),url('../fonts/fontawesome-webfont.woff2?v=4.4.0') format('woff2'),url('../fonts/fontawesome-webfont.woff?v=4.4.0') format('woff'),url('../fonts/fontawesome-webfont.ttf?v=4.4.0') format('truetype'),url('../fonts/fontawesome-webfont.svg?v=4.4.0#fontawesomeregular') format('svg');font-weight:normal;font-style:normal}.fa{display:inline-block;font:normal normal normal 14px/1 FontAwesome;font-size:inherit;text-rendering:auto;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.fa-lg{font-size:1.33333333em;line-height:.75em;vertical-align:-15%}.fa-2x{font-size:2em}.fa-3x{font-size:3em}.fa-4x{font-size:4em}.fa-5x{font-size:5em}.fa-fw{width:1.28571429em;text-align:center}.fa-ul{padding-left:0;margin-left:2.14285714em;list-style-type:none}.fa-ul>li{position:relative}.fa-li{position:absolute;left:-2.14285714em;width:2.14285714em;top:.14285714em;text-align:center}.fa-li.fa-lg{left:-1.85714286em}.fa-border{padding:.2em .25em .15em;border:solid .08em #eee;border-radius:.1em}.fa-pull-left{float:left}.fa-pull-right{float:right}.fa.fa-pull-left{margin-right:.3em}.fa.fa-pull-right{margin-left:.3em}.pull-right{float:right}.pull-left{float:left}.fa.pull-left{margin-right:.3em}.fa.pull-right{margin-left:.3em}.fa-spin{-webkit-animation:fa-spin 2s infinite linear;animation:fa-spin 2s infinite linear}.fa-pulse{-webkit-animation:fa-spin 1s infinite steps(8);animation:fa-spin 1s infinite steps(8)}@-webkit-keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}@keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}100%{-webkit-transform:rotate(359deg);transform:rotate(359deg)}}.fa-rotate-90{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=1);-webkit-transform:rotate(90deg);-ms-transform:rotate(90deg);transform:rotate(90deg)}.fa-rotate-180{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=2);-webkit-transform:rotate(180deg);-ms-transform:rotate(180deg);transform:rotate(180deg)}.fa-rotate-270{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=3);-webkit-transform:rotate(270deg);-ms-transform:rotate(270deg);transform:rotate(270deg)}.fa-flip-horizontal{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=0, mirror=1);-webkit-transform:scale(-1, 1);-ms-transform:scale(-1, 1);transform:scale(-1, 1)}.fa-flip-vertical{filter:progid:DXImageTransform.Microsoft.BasicImage(rotation=2, mirror=1);-webkit-transform:scale(1, -1);-ms-transform:scale(1, -1);transform:scale(1, -1)}:root .fa-rotate-90,:root .fa-rotate-180,:root .fa-rotate-270,:root .fa-flip-horizontal,:root .fa-flip-vertical{filter:none}.fa-stack{position:relative;display:inline-block;width:2em;height:2em;line-height:2em;vertical-align:middle}.fa-stack-1x,.fa-stack-2x{position:absolute;left:0;width:100%;text-align:center}.fa-stack-1x{line-height:inherit}.fa-stack-2x{font-size:2em}.fa-inverse{color:#fff}.fa-glass:before{content:"\f000"}.fa-music:before{content:"\f001"}.fa-search:before{content:"\f002"}.fa-envelope-o:before{content:"\f003"}.fa-heart:before{content:"\f004"}.fa-star:before{content:"\f005"}.fa-star-o:before{content:"\f006"}.fa-user:before{content:"\f007"}.fa-film:before{content:"\f008"}.fa-th-large:before{content:"\f009"}.fa-th:before{content:"\f00a"}.fa-th-list:before{content:"\f00b"}.fa-check:before{content:"\f00c"}.fa-remove:before,.fa-close:before,.fa-times:before{content:"\f00d"}.fa-search-plus:before{content:"\f00e"}.fa-search-minus:before{content:"\f010"}.fa-power-off:before{content:"\f011"}.fa-signal:before{content:"\f012"}.fa-gear:before,.fa-cog:before{content:"\f013"}.fa-trash-o:before{content:"\f014"}.fa-home:before{content:"\f015"}.fa-file-o:before{content:"\f016"}.fa-clock-o:before{content:"\f017"}.fa-road:before{content:"\f018"}.fa-download:before{content:"\f019"}.fa-arrow-circle-o-down:before{content:"\f01a"}.fa-arrow-circle-o-up:before{content:"\f01b"}.fa-inbox:before{content:"\f01c"}.fa-play-circle-o:before{content:"\f01d"}.fa-rotate-right:before,.fa-repeat:before{content:"\f01e"}.fa-refresh:before{content:"\f021"}.fa-list-alt:before{content:"\f022"}.fa-lock:before{content:"\f023"}.fa-flag:before{content:"\f024"}.fa-headphones:before{content:"\f025"}.fa-volume-off:before{content:"\f026"}.fa-volume-down:before{content:"\f027"}.fa-volume-up:before{content:"\f028"}.fa-qrcode:before{content:"\f029"}.fa-barcode:before{content:"\f02a"}.fa-tag:before{content:"\f02b"}.fa-tags:before{content:"\f02c"}.fa-book:before{content:"\f02d"}.fa-bookmark:before{content:"\f02e"}.fa-print:before{content:"\f02f"}.fa-camera:before{content:"\f030"}.fa-font:before{content:"\f031"}.fa-bold:before{content:"\f032"}.fa-italic:before{content:"\f033"}.fa-text-height:before{content:"\f034"}.fa-text-width:before{content:"\f035"}.fa-align-left:before{content:"\f036"}.fa-align-center:before{content:"\f037"}.fa-align-right:before{content:"\f038"}.fa-align-justify:before{content:"\f039"}.fa-list:before{content:"\f03a"}.fa-dedent:before,.fa-outdent:before{content:"\f03b"}.fa-indent:before{content:"\f03c"}.fa-video-camera:before{content:"\f03d"}.fa-photo:before,.fa-image:before,.fa-picture-o:before{content:"\f03e"}.fa-pencil:before{content:"\f040"}.fa-map-marker:before{content:"\f041"}.fa-adjust:before{content:"\f042"}.fa-tint:before{content:"\f043"}.fa-edit:before,.fa-pencil-square-o:before{content:"\f044"}.fa-share-square-o:before{content:"\f045"}.fa-check-square-o:before{content:"\f046"}.fa-arrows:before{content:"\f047"}.fa-step-backward:before{content:"\f048"}.fa-fast-backward:before{content:"\f049"}.fa-backward:before{content:"\f04a"}.fa-play:before{content:"\f04b"}.fa-pause:before{content:"\f04c"}.fa-stop:before{content:"\f04d"}.fa-forward:before{content:"\f04e"}.fa-fast-forward:before{content:"\f050"}.fa-step-forward:before{content:"\f051"}.fa-eject:before{content:"\f052"}.fa-chevron-left:before{content:"\f053"}.fa-chevron-right:before{content:"\f054"}.fa-plus-circle:before{content:"\f055"}.fa-minus-circle:before{content:"\f056"}.fa-times-circle:before{content:"\f057"}.fa-check-circle:before{content:"\f058"}.fa-question-circle:before{content:"\f059"}.fa-info-circle:before{content:"\f05a"}.fa-crosshairs:before{content:"\f05b"}.fa-times-circle-o:before{content:"\f05c"}.fa-check-circle-o:before{content:"\f05d"}.fa-ban:before{content:"\f05e"}.fa-arrow-left:before{content:"\f060"}.fa-arrow-right:before{content:"\f061"}.fa-arrow-up:before{content:"\f062"}.fa-arrow-down:before{content:"\f063"}.fa-mail-forward:before,.fa-share:before{content:"\f064"}.fa-expand:before{content:"\f065"}.fa-compress:before{content:"\f066"}.fa-plus:before{content:"\f067"}.fa-minus:before{content:"\f068"}.fa-asterisk:before{content:"\f069"}.fa-exclamation-circle:before{content:"\f06a"}.fa-gift:before{content:"\f06b"}.fa-leaf:before{content:"\f06c"}.fa-fire:before{content:"\f06d"}.fa-eye:before{content:"\f06e"}.fa-eye-slash:before{content:"\f070"}.fa-warning:before,.fa-exclamation-triangle:before{content:"\f071"}.fa-plane:before{content:"\f072"}.fa-calendar:before{content:"\f073"}.fa-random:before{content:"\f074"}.fa-comment:before{content:"\f075"}.fa-magnet:before{content:"\f076"}.fa-chevron-up:before{content:"\f077"}.fa-chevron-down:before{content:"\f078"}.fa-retweet:before{content:"\f079"}.fa-shopping-cart:before{content:"\f07a"}.fa-folder:before{content:"\f07b"}.fa-folder-open:before{content:"\f07c"}.fa-arrows-v:before{content:"\f07d"}.fa-arrows-h:before{content:"\f07e"}.fa-bar-chart-o:before,.fa-bar-chart:before{content:"\f080"}.fa-twitter-square:before{content:"\f081"}.fa-facebook-square:before{content:"\f082"}.fa-camera-retro:before{content:"\f083"}.fa-key:before{content:"\f084"}.fa-gears:before,.fa-cogs:before{content:"\f085"}.fa-comments:before{content:"\f086"}.fa-thumbs-o-up:before{content:"\f087"}.fa-thumbs-o-down:before{content:"\f088"}.fa-star-half:before{content:"\f089"}.fa-heart-o:before{content:"\f08a"}.fa-sign-out:before{content:"\f08b"}.fa-linkedin-square:before{content:"\f08c"}.fa-thumb-tack:before{content:"\f08d"}.fa-external-link:before{content:"\f08e"}.fa-sign-in:before{content:"\f090"}.fa-trophy:before{content:"\f091"}.fa-github-square:before{content:"\f092"}.fa-upload:before{content:"\f093"}.fa-lemon-o:before{content:"\f094"}.fa-phone:before{content:"\f095"}.fa-square-o:before{content:"\f096"}.fa-bookmark-o:before{content:"\f097"}.fa-phone-square:before{content:"\f098"}.fa-twitter:before{content:"\f099"}.fa-facebook-f:before,.fa-facebook:before{content:"\f09a"}.fa-github:before{content:"\f09b"}.fa-unlock:before{content:"\f09c"}.fa-credit-card:before{content:"\f09d"}.fa-feed:before,.fa-rss:before{content:"\f09e"}.fa-hdd-o:before{content:"\f0a0"}.fa-bullhorn:before{content:"\f0a1"}.fa-bell:before{content:"\f0f3"}.fa-certificate:before{content:"\f0a3"}.fa-hand-o-right:before{content:"\f0a4"}.fa-hand-o-left:before{content:"\f0a5"}.fa-hand-o-up:before{content:"\f0a6"}.fa-hand-o-down:before{content:"\f0a7"}.fa-arrow-circle-left:before{content:"\f0a8"}.fa-arrow-circle-right:before{content:"\f0a9"}.fa-arrow-circle-up:before{content:"\f0aa"}.fa-arrow-circle-down:before{content:"\f0ab"}.fa-globe:before{content:"\f0ac"}.fa-wrench:before{content:"\f0ad"}.fa-tasks:before{content:"\f0ae"}.fa-filter:before{content:"\f0b0"}.fa-briefcase:before{content:"\f0b1"}.fa-arrows-alt:before{content:"\f0b2"}.fa-group:before,.fa-users:before{content:"\f0c0"}.fa-chain:before,.fa-link:before{content:"\f0c1"}.fa-cloud:before{content:"\f0c2"}.fa-flask:before{content:"\f0c3"}.fa-cut:before,.fa-scissors:before{content:"\f0c4"}.fa-copy:before,.fa-files-o:before{content:"\f0c5"}.fa-paperclip:before{content:"\f0c6"}.fa-save:before,.fa-floppy-o:before{content:"\f0c7"}.fa-square:before{content:"\f0c8"}.fa-navicon:before,.fa-reorder:before,.fa-bars:before{content:"\f0c9"}.fa-list-ul:before{content:"\f0ca"}.fa-list-ol:before{content:"\f0cb"}.fa-strikethrough:before{content:"\f0cc"}.fa-underline:before{content:"\f0cd"}.fa-table:before{content:"\f0ce"}.fa-magic:before{content:"\f0d0"}.fa-truck:before{content:"\f0d1"}.fa-pinterest:before{content:"\f0d2"}.fa-pinterest-square:before{content:"\f0d3"}.fa-google-plus-square:before{content:"\f0d4"}.fa-google-plus:before{content:"\f0d5"}.fa-money:before{content:"\f0d6"}.fa-caret-down:before{content:"\f0d7"}.fa-caret-up:before{content:"\f0d8"}.fa-caret-left:before{content:"\f0d9"}.fa-caret-right:before{content:"\f0da"}.fa-columns:before{content:"\f0db"}.fa-unsorted:before,.fa-sort:before{content:"\f0dc"}.fa-sort-down:before,.fa-sort-desc:before{content:"\f0dd"}.fa-sort-up:before,.fa-sort-asc:before{content:"\f0de"}.fa-envelope:before{content:"\f0e0"}.fa-linkedin:before{content:"\f0e1"}.fa-rotate-left:before,.fa-undo:before{content:"\f0e2"}.fa-legal:before,.fa-gavel:before{content:"\f0e3"}.fa-dashboard:before,.fa-tachometer:before{content:"\f0e4"}.fa-comment-o:before{content:"\f0e5"}.fa-comments-o:before{content:"\f0e6"}.fa-flash:before,.fa-bolt:before{content:"\f0e7"}.fa-sitemap:before{content:"\f0e8"}.fa-umbrella:before{content:"\f0e9"}.fa-paste:before,.fa-clipboard:before{content:"\f0ea"}.fa-lightbulb-o:before{content:"\f0eb"}.fa-exchange:before{content:"\f0ec"}.fa-cloud-download:before{content:"\f0ed"}.fa-cloud-upload:before{content:"\f0ee"}.fa-user-md:before{content:"\f0f0"}.fa-stethoscope:before{content:"\f0f1"}.fa-suitcase:before{content:"\f0f2"}.fa-bell-o:before{content:"\f0a2"}.fa-coffee:before{content:"\f0f4"}.fa-cutlery:before{content:"\f0f5"}.fa-file-text-o:before{content:"\f0f6"}.fa-building-o:before{content:"\f0f7"}.fa-hospital-o:before{content:"\f0f8"}.fa-ambulance:before{content:"\f0f9"}.fa-medkit:before{content:"\f0fa"}.fa-fighter-jet:before{content:"\f0fb"}.fa-beer:before{content:"\f0fc"}.fa-h-square:before{content:"\f0fd"}.fa-plus-square:before{content:"\f0fe"}.fa-angle-double-left:before{content:"\f100"}.fa-angle-double-right:before{content:"\f101"}.fa-angle-double-up:before{content:"\f102"}.fa-angle-double-down:before{content:"\f103"}.fa-angle-left:before{content:"\f104"}.fa-angle-right:before{content:"\f105"}.fa-angle-up:before{content:"\f106"}.fa-angle-down:before{content:"\f107"}.fa-desktop:before{content:"\f108"}.fa-laptop:before{content:"\f109"}.fa-tablet:before{content:"\f10a"}.fa-mobile-phone:before,.fa-mobile:before{content:"\f10b"}.fa-circle-o:before{content:"\f10c"}.fa-quote-left:before{content:"\f10d"}.fa-quote-right:before{content:"\f10e"}.fa-spinner:before{content:"\f110"}.fa-circle:before{content:"\f111"}.fa-mail-reply:before,.fa-reply:before{content:"\f112"}.fa-github-alt:before{content:"\f113"}.fa-folder-o:before{content:"\f114"}.fa-folder-open-o:before{content:"\f115"}.fa-smile-o:before{content:"\f118"}.fa-frown-o:before{content:"\f119"}.fa-meh-o:before{content:"\f11a"}.fa-gamepad:before{content:"\f11b"}.fa-keyboard-o:before{content:"\f11c"}.fa-flag-o:before{content:"\f11d"}.fa-flag-checkered:before{content:"\f11e"}.fa-terminal:before{content:"\f120"}.fa-code:before{content:"\f121"}.fa-mail-reply-all:before,.fa-reply-all:before{content:"\f122"}.fa-star-half-empty:before,.fa-star-half-full:before,.fa-star-half-o:before{content:"\f123"}.fa-location-arrow:before{content:"\f124"}.fa-crop:before{content:"\f125"}.fa-code-fork:before{content:"\f126"}.fa-unlink:before,.fa-chain-broken:before{content:"\f127"}.fa-question:before{content:"\f128"}.fa-info:before{content:"\f129"}.fa-exclamation:before{content:"\f12a"}.fa-superscript:before{content:"\f12b"}.fa-subscript:before{content:"\f12c"}.fa-eraser:before{content:"\f12d"}.fa-puzzle-piece:before{content:"\f12e"}.fa-microphone:before{content:"\f130"}.fa-microphone-slash:before{content:"\f131"}.fa-shield:before{content:"\f132"}.fa-calendar-o:before{content:"\f133"}.fa-fire-extinguisher:before{content:"\f134"}.fa-rocket:before{content:"\f135"}.fa-maxcdn:before{content:"\f136"}.fa-chevron-circle-left:before{content:"\f137"}.fa-chevron-circle-right:before{content:"\f138"}.fa-chevron-circle-up:before{content:"\f139"}.fa-chevron-circle-down:before{content:"\f13a"}.fa-html5:before{content:"\f13b"}.fa-css3:before{content:"\f13c"}.fa-anchor:before{content:"\f13d"}.fa-unlock-alt:before{content:"\f13e"}.fa-bullseye:before{content:"\f140"}.fa-ellipsis-h:before{content:"\f141"}.fa-ellipsis-v:before{content:"\f142"}.fa-rss-square:before{content:"\f143"}.fa-play-circle:before{content:"\f144"}.fa-ticket:before{content:"\f145"}.fa-minus-square:before{content:"\f146"}.fa-minus-square-o:before{content:"\f147"}.fa-level-up:before{content:"\f148"}.fa-level-down:before{content:"\f149"}.fa-check-square:before{content:"\f14a"}.fa-pencil-square:before{content:"\f14b"}.fa-external-link-square:before{content:"\f14c"}.fa-share-square:before{content:"\f14d"}.fa-compass:before{content:"\f14e"}.fa-toggle-down:before,.fa-caret-square-o-down:before{content:"\f150"}.fa-toggle-up:before,.fa-caret-square-o-up:before{content:"\f151"}.fa-toggle-right:before,.fa-caret-square-o-right:before{content:"\f152"}.fa-euro:before,.fa-eur:before{content:"\f153"}.fa-gbp:before{content:"\f154"}.fa-dollar:before,.fa-usd:before{content:"\f155"}.fa-rupee:before,.fa-inr:before{content:"\f156"}.fa-cny:before,.fa-rmb:before,.fa-yen:before,.fa-jpy:before{content:"\f157"}.fa-ruble:before,.fa-rouble:before,.fa-rub:before{content:"\f158"}.fa-won:before,.fa-krw:before{content:"\f159"}.fa-bitcoin:before,.fa-btc:before{content:"\f15a"}.fa-file:before{content:"\f15b"}.fa-file-text:before{content:"\f15c"}.fa-sort-alpha-asc:before{content:"\f15d"}.fa-sort-alpha-desc:before{content:"\f15e"}.fa-sort-amount-asc:before{content:"\f160"}.fa-sort-amount-desc:before{content:"\f161"}.fa-sort-numeric-asc:before{content:"\f162"}.fa-sort-numeric-desc:before{content:"\f163"}.fa-thumbs-up:before{content:"\f164"}.fa-thumbs-down:before{content:"\f165"}.fa-youtube-square:before{content:"\f166"}.fa-youtube:before{content:"\f167"}.fa-xing:before{content:"\f168"}.fa-xing-square:before{content:"\f169"}.fa-youtube-play:before{content:"\f16a"}.fa-dropbox:before{content:"\f16b"}.fa-stack-overflow:before{content:"\f16c"}.fa-instagram:before{content:"\f16d"}.fa-flickr:before{content:"\f16e"}.fa-adn:before{content:"\f170"}.fa-bitbucket:before{content:"\f171"}.fa-bitbucket-square:before{content:"\f172"}.fa-tumblr:before{content:"\f173"}.fa-tumblr-square:before{content:"\f174"}.fa-long-arrow-down:before{content:"\f175"}.fa-long-arrow-up:before{content:"\f176"}.fa-long-arrow-left:before{content:"\f177"}.fa-long-arrow-right:before{content:"\f178"}.fa-apple:before{content:"\f179"}.fa-windows:before{content:"\f17a"}.fa-android:before{content:"\f17b"}.fa-linux:before{content:"\f17c"}.fa-dribbble:before{content:"\f17d"}.fa-skype:before{content:"\f17e"}.fa-foursquare:before{content:"\f180"}.fa-trello:before{content:"\f181"}.fa-female:before{content:"\f182"}.fa-male:before{content:"\f183"}.fa-gittip:before,.fa-gratipay:before{content:"\f184"}.fa-sun-o:before{content:"\f185"}.fa-moon-o:before{content:"\f186"}.fa-archive:before{content:"\f187"}.fa-bug:before{content:"\f188"}.fa-vk:before{content:"\f189"}.fa-weibo:before{content:"\f18a"}.fa-renren:before{content:"\f18b"}.fa-pagelines:before{content:"\f18c"}.fa-stack-exchange:before{content:"\f18d"}.fa-arrow-circle-o-right:before{content:"\f18e"}.fa-arrow-circle-o-left:before{content:"\f190"}.fa-toggle-left:before,.fa-caret-square-o-left:before{content:"\f191"}.fa-dot-circle-o:before{content:"\f192"}.fa-wheelchair:before{content:"\f193"}.fa-vimeo-square:before{content:"\f194"}.fa-turkish-lira:before,.fa-try:before{content:"\f195"}.fa-plus-square-o:before{content:"\f196"}.fa-space-shuttle:before{content:"\f197"}.fa-slack:before{content:"\f198"}.fa-envelope-square:before{content:"\f199"}.fa-wordpress:before{content:"\f19a"}.fa-openid:before{content:"\f19b"}.fa-institution:before,.fa-bank:before,.fa-university:before{content:"\f19c"}.fa-mortar-board:before,.fa-graduation-cap:before{content:"\f19d"}.fa-yahoo:before{content:"\f19e"}.fa-google:before{content:"\f1a0"}.fa-reddit:before{content:"\f1a1"}.fa-reddit-square:before{content:"\f1a2"}.fa-stumbleupon-circle:before{content:"\f1a3"}.fa-stumbleupon:before{content:"\f1a4"}.fa-delicious:before{content:"\f1a5"}.fa-digg:before{content:"\f1a6"}.fa-pied-piper:before{content:"\f1a7"}.fa-pied-piper-alt:before{content:"\f1a8"}.fa-drupal:before{content:"\f1a9"}.fa-joomla:before{content:"\f1aa"}.fa-language:before{content:"\f1ab"}.fa-fax:before{content:"\f1ac"}.fa-building:before{content:"\f1ad"}.fa-child:before{content:"\f1ae"}.fa-paw:before{content:"\f1b0"}.fa-spoon:before{content:"\f1b1"}.fa-cube:before{content:"\f1b2"}.fa-cubes:before{content:"\f1b3"}.fa-behance:before{content:"\f1b4"}.fa-behance-square:before{content:"\f1b5"}.fa-steam:before{content:"\f1b6"}.fa-steam-square:before{content:"\f1b7"}.fa-recycle:before{content:"\f1b8"}.fa-automobile:before,.fa-car:before{content:"\f1b9"}.fa-cab:before,.fa-taxi:before{content:"\f1ba"}.fa-tree:before{content:"\f1bb"}.fa-spotify:before{content:"\f1bc"}.fa-deviantart:before{content:"\f1bd"}.fa-soundcloud:before{content:"\f1be"}.fa-database:before{content:"\f1c0"}.fa-file-pdf-o:before{content:"\f1c1"}.fa-file-word-o:before{content:"\f1c2"}.fa-file-excel-o:before{content:"\f1c3"}.fa-file-powerpoint-o:before{content:"\f1c4"}.fa-file-photo-o:before,.fa-file-picture-o:before,.fa-file-image-o:before{content:"\f1c5"}.fa-file-zip-o:before,.fa-file-archive-o:before{content:"\f1c6"}.fa-file-sound-o:before,.fa-file-audio-o:before{content:"\f1c7"}.fa-file-movie-o:before,.fa-file-video-o:before{content:"\f1c8"}.fa-file-code-o:before{content:"\f1c9"}.fa-vine:before{content:"\f1ca"}.fa-codepen:before{content:"\f1cb"}.fa-jsfiddle:before{content:"\f1cc"}.fa-life-bouy:before,.fa-life-buoy:before,.fa-life-saver:before,.fa-support:before,.fa-life-ring:before{content:"\f1cd"}.fa-circle-o-notch:before{content:"\f1ce"}.fa-ra:before,.fa-rebel:before{content:"\f1d0"}.fa-ge:before,.fa-empire:before{content:"\f1d1"}.fa-git-square:before{content:"\f1d2"}.fa-git:before{content:"\f1d3"}.fa-y-combinator-square:before,.fa-yc-square:before,.fa-hacker-news:before{content:"\f1d4"}.fa-tencent-weibo:before{content:"\f1d5"}.fa-qq:before{content:"\f1d6"}.fa-wechat:before,.fa-weixin:before{content:"\f1d7"}.fa-send:before,.fa-paper-plane:before{content:"\f1d8"}.fa-send-o:before,.fa-paper-plane-o:before{content:"\f1d9"}.fa-history:before{content:"\f1da"}.fa-circle-thin:before{content:"\f1db"}.fa-header:before{content:"\f1dc"}.fa-paragraph:before{content:"\f1dd"}.fa-sliders:before{content:"\f1de"}.fa-share-alt:before{content:"\f1e0"}.fa-share-alt-square:before{content:"\f1e1"}.fa-bomb:before{content:"\f1e2"}.fa-soccer-ball-o:before,.fa-futbol-o:before{content:"\f1e3"}.fa-tty:before{content:"\f1e4"}.fa-binoculars:before{content:"\f1e5"}.fa-plug:before{content:"\f1e6"}.fa-slideshare:before{content:"\f1e7"}.fa-twitch:before{content:"\f1e8"}.fa-yelp:before{content:"\f1e9"}.fa-newspaper-o:before{content:"\f1ea"}.fa-wifi:before{content:"\f1eb"}.fa-calculator:before{content:"\f1ec"}.fa-paypal:before{content:"\f1ed"}.fa-google-wallet:before{content:"\f1ee"}.fa-cc-visa:before{content:"\f1f0"}.fa-cc-mastercard:before{content:"\f1f1"}.fa-cc-discover:before{content:"\f1f2"}.fa-cc-amex:before{content:"\f1f3"}.fa-cc-paypal:before{content:"\f1f4"}.fa-cc-stripe:before{content:"\f1f5"}.fa-bell-slash:before{content:"\f1f6"}.fa-bell-slash-o:before{content:"\f1f7"}.fa-trash:before{content:"\f1f8"}.fa-copyright:before{content:"\f1f9"}.fa-at:before{content:"\f1fa"}.fa-eyedropper:before{content:"\f1fb"}.fa-paint-brush:before{content:"\f1fc"}.fa-birthday-cake:before{content:"\f1fd"}.fa-area-chart:before{content:"\f1fe"}.fa-pie-chart:before{content:"\f200"}.fa-line-chart:before{content:"\f201"}.fa-lastfm:before{content:"\f202"}.fa-lastfm-square:before{content:"\f203"}.fa-toggle-off:before{content:"\f204"}.fa-toggle-on:before{content:"\f205"}.fa-bicycle:before{content:"\f206"}.fa-bus:before{content:"\f207"}.fa-ioxhost:before{content:"\f208"}.fa-angellist:before{content:"\f209"}.fa-cc:before{content:"\f20a"}.fa-shekel:before,.fa-sheqel:before,.fa-ils:before{content:"\f20b"}.fa-meanpath:before{content:"\f20c"}.fa-buysellads:before{content:"\f20d"}.fa-connectdevelop:before{content:"\f20e"}.fa-dashcube:before{content:"\f210"}.fa-forumbee:before{content:"\f211"}.fa-leanpub:before{content:"\f212"}.fa-sellsy:before{content:"\f213"}.fa-shirtsinbulk:before{content:"\f214"}.fa-simplybuilt:before{content:"\f215"}.fa-skyatlas:before{content:"\f216"}.fa-cart-plus:before{content:"\f217"}.fa-cart-arrow-down:before{content:"\f218"}.fa-diamond:before{content:"\f219"}.fa-ship:before{content:"\f21a"}.fa-user-secret:before{content:"\f21b"}.fa-motorcycle:before{content:"\f21c"}.fa-street-view:before{content:"\f21d"}.fa-heartbeat:before{content:"\f21e"}.fa-venus:before{content:"\f221"}.fa-mars:before{content:"\f222"}.fa-mercury:before{content:"\f223"}.fa-intersex:before,.fa-transgender:before{content:"\f224"}.fa-transgender-alt:before{content:"\f225"}.fa-venus-double:before{content:"\f226"}.fa-mars-double:before{content:"\f227"}.fa-venus-mars:before{content:"\f228"}.fa-mars-stroke:before{content:"\f229"}.fa-mars-stroke-v:before{content:"\f22a"}.fa-mars-stroke-h:before{content:"\f22b"}.fa-neuter:before{content:"\f22c"}.fa-genderless:before{content:"\f22d"}.fa-facebook-official:before{content:"\f230"}.fa-pinterest-p:before{content:"\f231"}.fa-whatsapp:before{content:"\f232"}.fa-server:before{content:"\f233"}.fa-user-plus:before{content:"\f234"}.fa-user-times:before{content:"\f235"}.fa-hotel:before,.fa-bed:before{content:"\f236"}.fa-viacoin:before{content:"\f237"}.fa-train:before{content:"\f238"}.fa-subway:before{content:"\f239"}.fa-medium:before{content:"\f23a"}.fa-yc:before,.fa-y-combinator:before{content:"\f23b"}.fa-optin-monster:before{content:"\f23c"}.fa-opencart:before{content:"\f23d"}.fa-expeditedssl:before{content:"\f23e"}.fa-battery-4:before,.fa-battery-full:before{content:"\f240"}.fa-battery-3:before,.fa-battery-three-quarters:before{content:"\f241"}.fa-battery-2:before,.fa-battery-half:before{content:"\f242"}.fa-battery-1:before,.fa-battery-quarter:before{content:"\f243"}.fa-battery-0:before,.fa-battery-empty:before{content:"\f244"}.fa-mouse-pointer:before{content:"\f245"}.fa-i-cursor:before{content:"\f246"}.fa-object-group:before{content:"\f247"}.fa-object-ungroup:before{content:"\f248"}.fa-sticky-note:before{content:"\f249"}.fa-sticky-note-o:before{content:"\f24a"}.fa-cc-jcb:before{content:"\f24b"}.fa-cc-diners-club:before{content:"\f24c"}.fa-clone:before{content:"\f24d"}.fa-balance-scale:before{content:"\f24e"}.fa-hourglass-o:before{content:"\f250"}.fa-hourglass-1:before,.fa-hourglass-start:before{content:"\f251"}.fa-hourglass-2:before,.fa-hourglass-half:before{content:"\f252"}.fa-hourglass-3:before,.fa-hourglass-end:before{content:"\f253"}.fa-hourglass:before{content:"\f254"}.fa-hand-grab-o:before,.fa-hand-rock-o:before{content:"\f255"}.fa-hand-stop-o:before,.fa-hand-paper-o:before{content:"\f256"}.fa-hand-scissors-o:before{content:"\f257"}.fa-hand-lizard-o:before{content:"\f258"}.fa-hand-spock-o:before{content:"\f259"}.fa-hand-pointer-o:before{content:"\f25a"}.fa-hand-peace-o:before{content:"\f25b"}.fa-trademark:before{content:"\f25c"}.fa-registered:before{content:"\f25d"}.fa-creative-commons:before{content:"\f25e"}.fa-gg:before{content:"\f260"}.fa-gg-circle:before{content:"\f261"}.fa-tripadvisor:before{content:"\f262"}.fa-odnoklassniki:before{content:"\f263"}.fa-odnoklassniki-square:before{content:"\f264"}.fa-get-pocket:before{content:"\f265"}.fa-wikipedia-w:before{content:"\f266"}.fa-safari:before{content:"\f267"}.fa-chrome:before{content:"\f268"}.fa-firefox:before{content:"\f269"}.fa-opera:before{content:"\f26a"}.fa-internet-explorer:before{content:"\f26b"}.fa-tv:before,.fa-television:before{content:"\f26c"}.fa-contao:before{content:"\f26d"}.fa-500px:before{content:"\f26e"}.fa-amazon:before{content:"\f270"}.fa-calendar-plus-o:before{content:"\f271"}.fa-calendar-minus-o:before{content:"\f272"}.fa-calendar-times-o:before{content:"\f273"}.fa-calendar-check-o:before{content:"\f274"}.fa-industry:before{content:"\f275"}.fa-map-pin:before{content:"\f276"}.fa-map-signs:before{content:"\f277"}.fa-map-o:before{content:"\f278"}.fa-map:before{content:"\f279"}.fa-commenting:before{content:"\f27a"}.fa-commenting-o:before{content:"\f27b"}.fa-houzz:before{content:"\f27c"}.fa-vimeo:before{content:"\f27d"}.fa-black-tie:before{content:"\f27e"}.fa-fonticons:before{content:"\f280"} diff --git a/docs/guide/FontAwesome/fonts/FontAwesome.ttf b/docs/guide/FontAwesome/fonts/FontAwesome.ttf new file mode 100644 index 0000000000000000000000000000000000000000..d7994e13086b1ac1a216bd754c93e1bccd65f237 GIT binary patch literal 138204 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

C{?=nA@$uA-~ywR8==j;^KG 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 qH)^kw9m{i|0XWbA9M~i{(3Tvie+TN{f%>0! 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 literal 0 HcmV?d00001 diff --git a/docs/guide/FontAwesome/fonts/fontawesome-webfont.svg b/docs/guide/FontAwesome/fonts/fontawesome-webfont.svg new file mode 100644 index 0000000..6fd19ab --- /dev/null +++ b/docs/guide/FontAwesome/fonts/fontawesome-webfont.svg @@ -0,0 +1,640 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/docs/guide/FontAwesome/fonts/fontawesome-webfont.ttf b/docs/guide/FontAwesome/fonts/fontawesome-webfont.ttf new file mode 100644 index 0000000000000000000000000000000000000000..d7994e13086b1ac1a216bd754c93e1bccd65f237 GIT binary patch literal 138204 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

C{?=nA@$uA-~ywR8==j;^KG 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 dZh*27rtA`)|4 setTimeout(() => reject(new Error('timeout')), timeout)) + ]); + } + + var playpens = Array.from(document.querySelectorAll(".playpen")); + if (playpens.length > 0) { + fetch_with_timeout("https://play.rust-lang.org/meta/crates", { + headers: { + 'Content-Type': "application/json", + }, + method: 'POST', + mode: 'cors', + }) + .then(response => response.json()) + .then(response => { + // get list of crates available in the rust playground + let playground_crates = response.crates.map(item => item["id"]); + playpens.forEach(block => handle_crate_list_update(block, playground_crates)); + }); + } + + function handle_crate_list_update(playpen_block, playground_crates) { + // update the play buttons after receiving the response + update_play_button(playpen_block, playground_crates); + + // and install on change listener to dynamically update ACE editors + if (window.ace) { + let code_block = playpen_block.querySelector("code"); + if (code_block.classList.contains("editable")) { + let editor = window.ace.edit(code_block); + editor.addEventListener("change", function (e) { + update_play_button(playpen_block, playground_crates); + }); + } + } + } + + // updates the visibility of play button based on `no_run` class and + // used crates vs ones available on http://play.rust-lang.org + function update_play_button(pre_block, playground_crates) { + var play_button = pre_block.querySelector(".play-button"); + + // skip if code is `no_run` + if (pre_block.querySelector('code').classList.contains("no_run")) { + play_button.classList.add("hidden"); + return; + } + + // get list of `extern crate`'s from snippet + var txt = playpen_text(pre_block); + var re = /extern\s+crate\s+([a-zA-Z_0-9]+)\s*;/g; + var snippet_crates = []; + var item; + while (item = re.exec(txt)) { + snippet_crates.push(item[1]); + } + + // check if all used crates are available on play.rust-lang.org + var all_available = snippet_crates.every(function (elem) { + return playground_crates.indexOf(elem) > -1; + }); + + if (all_available) { + play_button.classList.remove("hidden"); + } else { + play_button.classList.add("hidden"); + } + } + + function run_rust_code(code_block) { + var result_block = code_block.querySelector(".result"); + if (!result_block) { + result_block = document.createElement('code'); + result_block.className = 'result hljs language-bash'; + + code_block.append(result_block); + } + + let text = playpen_text(code_block); + + var params = { + version: "stable", + optimize: "0", + code: text + }; + + if (text.indexOf("#![feature") !== -1) { + params.version = "nightly"; + } + + result_block.innerText = "Running..."; + + fetch_with_timeout("https://play.rust-lang.org/evaluate.json", { + headers: { + 'Content-Type': "application/json", + }, + method: 'POST', + mode: 'cors', + body: JSON.stringify(params) + }) + .then(response => response.json()) + .then(response => result_block.innerText = response.result) + .catch(error => result_block.innerText = "Playground Communication: " + error.message); + } + + // Syntax highlighting Configuration + hljs.configure({ + tabReplace: ' ', // 4 spaces + languages: [], // Languages used for auto-detection + }); + + if (window.ace) { + // language-rust class needs to be removed for editable + // blocks or highlightjs will capture events + Array + .from(document.querySelectorAll('code.editable')) + .forEach(function (block) { block.classList.remove('language-rust'); }); + + Array + .from(document.querySelectorAll('code:not(.editable)')) + .forEach(function (block) { hljs.highlightBlock(block); }); + } else { + Array + .from(document.querySelectorAll('code')) + .forEach(function (block) { hljs.highlightBlock(block); }); + } + + // Adding the hljs class gives code blocks the color css + // even if highlighting doesn't apply + Array + .from(document.querySelectorAll('code')) + .forEach(function (block) { block.classList.add('hljs'); }); + + Array.from(document.querySelectorAll("code.language-rust")).forEach(function (block) { + + var code_block = block; + var pre_block = block.parentNode; + // hide lines + var lines = code_block.innerHTML.split("\n"); + var first_non_hidden_line = false; + var lines_hidden = false; + var trimmed_line = ""; + + for (var n = 0; n < lines.length; n++) { + trimmed_line = lines[n].trim(); + if (trimmed_line[0] == hiding_character && trimmed_line[1] != hiding_character) { + if (first_non_hidden_line) { + lines[n] = "" + "\n" + lines[n].replace(/(\s*)# ?/, "$1") + ""; + } + else { + lines[n] = "" + lines[n].replace(/(\s*)# ?/, "$1") + "\n" + ""; + } + lines_hidden = true; + } + else if (first_non_hidden_line) { + lines[n] = "\n" + lines[n]; + } + else { + first_non_hidden_line = true; + } + if (trimmed_line[0] == hiding_character && trimmed_line[1] == hiding_character) { + lines[n] = lines[n].replace("##", "#") + } + } + code_block.innerHTML = lines.join(""); + + // If no lines were hidden, return + if (!lines_hidden) { return; } + + var buttons = document.createElement('div'); + buttons.className = 'buttons'; + buttons.innerHTML = ""; + + // add expand button + pre_block.insertBefore(buttons, pre_block.firstChild); + + pre_block.querySelector('.buttons').addEventListener('click', function (e) { + if (e.target.classList.contains('fa-expand')) { + var lines = pre_block.querySelectorAll('span.hidden'); + + e.target.classList.remove('fa-expand'); + e.target.classList.add('fa-compress'); + e.target.title = 'Hide lines'; + e.target.setAttribute('aria-label', e.target.title); + + Array.from(lines).forEach(function (line) { + line.classList.remove('hidden'); + line.classList.add('unhidden'); + }); + } else if (e.target.classList.contains('fa-compress')) { + var lines = pre_block.querySelectorAll('span.unhidden'); + + e.target.classList.remove('fa-compress'); + e.target.classList.add('fa-expand'); + e.target.title = 'Show hidden lines'; + e.target.setAttribute('aria-label', e.target.title); + + Array.from(lines).forEach(function (line) { + line.classList.remove('unhidden'); + line.classList.add('hidden'); + }); + } + }); + }); + + Array.from(document.querySelectorAll('pre code')).forEach(function (block) { + var pre_block = block.parentNode; + if (!pre_block.classList.contains('playpen')) { + var buttons = pre_block.querySelector(".buttons"); + if (!buttons) { + buttons = document.createElement('div'); + buttons.className = 'buttons'; + pre_block.insertBefore(buttons, pre_block.firstChild); + } + + var clipButton = document.createElement('button'); + clipButton.className = 'fa fa-copy clip-button'; + clipButton.title = 'Copy to clipboard'; + clipButton.setAttribute('aria-label', clipButton.title); + clipButton.innerHTML = ''; + + buttons.insertBefore(clipButton, buttons.firstChild); + } + }); + + // Process playpen code blocks + Array.from(document.querySelectorAll(".playpen")).forEach(function (pre_block) { + // Add play button + var buttons = pre_block.querySelector(".buttons"); + if (!buttons) { + buttons = document.createElement('div'); + buttons.className = 'buttons'; + pre_block.insertBefore(buttons, pre_block.firstChild); + } + + var runCodeButton = document.createElement('button'); + runCodeButton.className = 'fa fa-play play-button'; + runCodeButton.hidden = true; + runCodeButton.title = 'Run this code'; + runCodeButton.setAttribute('aria-label', runCodeButton.title); + + var copyCodeClipboardButton = document.createElement('button'); + copyCodeClipboardButton.className = 'fa fa-copy clip-button'; + copyCodeClipboardButton.innerHTML = ''; + copyCodeClipboardButton.title = 'Copy to clipboard'; + copyCodeClipboardButton.setAttribute('aria-label', copyCodeClipboardButton.title); + + buttons.insertBefore(runCodeButton, buttons.firstChild); + buttons.insertBefore(copyCodeClipboardButton, buttons.firstChild); + + runCodeButton.addEventListener('click', function (e) { + run_rust_code(pre_block); + }); + + let code_block = pre_block.querySelector("code"); + if (window.ace && code_block.classList.contains("editable")) { + var undoChangesButton = document.createElement('button'); + undoChangesButton.className = 'fa fa-history reset-button'; + undoChangesButton.title = 'Undo changes'; + undoChangesButton.setAttribute('aria-label', undoChangesButton.title); + + buttons.insertBefore(undoChangesButton, buttons.firstChild); + + undoChangesButton.addEventListener('click', function () { + let editor = window.ace.edit(code_block); + editor.setValue(editor.originalCode); + editor.clearSelection(); + }); + } + }); +})(); + +(function themes() { + var html = document.querySelector('html'); + var themeToggleButton = document.getElementById('theme-toggle'); + var themePopup = document.getElementById('theme-list'); + var themeColorMetaTag = document.querySelector('meta[name="theme-color"]'); + var stylesheets = { + ayuHighlight: document.querySelector("[href$='ayu-highlight.css']"), + tomorrowNight: document.querySelector("[href$='tomorrow-night.css']"), + highlight: document.querySelector("[href$='highlight.css']"), + }; + + function showThemes() { + themePopup.style.display = 'block'; + themeToggleButton.setAttribute('aria-expanded', true); + themePopup.querySelector("button#" + document.body.className).focus(); + } + + function hideThemes() { + themePopup.style.display = 'none'; + themeToggleButton.setAttribute('aria-expanded', false); + themeToggleButton.focus(); + } + + function set_theme(theme) { + let ace_theme; + + if (theme == 'coal' || theme == 'navy') { + stylesheets.ayuHighlight.disabled = true; + stylesheets.tomorrowNight.disabled = false; + stylesheets.highlight.disabled = true; + + ace_theme = "ace/theme/tomorrow_night"; + } else if (theme == 'ayu') { + stylesheets.ayuHighlight.disabled = false; + stylesheets.tomorrowNight.disabled = true; + stylesheets.highlight.disabled = true; + + ace_theme = "ace/theme/tomorrow_night"; + } else { + stylesheets.ayuHighlight.disabled = true; + stylesheets.tomorrowNight.disabled = true; + stylesheets.highlight.disabled = false; + + ace_theme = "ace/theme/dawn"; + } + + setTimeout(function () { + themeColorMetaTag.content = getComputedStyle(document.body).backgroundColor; + }, 1); + + if (window.ace && window.editors) { + window.editors.forEach(function (editor) { + editor.setTheme(ace_theme); + }); + } + + var previousTheme; + try { previousTheme = localStorage.getItem('mdbook-theme'); } catch (e) { } + if (previousTheme === null || previousTheme === undefined) { previousTheme = 'light'; } + + try { localStorage.setItem('mdbook-theme', theme); } catch (e) { } + + document.body.className = theme; + html.classList.remove(previousTheme); + html.classList.add(theme); + } + + // Set theme + var theme; + try { theme = localStorage.getItem('mdbook-theme'); } catch(e) { } + if (theme === null || theme === undefined) { theme = 'light'; } + + set_theme(theme); + + themeToggleButton.addEventListener('click', function () { + if (themePopup.style.display === 'block') { + hideThemes(); + } else { + showThemes(); + } + }); + + themePopup.addEventListener('click', function (e) { + var theme = e.target.id || e.target.parentElement.id; + set_theme(theme); + }); + + themePopup.addEventListener('focusout', function(e) { + // e.relatedTarget is null in Safari and Firefox on macOS (see workaround below) + if (!!e.relatedTarget && !themeToggleButton.contains(e.relatedTarget) && !themePopup.contains(e.relatedTarget)) { + hideThemes(); + } + }); + + // Should not be needed, but it works around an issue on macOS & iOS: https://github.com/rust-lang-nursery/mdBook/issues/628 + document.addEventListener('click', function(e) { + if (themePopup.style.display === 'block' && !themeToggleButton.contains(e.target) && !themePopup.contains(e.target)) { + hideThemes(); + } + }); + + document.addEventListener('keydown', function (e) { + if (e.altKey || e.ctrlKey || e.metaKey || e.shiftKey) { return; } + if (!themePopup.contains(e.target)) { return; } + + switch (e.key) { + case 'Escape': + e.preventDefault(); + hideThemes(); + break; + case 'ArrowUp': + e.preventDefault(); + var li = document.activeElement.parentElement; + if (li && li.previousElementSibling) { + li.previousElementSibling.querySelector('button').focus(); + } + break; + case 'ArrowDown': + e.preventDefault(); + var li = document.activeElement.parentElement; + if (li && li.nextElementSibling) { + li.nextElementSibling.querySelector('button').focus(); + } + break; + case 'Home': + e.preventDefault(); + themePopup.querySelector('li:first-child button').focus(); + break; + case 'End': + e.preventDefault(); + themePopup.querySelector('li:last-child button').focus(); + break; + } + }); +})(); + +(function sidebar() { + var html = document.querySelector("html"); + var sidebar = document.getElementById("sidebar"); + var sidebarLinks = document.querySelectorAll('#sidebar a'); + var sidebarToggleButton = document.getElementById("sidebar-toggle"); + var firstContact = null; + + function showSidebar() { + html.classList.remove('sidebar-hidden') + html.classList.add('sidebar-visible'); + Array.from(sidebarLinks).forEach(function (link) { + link.setAttribute('tabIndex', 0); + }); + sidebarToggleButton.setAttribute('aria-expanded', true); + sidebar.setAttribute('aria-hidden', false); + try { localStorage.setItem('mdbook-sidebar', 'visible'); } catch (e) { } + } + + function hideSidebar() { + html.classList.remove('sidebar-visible') + html.classList.add('sidebar-hidden'); + Array.from(sidebarLinks).forEach(function (link) { + link.setAttribute('tabIndex', -1); + }); + sidebarToggleButton.setAttribute('aria-expanded', false); + sidebar.setAttribute('aria-hidden', true); + try { localStorage.setItem('mdbook-sidebar', 'hidden'); } catch (e) { } + } + + // Toggle sidebar + sidebarToggleButton.addEventListener('click', function sidebarToggle() { + if (html.classList.contains("sidebar-hidden")) { + showSidebar(); + } else if (html.classList.contains("sidebar-visible")) { + hideSidebar(); + } else { + if (getComputedStyle(sidebar)['transform'] === 'none') { + hideSidebar(); + } else { + showSidebar(); + } + } + }); + + document.addEventListener('touchstart', function (e) { + firstContact = { + x: e.touches[0].clientX, + time: Date.now() + }; + }, { passive: true }); + + document.addEventListener('touchmove', function (e) { + if (!firstContact) + return; + + var curX = e.touches[0].clientX; + var xDiff = curX - firstContact.x, + tDiff = Date.now() - firstContact.time; + + if (tDiff < 250 && Math.abs(xDiff) >= 150) { + if (xDiff >= 0 && firstContact.x < Math.min(document.body.clientWidth * 0.25, 300)) + showSidebar(); + else if (xDiff < 0 && curX < 300) + hideSidebar(); + + firstContact = null; + } + }, { passive: true }); + + // Scroll sidebar to current active section + var activeSection = sidebar.querySelector(".active"); + if (activeSection) { + sidebar.scrollTop = activeSection.offsetTop; + } +})(); + +(function chapterNavigation() { + document.addEventListener('keydown', function (e) { + if (e.altKey || e.ctrlKey || e.metaKey || e.shiftKey) { return; } + if (window.search && window.search.hasFocus()) { return; } + + switch (e.key) { + case 'ArrowRight': + e.preventDefault(); + var nextButton = document.querySelector('.nav-chapters.next'); + if (nextButton) { + window.location.href = nextButton.href; + } + break; + case 'ArrowLeft': + e.preventDefault(); + var previousButton = document.querySelector('.nav-chapters.previous'); + if (previousButton) { + window.location.href = previousButton.href; + } + break; + } + }); +})(); + +(function clipboard() { + var clipButtons = document.querySelectorAll('.clip-button'); + + function hideTooltip(elem) { + elem.firstChild.innerText = ""; + elem.className = 'fa fa-copy clip-button'; + } + + function showTooltip(elem, msg) { + elem.firstChild.innerText = msg; + elem.className = 'fa fa-copy tooltipped'; + } + + var clipboardSnippets = new Clipboard('.clip-button', { + text: function (trigger) { + hideTooltip(trigger); + let playpen = trigger.closest("pre"); + return playpen_text(playpen); + } + }); + + Array.from(clipButtons).forEach(function (clipButton) { + clipButton.addEventListener('mouseout', function (e) { + hideTooltip(e.currentTarget); + }); + }); + + clipboardSnippets.on('success', function (e) { + e.clearSelection(); + showTooltip(e.trigger, "Copied!"); + }); + + clipboardSnippets.on('error', function (e) { + showTooltip(e.trigger, "Clipboard error!"); + }); +})(); + +(function scrollToTop () { + var menuTitle = document.querySelector('.menu-title'); + + menuTitle.addEventListener('click', function () { + document.scrollingElement.scrollTo({ top: 0, behavior: 'smooth' }); + }); +})(); + +(function autoHideMenu() { + var menu = document.getElementById('menu-bar'); + + var previousScrollTop = document.scrollingElement.scrollTop; + + document.addEventListener('scroll', function () { + if (menu.classList.contains('folded') && document.scrollingElement.scrollTop < previousScrollTop) { + menu.classList.remove('folded'); + } else if (!menu.classList.contains('folded') && document.scrollingElement.scrollTop > previousScrollTop) { + menu.classList.add('folded'); + } + + if (!menu.classList.contains('bordered') && document.scrollingElement.scrollTop > 0) { + menu.classList.add('bordered'); + } + + if (menu.classList.contains('bordered') && document.scrollingElement.scrollTop === 0) { + menu.classList.remove('bordered'); + } + + previousScrollTop = document.scrollingElement.scrollTop; + }, { passive: true }); +})(); diff --git a/docs/guide/chapter_1.html b/docs/guide/chapter_1.html new file mode 100644 index 0000000..5ed75f3 --- /dev/null +++ b/docs/guide/chapter_1.html @@ -0,0 +1,184 @@ + + + + + + Chapter 1 - Elefren Guide + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +

+ +
+ +
+ + + + + + + + + + +
+
+

Chapter 1

+ +
+ + +
+
+ + + +
+ + + + + + + + + + + + + + + + + + + + + + + + diff --git a/docs/guide/clipboard.min.js b/docs/guide/clipboard.min.js new file mode 100644 index 0000000..1993676 --- /dev/null +++ b/docs/guide/clipboard.min.js @@ -0,0 +1,7 @@ +/*! + * clipboard.js v1.6.1 + * https://zenorocha.github.io/clipboard.js + * + * Licensed MIT © Zeno Rocha + */ +!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var t;t="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this,t.Clipboard=e()}}(function(){var e,t,n;return function e(t,n,o){function i(a,c){if(!n[a]){if(!t[a]){var l="function"==typeof require&&require;if(!c&&l)return l(a,!0);if(r)return r(a,!0);var u=new Error("Cannot find module '"+a+"'");throw u.code="MODULE_NOT_FOUND",u}var s=n[a]={exports:{}};t[a][0].call(s.exports,function(e){var n=t[a][1][e];return i(n?n:e)},s,s.exports,e,t,n,o)}return n[a].exports}for(var r="function"==typeof require&&require,a=0;a0&&void 0!==arguments[0]?arguments[0]:{};this.action=t.action,this.emitter=t.emitter,this.target=t.target,this.text=t.text,this.trigger=t.trigger,this.selectedText=""}},{key:"initSelection",value:function e(){this.text?this.selectFake():this.target&&this.selectTarget()}},{key:"selectFake",value:function e(){var t=this,n="rtl"==document.documentElement.getAttribute("dir");this.removeFake(),this.fakeHandlerCallback=function(){return t.removeFake()},this.fakeHandler=document.body.addEventListener("click",this.fakeHandlerCallback)||!0,this.fakeElem=document.createElement("textarea"),this.fakeElem.style.fontSize="12pt",this.fakeElem.style.border="0",this.fakeElem.style.padding="0",this.fakeElem.style.margin="0",this.fakeElem.style.position="absolute",this.fakeElem.style[n?"right":"left"]="-9999px";var o=window.pageYOffset||document.documentElement.scrollTop;this.fakeElem.style.top=o+"px",this.fakeElem.setAttribute("readonly",""),this.fakeElem.value=this.text,document.body.appendChild(this.fakeElem),this.selectedText=(0,i.default)(this.fakeElem),this.copyText()}},{key:"removeFake",value:function e(){this.fakeHandler&&(document.body.removeEventListener("click",this.fakeHandlerCallback),this.fakeHandler=null,this.fakeHandlerCallback=null),this.fakeElem&&(document.body.removeChild(this.fakeElem),this.fakeElem=null)}},{key:"selectTarget",value:function e(){this.selectedText=(0,i.default)(this.target),this.copyText()}},{key:"copyText",value:function e(){var t=void 0;try{t=document.execCommand(this.action)}catch(e){t=!1}this.handleResult(t)}},{key:"handleResult",value:function e(t){this.emitter.emit(t?"success":"error",{action:this.action,text:this.selectedText,trigger:this.trigger,clearSelection:this.clearSelection.bind(this)})}},{key:"clearSelection",value:function e(){this.target&&this.target.blur(),window.getSelection().removeAllRanges()}},{key:"destroy",value:function e(){this.removeFake()}},{key:"action",set:function e(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:"copy";if(this._action=t,"copy"!==this._action&&"cut"!==this._action)throw new Error('Invalid "action" value, use either "copy" or "cut"')},get:function e(){return this._action}},{key:"target",set:function e(t){if(void 0!==t){if(!t||"object"!==("undefined"==typeof t?"undefined":r(t))||1!==t.nodeType)throw new Error('Invalid "target" value, use a valid Element');if("copy"===this.action&&t.hasAttribute("disabled"))throw new Error('Invalid "target" attribute. Please use "readonly" instead of "disabled" attribute');if("cut"===this.action&&(t.hasAttribute("readonly")||t.hasAttribute("disabled")))throw new Error('Invalid "target" attribute. You can\'t cut text from elements with "readonly" or "disabled" attributes');this._target=t}},get:function e(){return this._target}}]),e}();e.exports=c})},{select:5}],8:[function(t,n,o){!function(i,r){if("function"==typeof e&&e.amd)e(["module","./clipboard-action","tiny-emitter","good-listener"],r);else if("undefined"!=typeof o)r(n,t("./clipboard-action"),t("tiny-emitter"),t("good-listener"));else{var a={exports:{}};r(a,i.clipboardAction,i.tinyEmitter,i.goodListener),i.clipboard=a.exports}}(this,function(e,t,n,o){"use strict";function i(e){return e&&e.__esModule?e:{default:e}}function r(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}function a(e,t){if(!e)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return!t||"object"!=typeof t&&"function"!=typeof t?e:t}function c(e,t){if("function"!=typeof t&&null!==t)throw new TypeError("Super expression must either be null or a function, not "+typeof t);e.prototype=Object.create(t&&t.prototype,{constructor:{value:e,enumerable:!1,writable:!0,configurable:!0}}),t&&(Object.setPrototypeOf?Object.setPrototypeOf(e,t):e.__proto__=t)}function l(e,t){var n="data-clipboard-"+e;if(t.hasAttribute(n))return t.getAttribute(n)}var u=i(t),s=i(n),f=i(o),d=function(){function e(e,t){for(var n=0;n0&&void 0!==arguments[0]?arguments[0]:{};this.action="function"==typeof t.action?t.action:this.defaultAction,this.target="function"==typeof t.target?t.target:this.defaultTarget,this.text="function"==typeof t.text?t.text:this.defaultText}},{key:"listenClick",value:function e(t){var n=this;this.listener=(0,f.default)(t,"click",function(e){return n.onClick(e)})}},{key:"onClick",value:function e(t){var n=t.delegateTarget||t.currentTarget;this.clipboardAction&&(this.clipboardAction=null),this.clipboardAction=new u.default({action:this.action(n),target:this.target(n),text:this.text(n),trigger:n,emitter:this})}},{key:"defaultAction",value:function e(t){return l("action",t)}},{key:"defaultTarget",value:function e(t){var n=l("target",t);if(n)return document.querySelector(n)}},{key:"defaultText",value:function e(t){return l("text",t)}},{key:"destroy",value:function e(){this.listener.destroy(),this.clipboardAction&&(this.clipboardAction.destroy(),this.clipboardAction=null)}}],[{key:"isSupported",value:function e(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:["copy","cut"],n="string"==typeof t?[t]:t,o=!!document.queryCommandSupported;return n.forEach(function(e){o=o&&!!document.queryCommandSupported(e)}),o}}]),t}(s.default);e.exports=h})},{"./clipboard-action":7,"good-listener":4,"tiny-emitter":6}]},{},[8])(8)}); \ No newline at end of file diff --git a/docs/guide/css/chrome.css b/docs/guide/css/chrome.css new file mode 100644 index 0000000..82883e6 --- /dev/null +++ b/docs/guide/css/chrome.css @@ -0,0 +1,417 @@ +/* CSS for UI elements (a.k.a. chrome) */ + +@import 'variables.css'; + +::-webkit-scrollbar { + background: var(--bg); +} +::-webkit-scrollbar-thumb { + background: var(--scrollbar); +} + +#searchresults a, +.content a:link, +a:visited, +a > .hljs { + color: var(--links); +} + +/* Menu Bar */ + +#menu-bar { + position: -webkit-sticky; + position: sticky; + top: 0; + z-index: 101; + margin: auto calc(0px - var(--page-padding)); +} +#menu-bar > #menu-bar-sticky-container { + display: flex; + flex-wrap: wrap; + background-color: var(--bg); + border-bottom-color: var(--bg); + border-bottom-width: 1px; + border-bottom-style: solid; +} +.js #menu-bar > #menu-bar-sticky-container { + transition: transform 0.3s; +} +#menu-bar.bordered > #menu-bar-sticky-container { + border-bottom-color: var(--table-border-color); +} +#menu-bar i, #menu-bar .icon-button { + position: relative; + padding: 0 8px; + z-index: 10; + line-height: 50px; + cursor: pointer; + transition: color 0.5s; +} +@media only screen and (max-width: 420px) { + #menu-bar i, #menu-bar .icon-button { + padding: 0 5px; + } +} + +.icon-button { + border: none; + background: none; + padding: 0; + color: inherit; +} +.icon-button i { + margin: 0; +} + +#print-button { + margin: 0 15px; +} + +html:not(.sidebar-visible) #menu-bar:not(:hover).folded > #menu-bar-sticky-container { + transform: translateY(-60px); +} + +.left-buttons { + display: flex; + margin: 0 5px; +} +.no-js .left-buttons { + display: none; +} + +.menu-title { + display: inline-block; + font-weight: 200; + font-size: 20px; + line-height: 50px; + text-align: center; + margin: 0; + flex: 1; + white-space: nowrap; + overflow: hidden; + text-overflow: ellipsis; +} +.js .menu-title { + cursor: pointer; +} + +.menu-bar, +.menu-bar:visited, +.nav-chapters, +.nav-chapters:visited, +.mobile-nav-chapters, +.mobile-nav-chapters:visited, +.menu-bar .icon-button, +.menu-bar a i { + color: var(--icons); +} + +.menu-bar i:hover, +.menu-bar .icon-button:hover, +.nav-chapters:hover, +.mobile-nav-chapters i:hover { + color: var(--icons-hover); +} + +/* Nav Icons */ + +.nav-chapters { + font-size: 2.5em; + text-align: center; + text-decoration: none; + + position: fixed; + top: 50px; /* Height of menu-bar */ + bottom: 0; + margin: 0; + max-width: 150px; + min-width: 90px; + + display: flex; + justify-content: center; + align-content: center; + flex-direction: column; + + transition: color 0.5s; +} + +.nav-chapters:hover { text-decoration: none; } + +.nav-wrapper { + margin-top: 50px; + display: none; +} + +.mobile-nav-chapters { + font-size: 2.5em; + text-align: center; + text-decoration: none; + width: 90px; + border-radius: 5px; + background-color: var(--sidebar-bg); +} + +.previous { + float: left; +} + +.next { + float: right; + right: var(--page-padding); +} + +@media only screen and (max-width: 1080px) { + .nav-wide-wrapper { display: none; } + .nav-wrapper { display: block; } +} + +@media only screen and (max-width: 1380px) { + .sidebar-visible .nav-wide-wrapper { display: none; } + .sidebar-visible .nav-wrapper { display: block; } +} + +/* Inline code */ + +:not(pre) > .hljs { + display: inline-block; + vertical-align: middle; + padding: 0.1em 0.3em; + border-radius: 3px; + color: var(--inline-code-color); +} + +a:hover > .hljs { + text-decoration: underline; +} + +pre { + position: relative; +} +pre > .buttons { + position: absolute; + z-index: 100; + right: 5px; + top: 5px; + + color: var(--sidebar-fg); + cursor: pointer; +} +pre > .buttons :hover { + color: var(--sidebar-active); +} +pre > .buttons i { + margin-left: 8px; +} +pre > .buttons button { + color: inherit; + background: transparent; + border: none; + cursor: inherit; +} +pre > .result { + margin-top: 10px; +} + +/* Search */ + +#searchresults a { + text-decoration: none; +} + +mark { + border-radius: 2px; + padding: 0 3px 1px 3px; + margin: 0 -3px -1px -3px; + background-color: var(--search-mark-bg); + transition: background-color 300ms linear; + cursor: pointer; +} + +mark.fade-out { + background-color: rgba(0,0,0,0) !important; + cursor: auto; +} + +.searchbar-outer { + margin-left: auto; + margin-right: auto; + max-width: var(--content-max-width); +} + +#searchbar { + width: 100%; + margin: 5px auto 0px auto; + padding: 10px 16px; + transition: box-shadow 300ms ease-in-out; + border: 1px solid var(--searchbar-border-color); + border-radius: 3px; + background-color: var(--searchbar-bg); + color: var(--searchbar-fg); +} +#searchbar:focus, +#searchbar.active { + box-shadow: 0 0 3px var(--searchbar-shadow-color); +} + +.searchresults-header { + font-weight: bold; + font-size: 1em; + padding: 18px 0 0 5px; + color: var(--searchresults-header-fg); +} + +.searchresults-outer { + margin-left: auto; + margin-right: auto; + max-width: var(--content-max-width); + border-bottom: 1px dashed var(--searchresults-border-color); +} + +ul#searchresults { + list-style: none; + padding-left: 20px; +} +ul#searchresults li { + margin: 10px 0px; + padding: 2px; + border-radius: 2px; +} +ul#searchresults li.focus { + background-color: var(--searchresults-li-bg); +} +ul#searchresults span.teaser { + display: block; + clear: both; + margin: 5px 0 0 20px; + font-size: 0.8em; +} +ul#searchresults span.teaser em { + font-weight: bold; + font-style: normal; +} + +/* Sidebar */ + +.sidebar { + position: fixed; + left: 0; + top: 0; + bottom: 0; + width: var(--sidebar-width); + overflow-y: auto; + padding: 10px 10px; + font-size: 0.875em; + box-sizing: border-box; + -webkit-overflow-scrolling: touch; + overscroll-behavior-y: contain; + background-color: var(--sidebar-bg); + color: var(--sidebar-fg); +} +.js .sidebar { + transition: transform 0.3s; /* Animation: slide away */ +} +.sidebar code { + line-height: 2em; +} +.sidebar-hidden .sidebar { + transform: translateX(calc(0px - var(--sidebar-width))); +} +.sidebar::-webkit-scrollbar { + background: var(--sidebar-bg); +} +.sidebar::-webkit-scrollbar-thumb { + background: var(--scrollbar); +} + +.sidebar-visible .page-wrapper { + transform: translateX(var(--sidebar-width)); +} +@media only screen and (min-width: 620px) { + .sidebar-visible .page-wrapper { + transform: none; + margin-left: var(--sidebar-width); + } +} + +.chapter { + list-style: none outside none; + padding-left: 0; + line-height: 2.2em; +} +.chapter li { + color: var(--sidebar-non-existant); +} +.chapter li a { + color: var(--sidebar-fg); + display: block; + padding: 0; + text-decoration: none; +} +.chapter li a:hover { text-decoration: none } +.chapter li .active, +a:hover { + /* Animate color change */ + color: var(--sidebar-active); +} + +.spacer { + width: 100%; + height: 3px; + margin: 5px 0px; +} +.chapter .spacer { + background-color: var(--sidebar-spacer); +} + +@media (-moz-touch-enabled: 1), (pointer: coarse) { + .chapter li a { padding: 5px 0; } + .spacer { margin: 10px 0; } +} + +.section { + list-style: none outside none; + padding-left: 20px; + line-height: 1.9em; +} + +/* Theme Menu Popup */ + +.theme-popup { + position: absolute; + left: 10px; + top: 50px; + z-index: 1000; + border-radius: 4px; + font-size: 0.7em; + color: var(--fg); + background: var(--theme-popup-bg); + border: 1px solid var(--theme-popup-border); + margin: 0; + padding: 0; + list-style: none; + display: none; +} +.theme-popup .default { + color: var(--icons); +} +.theme-popup .theme { + width: 100%; + border: 0; + margin: 0; + padding: 2px 10px; + line-height: 25px; + white-space: nowrap; + text-align: left; + cursor: pointer; + color: inherit; + background: inherit; + font-size: inherit; +} +.theme-popup .theme:hover { + background-color: var(--theme-hover); +} +.theme-popup .theme:hover:first-child, +.theme-popup .theme:hover:last-child { + border-top-left-radius: inherit; + border-top-right-radius: inherit; +} diff --git a/docs/guide/css/general.css b/docs/guide/css/general.css new file mode 100644 index 0000000..aedfb33 --- /dev/null +++ b/docs/guide/css/general.css @@ -0,0 +1,144 @@ +/* Base styles and content styles */ + +@import 'variables.css'; + +html { + font-family: "Open Sans", sans-serif; + color: var(--fg); + background-color: var(--bg); + text-size-adjust: none; +} + +body { + margin: 0; + font-size: 1rem; + overflow-x: hidden; +} + +code { + font-family: "Source Code Pro", Consolas, "Ubuntu Mono", Menlo, "DejaVu Sans Mono", monospace, monospace; + font-size: 0.875em; /* please adjust the ace font size accordingly in editor.js */ +} + +.left { float: left; } +.right { float: right; } +.hidden { display: none; } +.play-button.hidden { display: none; } + +h2, h3 { margin-top: 2.5em; } +h4, h5 { margin-top: 2em; } + +.header + .header h3, +.header + .header h4, +.header + .header h5 { + margin-top: 1em; +} + +a.header:target h1:before, +a.header:target h2:before, +a.header:target h3:before, +a.header:target h4:before { + display: inline-block; + content: "»"; + margin-left: -30px; + width: 30px; +} + +.page { + outline: 0; + padding: 0 var(--page-padding); +} +.page-wrapper { + box-sizing: border-box; +} +.js .page-wrapper { + transition: margin-left 0.3s ease, transform 0.3s ease; /* Animation: slide away */ +} + +.content { + overflow-y: auto; + padding: 0 15px; + padding-bottom: 50px; +} +.content main { + margin-left: auto; + margin-right: auto; + max-width: var(--content-max-width); +} +.content a { text-decoration: none; } +.content a:hover { text-decoration: underline; } +.content img { max-width: 100%; } +.content .header:link, +.content .header:visited { + color: var(--fg); +} +.content .header:link, +.content .header:visited:hover { + text-decoration: none; +} + +table { + margin: 0 auto; + border-collapse: collapse; +} +table td { + padding: 3px 20px; + border: 1px var(--table-border-color) solid; +} +table thead { + background: var(--table-header-bg); +} +table thead td { + font-weight: 700; + border: none; +} +table thead tr { + border: 1px var(--table-header-bg) solid; +} +/* Alternate background colors for rows */ +table tbody tr:nth-child(2n) { + background: var(--table-alternate-bg); +} + + +blockquote { + margin: 20px 0; + padding: 0 20px; + color: var(--fg); + background-color: var(--quote-bg); + border-top: .1em solid var(--quote-border); + border-bottom: .1em solid var(--quote-border); +} + + +:not(.footnote-definition) + .footnote-definition, +.footnote-definition + :not(.footnote-definition) { + margin-top: 2em; +} +.footnote-definition { + font-size: 0.9em; + margin: 0.5em 0; +} +.footnote-definition p { + display: inline; +} + +.tooltiptext { + position: absolute; + visibility: hidden; + color: #fff; + background-color: #333; + transform: translateX(-50%); /* Center by moving tooltip 50% of its width left */ + left: -8px; /* Half of the width of the icon */ + top: -35px; + font-size: 0.8em; + text-align: center; + border-radius: 6px; + padding: 5px 8px; + margin: 5px; + z-index: 1000; +} +.tooltipped .tooltiptext { + visibility: visible; +} + \ No newline at end of file diff --git a/docs/guide/css/print.css b/docs/guide/css/print.css new file mode 100644 index 0000000..5e690f7 --- /dev/null +++ b/docs/guide/css/print.css @@ -0,0 +1,54 @@ + +#sidebar, +#menu-bar, +.nav-chapters, +.mobile-nav-chapters { + display: none; +} + +#page-wrapper.page-wrapper { + transform: none; + margin-left: 0px; + overflow-y: initial; +} + +#content { + max-width: none; + margin: 0; + padding: 0; +} + +.page { + overflow-y: initial; +} + +code { + background-color: #666666; + border-radius: 5px; + + /* Force background to be printed in Chrome */ + -webkit-print-color-adjust: exact; +} + +pre > .buttons { + z-index: 2; +} + +a, a:visited, a:active, a:hover { + color: #4183c4; + text-decoration: none; +} + +h1, h2, h3, h4, h5, h6 { + page-break-inside: avoid; + page-break-after: avoid; +} + +pre, code { + page-break-inside: avoid; + white-space: pre-wrap; +} + +.fa { + display: none !important; +} diff --git a/docs/guide/css/variables.css b/docs/guide/css/variables.css new file mode 100644 index 0000000..29daa07 --- /dev/null +++ b/docs/guide/css/variables.css @@ -0,0 +1,210 @@ + +/* Globals */ + +:root { + --sidebar-width: 300px; + --page-padding: 15px; + --content-max-width: 750px; +} + +/* Themes */ + +.ayu { + --bg: hsl(210, 25%, 8%); + --fg: #c5c5c5; + + --sidebar-bg: #14191f; + --sidebar-fg: #c8c9db; + --sidebar-non-existant: #5c6773; + --sidebar-active: #ffb454; + --sidebar-spacer: #2d334f; + + --scrollbar: var(--sidebar-fg); + + --icons: #737480; + --icons-hover: #b7b9cc; + + --links: #0096cf; + + --inline-code-color: #ffb454; + + --theme-popup-bg: #14191f; + --theme-popup-border: #5c6773; + --theme-hover: #191f26; + + --quote-bg: hsl(226, 15%, 17%); + --quote-border: hsl(226, 15%, 22%); + + --table-border-color: hsl(210, 25%, 13%); + --table-header-bg: hsl(210, 25%, 28%); + --table-alternate-bg: hsl(210, 25%, 11%); + + --searchbar-border-color: #848484; + --searchbar-bg: #424242; + --searchbar-fg: #fff; + --searchbar-shadow-color: #d4c89f; + --searchresults-header-fg: #666; + --searchresults-border-color: #888; + --searchresults-li-bg: #252932; + --search-mark-bg: #e3b171; +} + +.coal { + --bg: hsl(200, 7%, 8%); + --fg: #98a3ad; + + --sidebar-bg: #292c2f; + --sidebar-fg: #a1adb8; + --sidebar-non-existant: #505254; + --sidebar-active: #3473ad; + --sidebar-spacer: #393939; + + --scrollbar: var(--sidebar-fg); + + --icons: #43484d; + --icons-hover: #b3c0cc; + + --links: #2b79a2; + + --inline-code-color: #c5c8c6;; + + --theme-popup-bg: #141617; + --theme-popup-border: #43484d; + --theme-hover: #1f2124; + + --quote-bg: hsl(234, 21%, 18%); + --quote-border: hsl(234, 21%, 23%); + + --table-border-color: hsl(200, 7%, 13%); + --table-header-bg: hsl(200, 7%, 28%); + --table-alternate-bg: hsl(200, 7%, 11%); + + --searchbar-border-color: #aaa; + --searchbar-bg: #b7b7b7; + --searchbar-fg: #000; + --searchbar-shadow-color: #aaa; + --searchresults-header-fg: #666; + --searchresults-border-color: #98a3ad; + --searchresults-li-bg: #2b2b2f; + --search-mark-bg: #355c7d; +} + +.light { + --bg: hsl(0, 0%, 100%); + --fg: #333333; + + --sidebar-bg: #fafafa; + --sidebar-fg: #364149; + --sidebar-non-existant: #aaaaaa; + --sidebar-active: #008cff; + --sidebar-spacer: #f4f4f4; + + --scrollbar: #cccccc; + + --icons: #cccccc; + --icons-hover: #333333; + + --links: #4183c4; + + --inline-code-color: #6e6b5e; + + --theme-popup-bg: #fafafa; + --theme-popup-border: #cccccc; + --theme-hover: #e6e6e6; + + --quote-bg: hsl(197, 37%, 96%); + --quote-border: hsl(197, 37%, 91%); + + --table-border-color: hsl(0, 0%, 95%); + --table-header-bg: hsl(0, 0%, 80%); + --table-alternate-bg: hsl(0, 0%, 97%); + + --searchbar-border-color: #aaa; + --searchbar-bg: #fafafa; + --searchbar-fg: #000; + --searchbar-shadow-color: #aaa; + --searchresults-header-fg: #666; + --searchresults-border-color: #888; + --searchresults-li-bg: #e4f2fe; + --search-mark-bg: #a2cff5; +} + +.navy { + --bg: hsl(226, 23%, 11%); + --fg: #bcbdd0; + + --sidebar-bg: #282d3f; + --sidebar-fg: #c8c9db; + --sidebar-non-existant: #505274; + --sidebar-active: #2b79a2; + --sidebar-spacer: #2d334f; + + --scrollbar: var(--sidebar-fg); + + --icons: #737480; + --icons-hover: #b7b9cc; + + --links: #2b79a2; + + --inline-code-color: #c5c8c6;; + + --theme-popup-bg: #161923; + --theme-popup-border: #737480; + --theme-hover: #282e40; + + --quote-bg: hsl(226, 15%, 17%); + --quote-border: hsl(226, 15%, 22%); + + --table-border-color: hsl(226, 23%, 16%); + --table-header-bg: hsl(226, 23%, 31%); + --table-alternate-bg: hsl(226, 23%, 14%); + + --searchbar-border-color: #aaa; + --searchbar-bg: #aeaec6; + --searchbar-fg: #000; + --searchbar-shadow-color: #aaa; + --searchresults-header-fg: #5f5f71; + --searchresults-border-color: #5c5c68; + --searchresults-li-bg: #242430; + --search-mark-bg: #a2cff5; +} + +.rust { + --bg: hsl(60, 9%, 87%); + --fg: #262625; + + --sidebar-bg: #3b2e2a; + --sidebar-fg: #c8c9db; + --sidebar-non-existant: #505254; + --sidebar-active: #e69f67; + --sidebar-spacer: #45373a; + + --scrollbar: var(--sidebar-fg); + + --icons: #737480; + --icons-hover: #262625; + + --links: #2b79a2; + + --inline-code-color: #6e6b5e; + + --theme-popup-bg: #e1e1db; + --theme-popup-border: #b38f6b; + --theme-hover: #99908a; + + --quote-bg: hsl(60, 5%, 75%); + --quote-border: hsl(60, 5%, 70%); + + --table-border-color: hsl(60, 9%, 82%); + --table-header-bg: #b3a497; + --table-alternate-bg: hsl(60, 9%, 84%); + + --searchbar-border-color: #aaa; + --searchbar-bg: #fafafa; + --searchbar-fg: #000; + --searchbar-shadow-color: #aaa; + --searchresults-header-fg: #666; + --searchresults-border-color: #888; + --searchresults-li-bg: #dec2a2; + --search-mark-bg: #e69f67; +} diff --git a/docs/guide/elasticlunr.min.js b/docs/guide/elasticlunr.min.js new file mode 100644 index 0000000..94b20dd --- /dev/null +++ b/docs/guide/elasticlunr.min.js @@ -0,0 +1,10 @@ +/** + * elasticlunr - http://weixsong.github.io + * Lightweight full-text search engine in Javascript for browser search and offline search. - 0.9.5 + * + * Copyright (C) 2017 Oliver Nightingale + * Copyright (C) 2017 Wei Song + * MIT Licensed + * @license + */ +!function(){function e(e){if(null===e||"object"!=typeof e)return e;var t=e.constructor();for(var n in e)e.hasOwnProperty(n)&&(t[n]=e[n]);return t}var t=function(e){var n=new t.Index;return n.pipeline.add(t.trimmer,t.stopWordFilter,t.stemmer),e&&e.call(n,n),n};t.version="0.9.5",lunr=t,t.utils={},t.utils.warn=function(e){return function(t){e.console&&console.warn&&console.warn(t)}}(this),t.utils.toString=function(e){return void 0===e||null===e?"":e.toString()},t.EventEmitter=function(){this.events={}},t.EventEmitter.prototype.addListener=function(){var e=Array.prototype.slice.call(arguments),t=e.pop(),n=e;if("function"!=typeof t)throw new TypeError("last argument must be a function");n.forEach(function(e){this.hasHandler(e)||(this.events[e]=[]),this.events[e].push(t)},this)},t.EventEmitter.prototype.removeListener=function(e,t){if(this.hasHandler(e)){var n=this.events[e].indexOf(t);-1!==n&&(this.events[e].splice(n,1),0==this.events[e].length&&delete this.events[e])}},t.EventEmitter.prototype.emit=function(e){if(this.hasHandler(e)){var t=Array.prototype.slice.call(arguments,1);this.events[e].forEach(function(e){e.apply(void 0,t)},this)}},t.EventEmitter.prototype.hasHandler=function(e){return e in this.events},t.tokenizer=function(e){if(!arguments.length||null===e||void 0===e)return[];if(Array.isArray(e)){var n=e.filter(function(e){return null===e||void 0===e?!1:!0});n=n.map(function(e){return t.utils.toString(e).toLowerCase()});var i=[];return n.forEach(function(e){var n=e.split(t.tokenizer.seperator);i=i.concat(n)},this),i}return e.toString().trim().toLowerCase().split(t.tokenizer.seperator)},t.tokenizer.defaultSeperator=/[\s\-]+/,t.tokenizer.seperator=t.tokenizer.defaultSeperator,t.tokenizer.setSeperator=function(e){null!==e&&void 0!==e&&"object"==typeof e&&(t.tokenizer.seperator=e)},t.tokenizer.resetSeperator=function(){t.tokenizer.seperator=t.tokenizer.defaultSeperator},t.tokenizer.getSeperator=function(){return t.tokenizer.seperator},t.Pipeline=function(){this._queue=[]},t.Pipeline.registeredFunctions={},t.Pipeline.registerFunction=function(e,n){n in t.Pipeline.registeredFunctions&&t.utils.warn("Overwriting existing registered function: "+n),e.label=n,t.Pipeline.registeredFunctions[n]=e},t.Pipeline.getRegisteredFunction=function(e){return e in t.Pipeline.registeredFunctions!=!0?null:t.Pipeline.registeredFunctions[e]},t.Pipeline.warnIfFunctionNotRegistered=function(e){var n=e.label&&e.label in this.registeredFunctions;n||t.utils.warn("Function is not registered with pipeline. This may cause problems when serialising the index.\n",e)},t.Pipeline.load=function(e){var n=new t.Pipeline;return e.forEach(function(e){var i=t.Pipeline.getRegisteredFunction(e);if(!i)throw new Error("Cannot load un-registered function: "+e);n.add(i)}),n},t.Pipeline.prototype.add=function(){var e=Array.prototype.slice.call(arguments);e.forEach(function(e){t.Pipeline.warnIfFunctionNotRegistered(e),this._queue.push(e)},this)},t.Pipeline.prototype.after=function(e,n){t.Pipeline.warnIfFunctionNotRegistered(n);var i=this._queue.indexOf(e);if(-1===i)throw new Error("Cannot find existingFn");this._queue.splice(i+1,0,n)},t.Pipeline.prototype.before=function(e,n){t.Pipeline.warnIfFunctionNotRegistered(n);var i=this._queue.indexOf(e);if(-1===i)throw new Error("Cannot find existingFn");this._queue.splice(i,0,n)},t.Pipeline.prototype.remove=function(e){var t=this._queue.indexOf(e);-1!==t&&this._queue.splice(t,1)},t.Pipeline.prototype.run=function(e){for(var t=[],n=e.length,i=this._queue.length,o=0;n>o;o++){for(var r=e[o],s=0;i>s&&(r=this._queue[s](r,o,e),void 0!==r&&null!==r);s++);void 0!==r&&null!==r&&t.push(r)}return t},t.Pipeline.prototype.reset=function(){this._queue=[]},t.Pipeline.prototype.get=function(){return this._queue},t.Pipeline.prototype.toJSON=function(){return this._queue.map(function(e){return t.Pipeline.warnIfFunctionNotRegistered(e),e.label})},t.Index=function(){this._fields=[],this._ref="id",this.pipeline=new t.Pipeline,this.documentStore=new t.DocumentStore,this.index={},this.eventEmitter=new t.EventEmitter,this._idfCache={},this.on("add","remove","update",function(){this._idfCache={}}.bind(this))},t.Index.prototype.on=function(){var e=Array.prototype.slice.call(arguments);return this.eventEmitter.addListener.apply(this.eventEmitter,e)},t.Index.prototype.off=function(e,t){return this.eventEmitter.removeListener(e,t)},t.Index.load=function(e){e.version!==t.version&&t.utils.warn("version mismatch: current "+t.version+" importing "+e.version);var n=new this;n._fields=e.fields,n._ref=e.ref,n.documentStore=t.DocumentStore.load(e.documentStore),n.pipeline=t.Pipeline.load(e.pipeline),n.index={};for(var i in e.index)n.index[i]=t.InvertedIndex.load(e.index[i]);return n},t.Index.prototype.addField=function(e){return this._fields.push(e),this.index[e]=new t.InvertedIndex,this},t.Index.prototype.setRef=function(e){return this._ref=e,this},t.Index.prototype.saveDocument=function(e){return this.documentStore=new t.DocumentStore(e),this},t.Index.prototype.addDoc=function(e,n){if(e){var n=void 0===n?!0:n,i=e[this._ref];this.documentStore.addDoc(i,e),this._fields.forEach(function(n){var o=this.pipeline.run(t.tokenizer(e[n]));this.documentStore.addFieldLength(i,n,o.length);var r={};o.forEach(function(e){e in r?r[e]+=1:r[e]=1},this);for(var s in r){var u=r[s];u=Math.sqrt(u),this.index[n].addToken(s,{ref:i,tf:u})}},this),n&&this.eventEmitter.emit("add",e,this)}},t.Index.prototype.removeDocByRef=function(e){if(e&&this.documentStore.isDocStored()!==!1&&this.documentStore.hasDoc(e)){var t=this.documentStore.getDoc(e);this.removeDoc(t,!1)}},t.Index.prototype.removeDoc=function(e,n){if(e){var n=void 0===n?!0:n,i=e[this._ref];this.documentStore.hasDoc(i)&&(this.documentStore.removeDoc(i),this._fields.forEach(function(n){var o=this.pipeline.run(t.tokenizer(e[n]));o.forEach(function(e){this.index[n].removeToken(e,i)},this)},this),n&&this.eventEmitter.emit("remove",e,this))}},t.Index.prototype.updateDoc=function(e,t){var t=void 0===t?!0:t;this.removeDocByRef(e[this._ref],!1),this.addDoc(e,!1),t&&this.eventEmitter.emit("update",e,this)},t.Index.prototype.idf=function(e,t){var n="@"+t+"/"+e;if(Object.prototype.hasOwnProperty.call(this._idfCache,n))return this._idfCache[n];var i=this.index[t].getDocFreq(e),o=1+Math.log(this.documentStore.length/(i+1));return this._idfCache[n]=o,o},t.Index.prototype.getFields=function(){return this._fields.slice()},t.Index.prototype.search=function(e,n){if(!e)return[];e="string"==typeof e?{any:e}:JSON.parse(JSON.stringify(e));var i=null;null!=n&&(i=JSON.stringify(n));for(var o=new t.Configuration(i,this.getFields()).get(),r={},s=Object.keys(e),u=0;u0&&t.push(e);for(var i in n)"docs"!==i&&"df"!==i&&this.expandToken(e+i,t,n[i]);return t},t.InvertedIndex.prototype.toJSON=function(){return{root:this.root}},t.Configuration=function(e,n){var e=e||"";if(void 0==n||null==n)throw new Error("fields should not be null");this.config={};var i;try{i=JSON.parse(e),this.buildUserConfig(i,n)}catch(o){t.utils.warn("user configuration parse failed, will use default configuration"),this.buildDefaultConfig(n)}},t.Configuration.prototype.buildDefaultConfig=function(e){this.reset(),e.forEach(function(e){this.config[e]={boost:1,bool:"OR",expand:!1}},this)},t.Configuration.prototype.buildUserConfig=function(e,n){var i="OR",o=!1;if(this.reset(),"bool"in e&&(i=e.bool||i),"expand"in e&&(o=e.expand||o),"fields"in e)for(var r in e.fields)if(n.indexOf(r)>-1){var s=e.fields[r],u=o;void 0!=s.expand&&(u=s.expand),this.config[r]={boost:s.boost||0===s.boost?s.boost:1,bool:s.bool||i,expand:u}}else t.utils.warn("field name in user configuration not found in index instance fields");else this.addAllFields2UserConfig(i,o,n)},t.Configuration.prototype.addAllFields2UserConfig=function(e,t,n){n.forEach(function(n){this.config[n]={boost:1,bool:e,expand:t}},this)},t.Configuration.prototype.get=function(){return this.config},t.Configuration.prototype.reset=function(){this.config={}},lunr.SortedSet=function(){this.length=0,this.elements=[]},lunr.SortedSet.load=function(e){var t=new this;return t.elements=e,t.length=e.length,t},lunr.SortedSet.prototype.add=function(){var e,t;for(e=0;e1;){if(r===e)return o;e>r&&(t=o),r>e&&(n=o),i=n-t,o=t+Math.floor(i/2),r=this.elements[o]}return r===e?o:-1},lunr.SortedSet.prototype.locationFor=function(e){for(var t=0,n=this.elements.length,i=n-t,o=t+Math.floor(i/2),r=this.elements[o];i>1;)e>r&&(t=o),r>e&&(n=o),i=n-t,o=t+Math.floor(i/2),r=this.elements[o];return r>e?o:e>r?o+1:void 0},lunr.SortedSet.prototype.intersect=function(e){for(var t=new lunr.SortedSet,n=0,i=0,o=this.length,r=e.length,s=this.elements,u=e.elements;;){if(n>o-1||i>r-1)break;s[n]!==u[i]?s[n]u[i]&&i++:(t.add(s[n]),n++,i++)}return t},lunr.SortedSet.prototype.clone=function(){var e=new lunr.SortedSet;return e.elements=this.toArray(),e.length=e.elements.length,e},lunr.SortedSet.prototype.union=function(e){var t,n,i;this.length>=e.length?(t=this,n=e):(t=e,n=this),i=t.clone();for(var o=0,r=n.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 A;{X5v literal 0 HcmV?d00001 diff --git a/docs/guide/highlight.css b/docs/guide/highlight.css new file mode 100644 index 0000000..c667e3a --- /dev/null +++ b/docs/guide/highlight.css @@ -0,0 +1,69 @@ +/* Base16 Atelier Dune Light - Theme */ +/* by Bram de Haan (http://atelierbram.github.io/syntax-highlighting/atelier-schemes/dune) */ +/* Original Base16 color scheme by Chris Kempson (https://github.com/chriskempson/base16) */ + +/* Atelier-Dune Comment */ +.hljs-comment, +.hljs-quote { + color: #AAA; +} + +/* Atelier-Dune Red */ +.hljs-variable, +.hljs-template-variable, +.hljs-attribute, +.hljs-tag, +.hljs-name, +.hljs-regexp, +.hljs-link, +.hljs-name, +.hljs-selector-id, +.hljs-selector-class { + color: #d73737; +} + +/* Atelier-Dune Orange */ +.hljs-number, +.hljs-meta, +.hljs-built_in, +.hljs-builtin-name, +.hljs-literal, +.hljs-type, +.hljs-params { + color: #b65611; +} + +/* Atelier-Dune Green */ +.hljs-string, +.hljs-symbol, +.hljs-bullet { + color: #60ac39; +} + +/* Atelier-Dune Blue */ +.hljs-title, +.hljs-section { + color: #6684e1; +} + +/* Atelier-Dune Purple */ +.hljs-keyword, +.hljs-selector-tag { + color: #b854d4; +} + +.hljs { + display: block; + overflow-x: auto; + background: #f1f1f1; + color: #6e6b5e; + padding: 0.5em; +} + +.hljs-emphasis { + font-style: italic; +} + +.hljs-strong { + font-weight: bold; +} diff --git a/docs/guide/highlight.js b/docs/guide/highlight.js new file mode 100644 index 0000000..bb48d11 --- /dev/null +++ b/docs/guide/highlight.js @@ -0,0 +1,2 @@ +/*! highlight.js v9.12.0 | BSD3 License | git.io/hljslicense */ +!function(e){var n="object"==typeof window&&window||"object"==typeof self&&self;"undefined"!=typeof exports?e(exports):n&&(n.hljs=e({}),"function"==typeof define&&define.amd&&define([],function(){return n.hljs}))}(function(e){function n(e){return e.replace(/&/g,"&").replace(//g,">")}function t(e){return e.nodeName.toLowerCase()}function r(e,n){var t=e&&e.exec(n);return t&&0===t.index}function a(e){return k.test(e)}function i(e){var n,t,r,i,o=e.className+" ";if(o+=e.parentNode?e.parentNode.className:"",t=B.exec(o))return w(t[1])?t[1]:"no-highlight";for(o=o.split(/\s+/),n=0,r=o.length;r>n;n++)if(i=o[n],a(i)||w(i))return i}function o(e){var n,t={},r=Array.prototype.slice.call(arguments,1);for(n in e)t[n]=e[n];return r.forEach(function(e){for(n in e)t[n]=e[n]}),t}function u(e){var n=[];return function r(e,a){for(var i=e.firstChild;i;i=i.nextSibling)3===i.nodeType?a+=i.nodeValue.length:1===i.nodeType&&(n.push({event:"start",offset:a,node:i}),a=r(i,a),t(i).match(/br|hr|img|input/)||n.push({event:"stop",offset:a,node:i}));return a}(e,0),n}function c(e,r,a){function i(){return e.length&&r.length?e[0].offset!==r[0].offset?e[0].offset"}function u(e){s+=""}function c(e){("start"===e.event?o:u)(e.node)}for(var l=0,s="",f=[];e.length||r.length;){var g=i();if(s+=n(a.substring(l,g[0].offset)),l=g[0].offset,g===e){f.reverse().forEach(u);do c(g.splice(0,1)[0]),g=i();while(g===e&&g.length&&g[0].offset===l);f.reverse().forEach(o)}else"start"===g[0].event?f.push(g[0].node):f.pop(),c(g.splice(0,1)[0])}return s+n(a.substr(l))}function l(e){return e.v&&!e.cached_variants&&(e.cached_variants=e.v.map(function(n){return o(e,{v:null},n)})),e.cached_variants||e.eW&&[o(e)]||[e]}function s(e){function n(e){return e&&e.source||e}function t(t,r){return new RegExp(n(t),"m"+(e.cI?"i":"")+(r?"g":""))}function r(a,i){if(!a.compiled){if(a.compiled=!0,a.k=a.k||a.bK,a.k){var o={},u=function(n,t){e.cI&&(t=t.toLowerCase()),t.split(" ").forEach(function(e){var t=e.split("|");o[t[0]]=[n,t[1]?Number(t[1]):1]})};"string"==typeof a.k?u("keyword",a.k):x(a.k).forEach(function(e){u(e,a.k[e])}),a.k=o}a.lR=t(a.l||/\w+/,!0),i&&(a.bK&&(a.b="\\b("+a.bK.split(" ").join("|")+")\\b"),a.b||(a.b=/\B|\b/),a.bR=t(a.b),a.e||a.eW||(a.e=/\B|\b/),a.e&&(a.eR=t(a.e)),a.tE=n(a.e)||"",a.eW&&i.tE&&(a.tE+=(a.e?"|":"")+i.tE)),a.i&&(a.iR=t(a.i)),null==a.r&&(a.r=1),a.c||(a.c=[]),a.c=Array.prototype.concat.apply([],a.c.map(function(e){return l("self"===e?a:e)})),a.c.forEach(function(e){r(e,a)}),a.starts&&r(a.starts,i);var c=a.c.map(function(e){return e.bK?"\\.?("+e.b+")\\.?":e.b}).concat([a.tE,a.i]).map(n).filter(Boolean);a.t=c.length?t(c.join("|"),!0):{exec:function(){return null}}}}r(e)}function f(e,t,a,i){function o(e,n){var t,a;for(t=0,a=n.c.length;a>t;t++)if(r(n.c[t].bR,e))return n.c[t]}function u(e,n){if(r(e.eR,n)){for(;e.endsParent&&e.parent;)e=e.parent;return e}return e.eW?u(e.parent,n):void 0}function c(e,n){return!a&&r(n.iR,e)}function l(e,n){var t=N.cI?n[0].toLowerCase():n[0];return e.k.hasOwnProperty(t)&&e.k[t]}function p(e,n,t,r){var a=r?"":I.classPrefix,i='',i+n+o}function h(){var e,t,r,a;if(!E.k)return n(k);for(a="",t=0,E.lR.lastIndex=0,r=E.lR.exec(k);r;)a+=n(k.substring(t,r.index)),e=l(E,r),e?(B+=e[1],a+=p(e[0],n(r[0]))):a+=n(r[0]),t=E.lR.lastIndex,r=E.lR.exec(k);return a+n(k.substr(t))}function d(){var e="string"==typeof E.sL;if(e&&!y[E.sL])return n(k);var t=e?f(E.sL,k,!0,x[E.sL]):g(k,E.sL.length?E.sL:void 0);return E.r>0&&(B+=t.r),e&&(x[E.sL]=t.top),p(t.language,t.value,!1,!0)}function b(){L+=null!=E.sL?d():h(),k=""}function v(e){L+=e.cN?p(e.cN,"",!0):"",E=Object.create(e,{parent:{value:E}})}function m(e,n){if(k+=e,null==n)return b(),0;var t=o(n,E);if(t)return t.skip?k+=n:(t.eB&&(k+=n),b(),t.rB||t.eB||(k=n)),v(t,n),t.rB?0:n.length;var r=u(E,n);if(r){var a=E;a.skip?k+=n:(a.rE||a.eE||(k+=n),b(),a.eE&&(k=n));do E.cN&&(L+=C),E.skip||(B+=E.r),E=E.parent;while(E!==r.parent);return r.starts&&v(r.starts,""),a.rE?0:n.length}if(c(n,E))throw new Error('Illegal lexeme "'+n+'" for mode "'+(E.cN||"")+'"');return k+=n,n.length||1}var N=w(e);if(!N)throw new Error('Unknown language: "'+e+'"');s(N);var R,E=i||N,x={},L="";for(R=E;R!==N;R=R.parent)R.cN&&(L=p(R.cN,"",!0)+L);var k="",B=0;try{for(var M,j,O=0;;){if(E.t.lastIndex=O,M=E.t.exec(t),!M)break;j=m(t.substring(O,M.index),M[0]),O=M.index+j}for(m(t.substr(O)),R=E;R.parent;R=R.parent)R.cN&&(L+=C);return{r:B,value:L,language:e,top:E}}catch(T){if(T.message&&-1!==T.message.indexOf("Illegal"))return{r:0,value:n(t)};throw T}}function g(e,t){t=t||I.languages||x(y);var r={r:0,value:n(e)},a=r;return t.filter(w).forEach(function(n){var t=f(n,e,!1);t.language=n,t.r>a.r&&(a=t),t.r>r.r&&(a=r,r=t)}),a.language&&(r.second_best=a),r}function p(e){return I.tabReplace||I.useBR?e.replace(M,function(e,n){return I.useBR&&"\n"===e?"
":I.tabReplace?n.replace(/\t/g,I.tabReplace):""}):e}function h(e,n,t){var r=n?L[n]:t,a=[e.trim()];return e.match(/\bhljs\b/)||a.push("hljs"),-1===e.indexOf(r)&&a.push(r),a.join(" ").trim()}function d(e){var n,t,r,o,l,s=i(e);a(s)||(I.useBR?(n=document.createElementNS("http://www.w3.org/1999/xhtml","div"),n.innerHTML=e.innerHTML.replace(/\n/g,"").replace(//g,"\n")):n=e,l=n.textContent,r=s?f(s,l,!0):g(l),t=u(n),t.length&&(o=document.createElementNS("http://www.w3.org/1999/xhtml","div"),o.innerHTML=r.value,r.value=c(t,u(o),l)),r.value=p(r.value),e.innerHTML=r.value,e.className=h(e.className,s,r.language),e.result={language:r.language,re:r.r},r.second_best&&(e.second_best={language:r.second_best.language,re:r.second_best.r}))}function b(e){I=o(I,e)}function v(){if(!v.called){v.called=!0;var e=document.querySelectorAll("pre code");E.forEach.call(e,d)}}function m(){addEventListener("DOMContentLoaded",v,!1),addEventListener("load",v,!1)}function N(n,t){var r=y[n]=t(e);r.aliases&&r.aliases.forEach(function(e){L[e]=n})}function R(){return x(y)}function w(e){return e=(e||"").toLowerCase(),y[e]||y[L[e]]}var E=[],x=Object.keys,y={},L={},k=/^(no-?highlight|plain|text)$/i,B=/\blang(?:uage)?-([\w-]+)\b/i,M=/((^(<[^>]+>|\t|)+|(?:\n)))/gm,C="
",I={classPrefix:"hljs-",tabReplace:null,useBR:!1,languages:void 0};return e.highlight=f,e.highlightAuto=g,e.fixMarkup=p,e.highlightBlock=d,e.configure=b,e.initHighlighting=v,e.initHighlightingOnLoad=m,e.registerLanguage=N,e.listLanguages=R,e.getLanguage=w,e.inherit=o,e.IR="[a-zA-Z]\\w*",e.UIR="[a-zA-Z_]\\w*",e.NR="\\b\\d+(\\.\\d+)?",e.CNR="(-?)(\\b0[xX][a-fA-F0-9]+|(\\b\\d+(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)",e.BNR="\\b(0b[01]+)",e.RSR="!|!=|!==|%|%=|&|&&|&=|\\*|\\*=|\\+|\\+=|,|-|-=|/=|/|:|;|<<|<<=|<=|<|===|==|=|>>>=|>>=|>=|>>>|>>|>|\\?|\\[|\\{|\\(|\\^|\\^=|\\||\\|=|\\|\\||~",e.BE={b:"\\\\[\\s\\S]",r:0},e.ASM={cN:"string",b:"'",e:"'",i:"\\n",c:[e.BE]},e.QSM={cN:"string",b:'"',e:'"',i:"\\n",c:[e.BE]},e.PWM={b:/\b(a|an|the|are|I'm|isn't|don't|doesn't|won't|but|just|should|pretty|simply|enough|gonna|going|wtf|so|such|will|you|your|they|like|more)\b/},e.C=function(n,t,r){var a=e.inherit({cN:"comment",b:n,e:t,c:[]},r||{});return a.c.push(e.PWM),a.c.push({cN:"doctag",b:"(?:TODO|FIXME|NOTE|BUG|XXX):",r:0}),a},e.CLCM=e.C("//","$"),e.CBCM=e.C("/\\*","\\*/"),e.HCM=e.C("#","$"),e.NM={cN:"number",b:e.NR,r:0},e.CNM={cN:"number",b:e.CNR,r:0},e.BNM={cN:"number",b:e.BNR,r:0},e.CSSNM={cN:"number",b:e.NR+"(%|em|ex|ch|rem|vw|vh|vmin|vmax|cm|mm|in|pt|pc|px|deg|grad|rad|turn|s|ms|Hz|kHz|dpi|dpcm|dppx)?",r:0},e.RM={cN:"regexp",b:/\//,e:/\/[gimuy]*/,i:/\n/,c:[e.BE,{b:/\[/,e:/\]/,r:0,c:[e.BE]}]},e.TM={cN:"title",b:e.IR,r:0},e.UTM={cN:"title",b:e.UIR,r:0},e.METHOD_GUARD={b:"\\.\\s*"+e.UIR,r:0},e});hljs.registerLanguage("diff",function(e){return{aliases:["patch"],c:[{cN:"meta",r:10,v:[{b:/^@@ +\-\d+,\d+ +\+\d+,\d+ +@@$/},{b:/^\*\*\* +\d+,\d+ +\*\*\*\*$/},{b:/^\-\-\- +\d+,\d+ +\-\-\-\-$/}]},{cN:"comment",v:[{b:/Index: /,e:/$/},{b:/={3,}/,e:/$/},{b:/^\-{3}/,e:/$/},{b:/^\*{3} /,e:/$/},{b:/^\+{3}/,e:/$/},{b:/\*{5}/,e:/\*{5}$/}]},{cN:"addition",b:"^\\+",e:"$"},{cN:"deletion",b:"^\\-",e:"$"},{cN:"addition",b:"^\\!",e:"$"}]}});hljs.registerLanguage("nginx",function(e){var r={cN:"variable",v:[{b:/\$\d+/},{b:/\$\{/,e:/}/},{b:"[\\$\\@]"+e.UIR}]},b={eW:!0,l:"[a-z/_]+",k:{literal:"on off yes no true false none blocked debug info notice warn error crit select break last permanent redirect kqueue rtsig epoll poll /dev/poll"},r:0,i:"=>",c:[e.HCM,{cN:"string",c:[e.BE,r],v:[{b:/"/,e:/"/},{b:/'/,e:/'/}]},{b:"([a-z]+):/",e:"\\s",eW:!0,eE:!0,c:[r]},{cN:"regexp",c:[e.BE,r],v:[{b:"\\s\\^",e:"\\s|{|;",rE:!0},{b:"~\\*?\\s+",e:"\\s|{|;",rE:!0},{b:"\\*(\\.[a-z\\-]+)+"},{b:"([a-z\\-]+\\.)+\\*"}]},{cN:"number",b:"\\b\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}\\.\\d{1,3}(:\\d{1,5})?\\b"},{cN:"number",b:"\\b\\d+[kKmMgGdshdwy]*\\b",r:0},r]};return{aliases:["nginxconf"],c:[e.HCM,{b:e.UIR+"\\s+{",rB:!0,e:"{",c:[{cN:"section",b:e.UIR}],r:0},{b:e.UIR+"\\s",e:";|{",rB:!0,c:[{cN:"attribute",b:e.UIR,starts:b}],r:0}],i:"[^\\s\\}]"}});hljs.registerLanguage("objectivec",function(e){var t={cN:"built_in",b:"\\b(AV|CA|CF|CG|CI|CL|CM|CN|CT|MK|MP|MTK|MTL|NS|SCN|SK|UI|WK|XC)\\w+"},_={keyword:"int float while char export sizeof typedef const struct for union unsigned long volatile static bool mutable if do return goto void enum else break extern asm case short default double register explicit signed typename this switch continue wchar_t inline readonly assign readwrite self @synchronized id typeof nonatomic super unichar IBOutlet IBAction strong weak copy in out inout bycopy byref oneway __strong __weak __block __autoreleasing @private @protected @public @try @property @end @throw @catch @finally @autoreleasepool @synthesize @dynamic @selector @optional @required @encode @package @import @defs @compatibility_alias __bridge __bridge_transfer __bridge_retained __bridge_retain __covariant __contravariant __kindof _Nonnull _Nullable _Null_unspecified __FUNCTION__ __PRETTY_FUNCTION__ __attribute__ getter setter retain unsafe_unretained nonnull nullable null_unspecified null_resettable class instancetype NS_DESIGNATED_INITIALIZER NS_UNAVAILABLE NS_REQUIRES_SUPER NS_RETURNS_INNER_POINTER NS_INLINE NS_AVAILABLE NS_DEPRECATED NS_ENUM NS_OPTIONS NS_SWIFT_UNAVAILABLE NS_ASSUME_NONNULL_BEGIN NS_ASSUME_NONNULL_END NS_REFINED_FOR_SWIFT NS_SWIFT_NAME NS_SWIFT_NOTHROW NS_DURING NS_HANDLER NS_ENDHANDLER NS_VALUERETURN NS_VOIDRETURN",literal:"false true FALSE TRUE nil YES NO NULL",built_in:"BOOL dispatch_once_t dispatch_queue_t dispatch_sync dispatch_async dispatch_once"},i=/[a-zA-Z@][a-zA-Z0-9_]*/,n="@interface @class @protocol @implementation";return{aliases:["mm","objc","obj-c"],k:_,l:i,i:""}]}]},{cN:"class",b:"("+n.split(" ").join("|")+")\\b",e:"({|$)",eE:!0,k:n,l:i,c:[e.UTM]},{b:"\\."+e.UIR,r:0}]}});hljs.registerLanguage("xml",function(s){var e="[A-Za-z0-9\\._:-]+",t={eW:!0,i:/`]+/}]}]}]};return{aliases:["html","xhtml","rss","atom","xjb","xsd","xsl","plist"],cI:!0,c:[{cN:"meta",b:"",r:10,c:[{b:"\\[",e:"\\]"}]},s.C("",{r:10}),{b:"<\\!\\[CDATA\\[",e:"\\]\\]>",r:10},{b:/<\?(php)?/,e:/\?>/,sL:"php",c:[{b:"/\\*",e:"\\*/",skip:!0}]},{cN:"tag",b:"|$)",e:">",k:{name:"style"},c:[t],starts:{e:"",rE:!0,sL:["css","xml"]}},{cN:"tag",b:"|$)",e:">",k:{name:"script"},c:[t],starts:{e:"",rE:!0,sL:["actionscript","javascript","handlebars","xml"]}},{cN:"meta",v:[{b:/<\?xml/,e:/\?>/,r:10},{b:/<\?\w+/,e:/\?>/}]},{cN:"tag",b:"",c:[{cN:"name",b:/[^\/><\s]+/,r:0},t]}]}});hljs.registerLanguage("handlebars",function(e){var a={"builtin-name":"each in with if else unless bindattr action collection debugger log outlet template unbound view yield"};return{aliases:["hbs","html.hbs","html.handlebars"],cI:!0,sL:"xml",c:[e.C("{{!(--)?","(--)?}}"),{cN:"template-tag",b:/\{\{[#\/]/,e:/\}\}/,c:[{cN:"name",b:/[a-zA-Z\.-]+/,k:a,starts:{eW:!0,r:0,c:[e.QSM]}}]},{cN:"template-variable",b:/\{\{/,e:/\}\}/,k:a}]}});hljs.registerLanguage("ini",function(e){var b={cN:"string",c:[e.BE],v:[{b:"'''",e:"'''",r:10},{b:'"""',e:'"""',r:10},{b:'"',e:'"'},{b:"'",e:"'"}]};return{aliases:["toml"],cI:!0,i:/\S/,c:[e.C(";","$"),e.HCM,{cN:"section",b:/^\s*\[+/,e:/\]+/},{b:/^[a-z0-9\[\]_-]+\s*=\s*/,e:"$",rB:!0,c:[{cN:"attr",b:/[a-z0-9\[\]_-]+/},{b:/=/,eW:!0,r:0,c:[{cN:"literal",b:/\bon|off|true|false|yes|no\b/},{cN:"variable",v:[{b:/\$[\w\d"][\w\d_]*/},{b:/\$\{(.*?)}/}]},b,{cN:"number",b:/([\+\-]+)?[\d]+_[\d_]+/},e.NM]}]}]}});hljs.registerLanguage("javascript",function(e){var r="[A-Za-z$_][0-9A-Za-z$_]*",t={keyword:"in of if for while finally var new function do return void else break catch instanceof with throw case default try this switch continue typeof delete let yield const export super debugger as async await static import from as",literal:"true false null undefined NaN Infinity",built_in:"eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent encodeURI encodeURIComponent escape unescape Object Function Boolean Error EvalError InternalError RangeError ReferenceError StopIteration SyntaxError TypeError URIError Number Math Date String RegExp Array Float32Array Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require module console window document Symbol Set Map WeakSet WeakMap Proxy Reflect Promise"},a={cN:"number",v:[{b:"\\b(0[bB][01]+)"},{b:"\\b(0[oO][0-7]+)"},{b:e.CNR}],r:0},n={cN:"subst",b:"\\$\\{",e:"\\}",k:t,c:[]},c={cN:"string",b:"`",e:"`",c:[e.BE,n]};n.c=[e.ASM,e.QSM,c,a,e.RM];var s=n.c.concat([e.CBCM,e.CLCM]);return{aliases:["js","jsx"],k:t,c:[{cN:"meta",r:10,b:/^\s*['"]use (strict|asm)['"]/},{cN:"meta",b:/^#!/,e:/$/},e.ASM,e.QSM,c,e.CLCM,e.CBCM,a,{b:/[{,]\s*/,r:0,c:[{b:r+"\\s*:",rB:!0,r:0,c:[{cN:"attr",b:r,r:0}]}]},{b:"("+e.RSR+"|\\b(case|return|throw)\\b)\\s*",k:"return throw case",c:[e.CLCM,e.CBCM,e.RM,{cN:"function",b:"(\\(.*?\\)|"+r+")\\s*=>",rB:!0,e:"\\s*=>",c:[{cN:"params",v:[{b:r},{b:/\(\s*\)/},{b:/\(/,e:/\)/,eB:!0,eE:!0,k:t,c:s}]}]},{b://,sL:"xml",c:[{b:/<\w+\s*\/>/,skip:!0},{b:/<\w+/,e:/(\/\w+|\w+\/)>/,skip:!0,c:[{b:/<\w+\s*\/>/,skip:!0},"self"]}]}],r:0},{cN:"function",bK:"function",e:/\{/,eE:!0,c:[e.inherit(e.TM,{b:r}),{cN:"params",b:/\(/,e:/\)/,eB:!0,eE:!0,c:s}],i:/\[|%/},{b:/\$[(.]/},e.METHOD_GUARD,{cN:"class",bK:"class",e:/[{;=]/,eE:!0,i:/[:"\[\]]/,c:[{bK:"extends"},e.UTM]},{bK:"constructor",e:/\{/,eE:!0}],i:/#(?!!)/}});hljs.registerLanguage("python",function(e){var r={keyword:"and elif is global as in if from raise for except finally print import pass return exec else break not with class assert yield try while continue del or def lambda async await nonlocal|10 None True False",built_in:"Ellipsis NotImplemented"},b={cN:"meta",b:/^(>>>|\.\.\.) /},c={cN:"subst",b:/\{/,e:/\}/,k:r,i:/#/},a={cN:"string",c:[e.BE],v:[{b:/(u|b)?r?'''/,e:/'''/,c:[b],r:10},{b:/(u|b)?r?"""/,e:/"""/,c:[b],r:10},{b:/(fr|rf|f)'''/,e:/'''/,c:[b,c]},{b:/(fr|rf|f)"""/,e:/"""/,c:[b,c]},{b:/(u|r|ur)'/,e:/'/,r:10},{b:/(u|r|ur)"/,e:/"/,r:10},{b:/(b|br)'/,e:/'/},{b:/(b|br)"/,e:/"/},{b:/(fr|rf|f)'/,e:/'/,c:[c]},{b:/(fr|rf|f)"/,e:/"/,c:[c]},e.ASM,e.QSM]},s={cN:"number",r:0,v:[{b:e.BNR+"[lLjJ]?"},{b:"\\b(0o[0-7]+)[lLjJ]?"},{b:e.CNR+"[lLjJ]?"}]},i={cN:"params",b:/\(/,e:/\)/,c:["self",b,s,a]};return c.c=[a,s,b],{aliases:["py","gyp"],k:r,i:/(<\/|->|\?)|=>/,c:[b,s,a,e.HCM,{v:[{cN:"function",bK:"def"},{cN:"class",bK:"class"}],e:/:/,i:/[${=;\n,]/,c:[e.UTM,i,{b:/->/,eW:!0,k:"None"}]},{cN:"meta",b:/^[\t ]*@/,e:/$/},{b:/\b(print|exec)\(/}]}});hljs.registerLanguage("markdown",function(e){return{aliases:["md","mkdown","mkd"],c:[{cN:"section",v:[{b:"^#{1,6}",e:"$"},{b:"^.+?\\n[=-]{2,}$"}]},{b:"<",e:">",sL:"xml",r:0},{cN:"bullet",b:"^([*+-]|(\\d+\\.))\\s+"},{cN:"strong",b:"[*_]{2}.+?[*_]{2}"},{cN:"emphasis",v:[{b:"\\*.+?\\*"},{b:"_.+?_",r:0}]},{cN:"quote",b:"^>\\s+",e:"$"},{cN:"code",v:[{b:"^```w*s*$",e:"^```s*$"},{b:"`.+?`"},{b:"^( {4}| )",e:"$",r:0}]},{b:"^[-\\*]{3,}",e:"$"},{b:"\\[.+?\\][\\(\\[].*?[\\)\\]]",rB:!0,c:[{cN:"string",b:"\\[",e:"\\]",eB:!0,rE:!0,r:0},{cN:"link",b:"\\]\\(",e:"\\)",eB:!0,eE:!0},{cN:"symbol",b:"\\]\\[",e:"\\]",eB:!0,eE:!0}],r:10},{b:/^\[[^\n]+\]:/,rB:!0,c:[{cN:"symbol",b:/\[/,e:/\]/,eB:!0,eE:!0},{cN:"link",b:/:\s*/,e:/$/,eB:!0}]}]}});hljs.registerLanguage("php",function(e){var c={b:"\\$+[a-zA-Z_-ÿ][a-zA-Z0-9_-ÿ]*"},i={cN:"meta",b:/<\?(php)?|\?>/},t={cN:"string",c:[e.BE,i],v:[{b:'b"',e:'"'},{b:"b'",e:"'"},e.inherit(e.ASM,{i:null}),e.inherit(e.QSM,{i:null})]},a={v:[e.BNM,e.CNM]};return{aliases:["php3","php4","php5","php6"],cI:!0,k:"and include_once list abstract global private echo interface as static endswitch array null if endwhile or const for endforeach self var while isset public protected exit foreach throw elseif include __FILE__ empty require_once do xor return parent clone use __CLASS__ __LINE__ else break print eval new catch __METHOD__ case exception default die require __FUNCTION__ enddeclare final try switch continue endfor endif declare unset true false trait goto instanceof insteadof __DIR__ __NAMESPACE__ yield finally",c:[e.HCM,e.C("//","$",{c:[i]}),e.C("/\\*","\\*/",{c:[{cN:"doctag",b:"@[A-Za-z]+"}]}),e.C("__halt_compiler.+?;",!1,{eW:!0,k:"__halt_compiler",l:e.UIR}),{cN:"string",b:/<<<['"]?\w+['"]?$/,e:/^\w+;?$/,c:[e.BE,{cN:"subst",v:[{b:/\$\w+/},{b:/\{\$/,e:/\}/}]}]},i,{cN:"keyword",b:/\$this\b/},c,{b:/(::|->)+[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*/},{cN:"function",bK:"function",e:/[;{]/,eE:!0,i:"\\$|\\[|%",c:[e.UTM,{cN:"params",b:"\\(",e:"\\)",c:["self",c,e.CBCM,t,a]}]},{cN:"class",bK:"class interface",e:"{",eE:!0,i:/[:\(\$"]/,c:[{bK:"extends implements"},e.UTM]},{bK:"namespace",e:";",i:/[\.']/,c:[e.UTM]},{bK:"use",e:";",c:[e.UTM]},{b:"=>"},t,a]}});hljs.registerLanguage("d",function(e){var t={keyword:"abstract alias align asm assert auto body break byte case cast catch class const continue debug default delete deprecated do else enum export extern final finally for foreach foreach_reverse|10 goto if immutable import in inout int interface invariant is lazy macro mixin module new nothrow out override package pragma private protected public pure ref return scope shared static struct super switch synchronized template this throw try typedef typeid typeof union unittest version void volatile while with __FILE__ __LINE__ __gshared|10 __thread __traits __DATE__ __EOF__ __TIME__ __TIMESTAMP__ __VENDOR__ __VERSION__",built_in:"bool cdouble cent cfloat char creal dchar delegate double dstring float function idouble ifloat ireal long real short string ubyte ucent uint ulong ushort wchar wstring",literal:"false null true"},r="(0|[1-9][\\d_]*)",a="(0|[1-9][\\d_]*|\\d[\\d_]*|[\\d_]+?\\d)",i="0[bB][01_]+",n="([\\da-fA-F][\\da-fA-F_]*|_[\\da-fA-F][\\da-fA-F_]*)",_="0[xX]"+n,c="([eE][+-]?"+a+")",d="("+a+"(\\.\\d*|"+c+")|\\d+\\."+a+a+"|\\."+r+c+"?)",o="(0[xX]("+n+"\\."+n+"|\\.?"+n+")[pP][+-]?"+a+")",s="("+r+"|"+i+"|"+_+")",l="("+o+"|"+d+")",u="\\\\(['\"\\?\\\\abfnrtv]|u[\\dA-Fa-f]{4}|[0-7]{1,3}|x[\\dA-Fa-f]{2}|U[\\dA-Fa-f]{8})|&[a-zA-Z\\d]{2,};",b={cN:"number",b:"\\b"+s+"(L|u|U|Lu|LU|uL|UL)?",r:0},f={cN:"number",b:"\\b("+l+"([fF]|L|i|[fF]i|Li)?|"+s+"(i|[fF]i|Li))",r:0},g={cN:"string",b:"'("+u+"|.)",e:"'",i:"."},h={b:u,r:0},p={cN:"string",b:'"',c:[h],e:'"[cwd]?'},m={cN:"string",b:'[rq]"',e:'"[cwd]?',r:5},w={cN:"string",b:"`",e:"`[cwd]?"},N={cN:"string",b:'x"[\\da-fA-F\\s\\n\\r]*"[cwd]?',r:10},A={cN:"string",b:'q"\\{',e:'\\}"'},F={cN:"meta",b:"^#!",e:"$",r:5},y={cN:"meta",b:"#(line)",e:"$",r:5},L={cN:"keyword",b:"@[a-zA-Z_][a-zA-Z_\\d]*"},v=e.C("\\/\\+","\\+\\/",{c:["self"],r:10});return{l:e.UIR,k:t,c:[e.CLCM,e.CBCM,v,N,p,m,w,A,f,b,g,F,y,L]}});hljs.registerLanguage("json",function(e){var i={literal:"true false null"},n=[e.QSM,e.CNM],r={e:",",eW:!0,eE:!0,c:n,k:i},t={b:"{",e:"}",c:[{cN:"attr",b:/"/,e:/"/,c:[e.BE],i:"\\n"},e.inherit(r,{b:/:/})],i:"\\S"},c={b:"\\[",e:"\\]",c:[e.inherit(r)],i:"\\S"};return n.splice(n.length,0,t,c),{c:n,k:i,i:"\\S"}});hljs.registerLanguage("go",function(e){var t={keyword:"break default func interface select case map struct chan else goto package switch const fallthrough if range type continue for import return var go defer bool byte complex64 complex128 float32 float64 int8 int16 int32 int64 string uint8 uint16 uint32 uint64 int uint uintptr rune",literal:"true false iota nil",built_in:"append cap close complex copy imag len make new panic print println real recover delete"};return{aliases:["golang"],k:t,i:"{",e:"}"},n={v:[{b:/\$\d/},{b:/[\$%@](\^\w\b|#\w+(::\w+)*|{\w+}|\w+(::\w*)*)/},{b:/[\$%@][^\s\w{]/,r:0}]},i=[e.BE,r,n],o=[n,e.HCM,e.C("^\\=\\w","\\=cut",{eW:!0}),s,{cN:"string",c:i,v:[{b:"q[qwxr]?\\s*\\(",e:"\\)",r:5},{b:"q[qwxr]?\\s*\\[",e:"\\]",r:5},{b:"q[qwxr]?\\s*\\{",e:"\\}",r:5},{b:"q[qwxr]?\\s*\\|",e:"\\|",r:5},{b:"q[qwxr]?\\s*\\<",e:"\\>",r:5},{b:"qw\\s+q",e:"q",r:5},{b:"'",e:"'",c:[e.BE]},{b:'"',e:'"'},{b:"`",e:"`",c:[e.BE]},{b:"{\\w+}",c:[],r:0},{b:"-?\\w+\\s*\\=\\>",c:[],r:0}]},{cN:"number",b:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b",r:0},{b:"(\\/\\/|"+e.RSR+"|\\b(split|return|print|reverse|grep)\\b)\\s*",k:"split return print reverse grep",r:0,c:[e.HCM,{cN:"regexp",b:"(s|tr|y)/(\\\\.|[^/])*/(\\\\.|[^/])*/[a-z]*",r:10},{cN:"regexp",b:"(m|qr)?/",e:"/[a-z]*",c:[e.BE],r:0}]},{cN:"function",bK:"sub",e:"(\\s*\\(.*?\\))?[;{]",eE:!0,r:5,c:[e.TM]},{b:"-\\w\\b",r:0},{b:"^__DATA__$",e:"^__END__$",sL:"mojolicious",c:[{b:"^@@.*",e:"$",cN:"comment"}]}];return r.c=o,s.c=o,{aliases:["pl","pm"],l:/[\w\.]+/,k:t,c:o}});hljs.registerLanguage("rust",function(e){var t="([ui](8|16|32|64|128|size)|f(32|64))?",r="alignof as be box break const continue crate do else enum extern false fn for if impl in let loop match mod mut offsetof once priv proc pub pure ref return self Self sizeof static struct super trait true type typeof unsafe unsized use virtual while where yield move default",n="drop i8 i16 i32 i64 i128 isize u8 u16 u32 u64 u128 usize f32 f64 str char bool Box Option Result String Vec Copy Send Sized Sync Drop Fn FnMut FnOnce ToOwned Clone Debug PartialEq PartialOrd Eq Ord AsRef AsMut Into From Default Iterator Extend IntoIterator DoubleEndedIterator ExactSizeIterator SliceConcatExt ToString assert! assert_eq! bitflags! bytes! cfg! col! concat! concat_idents! debug_assert! debug_assert_eq! env! panic! file! format! format_args! include_bin! include_str! line! local_data_key! module_path! option_env! print! println! select! stringify! try! unimplemented! unreachable! vec! write! writeln! macro_rules! assert_ne! debug_assert_ne!";return{aliases:["rs"],k:{keyword:r,literal:"true false Some None Ok Err",built_in:n},l:e.IR+"!?",i:""}]}});hljs.registerLanguage("ruby",function(e){var b="[a-zA-Z_]\\w*[!?=]?|[-+~]\\@|<<|>>|=~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~`|]|\\[\\]=?",r={keyword:"and then defined module in return redo if BEGIN retry end for self when next until do begin unless END rescue else break undef not super class case require yield alias while ensure elsif or include attr_reader attr_writer attr_accessor",literal:"true false nil"},c={cN:"doctag",b:"@[A-Za-z]+"},a={b:"#<",e:">"},s=[e.C("#","$",{c:[c]}),e.C("^\\=begin","^\\=end",{c:[c],r:10}),e.C("^__END__","\\n$")],n={cN:"subst",b:"#\\{",e:"}",k:r},t={cN:"string",c:[e.BE,n],v:[{b:/'/,e:/'/},{b:/"/,e:/"/},{b:/`/,e:/`/},{b:"%[qQwWx]?\\(",e:"\\)"},{b:"%[qQwWx]?\\[",e:"\\]"},{b:"%[qQwWx]?{",e:"}"},{b:"%[qQwWx]?<",e:">"},{b:"%[qQwWx]?/",e:"/"},{b:"%[qQwWx]?%",e:"%"},{b:"%[qQwWx]?-",e:"-"},{b:"%[qQwWx]?\\|",e:"\\|"},{b:/\B\?(\\\d{1,3}|\\x[A-Fa-f0-9]{1,2}|\\u[A-Fa-f0-9]{4}|\\?\S)\b/},{b:/<<(-?)\w+$/,e:/^\s*\w+$/}]},i={cN:"params",b:"\\(",e:"\\)",endsParent:!0,k:r},d=[t,a,{cN:"class",bK:"class module",e:"$|;",i:/=/,c:[e.inherit(e.TM,{b:"[A-Za-z_]\\w*(::\\w+)*(\\?|\\!)?"}),{b:"<\\s*",c:[{b:"("+e.IR+"::)?"+e.IR}]}].concat(s)},{cN:"function",bK:"def",e:"$|;",c:[e.inherit(e.TM,{b:b}),i].concat(s)},{b:e.IR+"::"},{cN:"symbol",b:e.UIR+"(\\!|\\?)?:",r:0},{cN:"symbol",b:":(?!\\s)",c:[t,{b:b}],r:0},{cN:"number",b:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b",r:0},{b:"(\\$\\W)|((\\$|\\@\\@?)(\\w+))"},{cN:"params",b:/\|/,e:/\|/,k:r},{b:"("+e.RSR+"|unless)\\s*",k:"unless",c:[a,{cN:"regexp",c:[e.BE,n],i:/\n/,v:[{b:"/",e:"/[a-z]*"},{b:"%r{",e:"}[a-z]*"},{b:"%r\\(",e:"\\)[a-z]*"},{b:"%r!",e:"![a-z]*"},{b:"%r\\[",e:"\\][a-z]*"}]}].concat(s),r:0}].concat(s);n.c=d,i.c=d;var l="[>?]>",o="[\\w#]+\\(\\w+\\):\\d+:\\d+>",u="(\\w+-)?\\d+\\.\\d+\\.\\d(p\\d+)?[^>]+>",w=[{b:/^\s*=>/,starts:{e:"$",c:d}},{cN:"meta",b:"^("+l+"|"+o+"|"+u+")",starts:{e:"$",c:d}}];return{aliases:["rb","gemspec","podspec","thor","irb"],k:r,i:/\/\*/,c:s.concat(w).concat(d)}});hljs.registerLanguage("makefile",function(e){var i={cN:"variable",v:[{b:"\\$\\("+e.UIR+"\\)",c:[e.BE]},{b:/\$[@%] *$",rE:!0,c:l.c,e:t.v[0].b},{b:"<%[%=-]?",e:"[%-]?%>",sL:"ruby",eB:!0,eE:!0,r:0},{cN:"type",b:"!!"+e.UIR},{cN:"meta",b:"&"+e.UIR+"$"},{cN:"meta",b:"\\*"+e.UIR+"$"},{cN:"bullet",b:"^ *-",r:0},e.HCM,{bK:b,k:{literal:b}},e.CNM,l]}});hljs.registerLanguage("css",function(e){var c="[a-zA-Z-][a-zA-Z0-9_-]*",t={b:/[A-Z\_\.\-]+\s*:/,rB:!0,e:";",eW:!0,c:[{cN:"attribute",b:/\S/,e:":",eE:!0,starts:{eW:!0,eE:!0,c:[{b:/[\w-]+\(/,rB:!0,c:[{cN:"built_in",b:/[\w-]+/},{b:/\(/,e:/\)/,c:[e.ASM,e.QSM]}]},e.CSSNM,e.QSM,e.ASM,e.CBCM,{cN:"number",b:"#[0-9A-Fa-f]+"},{cN:"meta",b:"!important"}]}}]};return{cI:!0,i:/[=\/|'\$]/,c:[e.CBCM,{cN:"selector-id",b:/#[A-Za-z0-9_-]+/},{cN:"selector-class",b:/\.[A-Za-z0-9_-]+/},{cN:"selector-attr",b:/\[/,e:/\]/,i:"$"},{cN:"selector-pseudo",b:/:(:)?[a-zA-Z0-9\_\-\+\(\)"'.]+/},{b:"@(font-face|page)",l:"[a-z-]+",k:"font-face page"},{b:"@",e:"[{;]",i:/:/,c:[{cN:"keyword",b:/\w+/},{b:/\s/,eW:!0,eE:!0,r:0,c:[e.ASM,e.QSM,e.CSSNM]}]},{cN:"selector-tag",b:c,r:0},{b:"{",e:"}",i:/\S/,c:[e.CBCM,t]}]}});hljs.registerLanguage("java",function(e){var a="[À-ʸa-zA-Z_$][À-ʸa-zA-Z_$0-9]*",t=a+"(<"+a+"(\\s*,\\s*"+a+")*>)?",r="false synchronized int abstract float private char boolean static null if const for true while long strictfp finally protected import native final void enum else break transient catch instanceof byte super volatile case assert short package default double public try this switch continue throws protected public private module requires exports do",s="\\b(0[bB]([01]+[01_]+[01]+|[01]+)|0[xX]([a-fA-F0-9]+[a-fA-F0-9_]+[a-fA-F0-9]+|[a-fA-F0-9]+)|(([\\d]+[\\d_]+[\\d]+|[\\d]+)(\\.([\\d]+[\\d_]+[\\d]+|[\\d]+))?|\\.([\\d]+[\\d_]+[\\d]+|[\\d]+))([eE][-+]?\\d+)?)[lLfF]?",c={cN:"number",b:s,r:0};return{aliases:["jsp"],k:r,i:/<\/|#/,c:[e.C("/\\*\\*","\\*/",{r:0,c:[{b:/\w+@/,r:0},{cN:"doctag",b:"@[A-Za-z]+"}]}),e.CLCM,e.CBCM,e.ASM,e.QSM,{cN:"class",bK:"class interface",e:/[{;=]/,eE:!0,k:"class interface",i:/[:"\[\]]/,c:[{bK:"extends implements"},e.UTM]},{bK:"new throw return else",r:0},{cN:"function",b:"("+t+"\\s+)+"+e.UIR+"\\s*\\(",rB:!0,e:/[{;=]/,eE:!0,k:r,c:[{b:e.UIR+"\\s*\\(",rB:!0,r:0,c:[e.UTM]},{cN:"params",b:/\(/,e:/\)/,k:r,r:0,c:[e.ASM,e.QSM,e.CNM,e.CBCM]},e.CLCM,e.CBCM]},c,{cN:"meta",b:"@[A-Za-z]+"}]}});hljs.registerLanguage("armasm",function(s){return{cI:!0,aliases:["arm"],l:"\\.?"+s.IR,k:{meta:".2byte .4byte .align .ascii .asciz .balign .byte .code .data .else .end .endif .endm .endr .equ .err .exitm .extern .global .hword .if .ifdef .ifndef .include .irp .long .macro .rept .req .section .set .skip .space .text .word .arm .thumb .code16 .code32 .force_thumb .thumb_func .ltorg ALIAS ALIGN ARM AREA ASSERT ATTR CN CODE CODE16 CODE32 COMMON CP DATA DCB DCD DCDU DCDO DCFD DCFDU DCI DCQ DCQU DCW DCWU DN ELIF ELSE END ENDFUNC ENDIF ENDP ENTRY EQU EXPORT EXPORTAS EXTERN FIELD FILL FUNCTION GBLA GBLL GBLS GET GLOBAL IF IMPORT INCBIN INCLUDE INFO KEEP LCLA LCLL LCLS LTORG MACRO MAP MEND MEXIT NOFP OPT PRESERVE8 PROC QN READONLY RELOC REQUIRE REQUIRE8 RLIST FN ROUT SETA SETL SETS SN SPACE SUBT THUMB THUMBX TTL WHILE WEND ",built_in:"r0 r1 r2 r3 r4 r5 r6 r7 r8 r9 r10 r11 r12 r13 r14 r15 pc lr sp ip sl sb fp a1 a2 a3 a4 v1 v2 v3 v4 v5 v6 v7 v8 f0 f1 f2 f3 f4 f5 f6 f7 p0 p1 p2 p3 p4 p5 p6 p7 p8 p9 p10 p11 p12 p13 p14 p15 c0 c1 c2 c3 c4 c5 c6 c7 c8 c9 c10 c11 c12 c13 c14 c15 q0 q1 q2 q3 q4 q5 q6 q7 q8 q9 q10 q11 q12 q13 q14 q15 cpsr_c cpsr_x cpsr_s cpsr_f cpsr_cx cpsr_cxs cpsr_xs cpsr_xsf cpsr_sf cpsr_cxsf spsr_c spsr_x spsr_s spsr_f spsr_cx spsr_cxs spsr_xs spsr_xsf spsr_sf spsr_cxsf s0 s1 s2 s3 s4 s5 s6 s7 s8 s9 s10 s11 s12 s13 s14 s15 s16 s17 s18 s19 s20 s21 s22 s23 s24 s25 s26 s27 s28 s29 s30 s31 d0 d1 d2 d3 d4 d5 d6 d7 d8 d9 d10 d11 d12 d13 d14 d15 d16 d17 d18 d19 d20 d21 d22 d23 d24 d25 d26 d27 d28 d29 d30 d31 {PC} {VAR} {TRUE} {FALSE} {OPT} {CONFIG} {ENDIAN} {CODESIZE} {CPU} {FPU} {ARCHITECTURE} {PCSTOREOFFSET} {ARMASM_VERSION} {INTER} {ROPI} {RWPI} {SWST} {NOSWST} . @"},c:[{cN:"keyword",b:"\\b(adc|(qd?|sh?|u[qh]?)?add(8|16)?|usada?8|(q|sh?|u[qh]?)?(as|sa)x|and|adrl?|sbc|rs[bc]|asr|b[lx]?|blx|bxj|cbn?z|tb[bh]|bic|bfc|bfi|[su]bfx|bkpt|cdp2?|clz|clrex|cmp|cmn|cpsi[ed]|cps|setend|dbg|dmb|dsb|eor|isb|it[te]{0,3}|lsl|lsr|ror|rrx|ldm(([id][ab])|f[ds])?|ldr((s|ex)?[bhd])?|movt?|mvn|mra|mar|mul|[us]mull|smul[bwt][bt]|smu[as]d|smmul|smmla|mla|umlaal|smlal?([wbt][bt]|d)|mls|smlsl?[ds]|smc|svc|sev|mia([bt]{2}|ph)?|mrr?c2?|mcrr2?|mrs|msr|orr|orn|pkh(tb|bt)|rbit|rev(16|sh)?|sel|[su]sat(16)?|nop|pop|push|rfe([id][ab])?|stm([id][ab])?|str(ex)?[bhd]?|(qd?)?sub|(sh?|q|u[qh]?)?sub(8|16)|[su]xt(a?h|a?b(16)?)|srs([id][ab])?|swpb?|swi|smi|tst|teq|wfe|wfi|yield)(eq|ne|cs|cc|mi|pl|vs|vc|hi|ls|ge|lt|gt|le|al|hs|lo)?[sptrx]?",e:"\\s"},s.C("[;@]","$",{r:0}),s.CBCM,s.QSM,{cN:"string",b:"'",e:"[^\\\\]'",r:0},{cN:"title",b:"\\|",e:"\\|",i:"\\n",r:0},{cN:"number",v:[{b:"[#$=]?0x[0-9a-f]+"},{b:"[#$=]?0b[01]+"},{b:"[#$=]\\d+"},{b:"\\b\\d+"}],r:0},{cN:"symbol",v:[{b:"^[a-z_\\.\\$][a-z0-9_\\.\\$]+"},{b:"^\\s*[a-z_\\.\\$][a-z0-9_\\.\\$]+:"},{b:"[=#]\\w+"}],r:0}]}});hljs.registerLanguage("swift",function(e){var i={keyword:"__COLUMN__ __FILE__ __FUNCTION__ __LINE__ as as! as? associativity break case catch class continue convenience default defer deinit didSet do dynamic dynamicType else enum extension fallthrough false fileprivate final for func get guard if import in indirect infix init inout internal is lazy left let mutating nil none nonmutating open operator optional override postfix precedence prefix private protocol Protocol public repeat required rethrows return right self Self set static struct subscript super switch throw throws true try try! try? Type typealias unowned var weak where while willSet",literal:"true false nil",built_in:"abs advance alignof alignofValue anyGenerator assert assertionFailure bridgeFromObjectiveC bridgeFromObjectiveCUnconditional bridgeToObjectiveC bridgeToObjectiveCUnconditional c contains count countElements countLeadingZeros debugPrint debugPrintln distance dropFirst dropLast dump encodeBitsAsWords enumerate equal fatalError filter find getBridgedObjectiveCType getVaList indices insertionSort isBridgedToObjectiveC isBridgedVerbatimToObjectiveC isUniquelyReferenced isUniquelyReferencedNonObjC join lazy lexicographicalCompare map max maxElement min minElement numericCast overlaps partition posix precondition preconditionFailure print println quickSort readLine reduce reflect reinterpretCast reverse roundUpToAlignment sizeof sizeofValue sort split startsWith stride strideof strideofValue swap toString transcode underestimateCount unsafeAddressOf unsafeBitCast unsafeDowncast unsafeUnwrap unsafeReflect withExtendedLifetime withObjectAtPlusZero withUnsafePointer withUnsafePointerToObject withUnsafeMutablePointer withUnsafeMutablePointers withUnsafePointer withUnsafePointers withVaList zip"},t={cN:"type",b:"\\b[A-Z][\\wÀ-ʸ']*",r:0},n=e.C("/\\*","\\*/",{c:["self"]}),r={cN:"subst",b:/\\\(/,e:"\\)",k:i,c:[]},a={cN:"number",b:"\\b([\\d_]+(\\.[\\deE_]+)?|0x[a-fA-F0-9_]+(\\.[a-fA-F0-9p_]+)?|0b[01_]+|0o[0-7_]+)\\b",r:0},o=e.inherit(e.QSM,{c:[r,e.BE]});return r.c=[a],{k:i,c:[o,e.CLCM,n,t,a,{cN:"function",bK:"func",e:"{",eE:!0,c:[e.inherit(e.TM,{b:/[A-Za-z$_][0-9A-Za-z$_]*/}),{b://},{cN:"params",b:/\(/,e:/\)/,endsParent:!0,k:i,c:["self",a,o,e.CBCM,{b:":"}],i:/["']/}],i:/\[|%/},{cN:"class",bK:"struct protocol class extension enum",k:i,e:"\\{",eE:!0,c:[e.inherit(e.TM,{b:/[A-Za-z$_][\u00C0-\u02B80-9A-Za-z$_]*/})]},{cN:"meta",b:"(@warn_unused_result|@exported|@lazy|@noescape|@NSCopying|@NSManaged|@objc|@convention|@required|@noreturn|@IBAction|@IBDesignable|@IBInspectable|@IBOutlet|@infix|@prefix|@postfix|@autoclosure|@testable|@available|@nonobjc|@NSApplicationMain|@UIApplicationMain)"},{bK:"import",e:/$/,c:[e.CLCM,n]}]}});hljs.registerLanguage("cpp",function(t){var e={cN:"keyword",b:"\\b[a-z\\d_]*_t\\b"},r={cN:"string",v:[{b:'(u8?|U)?L?"',e:'"',i:"\\n",c:[t.BE]},{b:'(u8?|U)?R"',e:'"',c:[t.BE]},{b:"'\\\\?.",e:"'",i:"."}]},s={cN:"number",v:[{b:"\\b(0b[01']+)"},{b:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)"},{b:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)"}],r:0},i={cN:"meta",b:/#\s*[a-z]+\b/,e:/$/,k:{"meta-keyword":"if else elif endif define undef warning error line pragma ifdef ifndef include"},c:[{b:/\\\n/,r:0},t.inherit(r,{cN:"meta-string"}),{cN:"meta-string",b:/<[^\n>]*>/,e:/$/,i:"\\n"},t.CLCM,t.CBCM]},a=t.IR+"\\s*\\(",c={keyword:"int float while private char catch import module export virtual operator sizeof dynamic_cast|10 typedef const_cast|10 const for static_cast|10 union namespace unsigned long volatile static protected bool template mutable if public friend do goto auto void enum else break extern using asm case typeid short reinterpret_cast|10 default double register explicit signed typename try this switch continue inline delete alignof constexpr decltype noexcept static_assert thread_local restrict _Bool complex _Complex _Imaginary atomic_bool atomic_char atomic_schar atomic_uchar atomic_short atomic_ushort atomic_int atomic_uint atomic_long atomic_ulong atomic_llong atomic_ullong new throw return and or not",built_in:"std string cin cout cerr clog stdin stdout stderr stringstream istringstream ostringstream auto_ptr deque list queue stack vector map set bitset multiset multimap unordered_set unordered_map unordered_multiset unordered_multimap array shared_ptr abort abs acos asin atan2 atan calloc ceil cosh cos exit exp fabs floor fmod fprintf fputs free frexp fscanf isalnum isalpha iscntrl isdigit isgraph islower isprint ispunct isspace isupper isxdigit tolower toupper labs ldexp log10 log malloc realloc memchr memcmp memcpy memset modf pow printf putchar puts scanf sinh sin snprintf sprintf sqrt sscanf strcat strchr strcmp strcpy strcspn strlen strncat strncmp strncpy strpbrk strrchr strspn strstr tanh tan vfprintf vprintf vsprintf endl initializer_list unique_ptr",literal:"true false nullptr NULL"},n=[e,t.CLCM,t.CBCM,s,r];return{aliases:["c","cc","h","c++","h++","hpp"],k:c,i:"",k:c,c:["self",e]},{b:t.IR+"::",k:c},{v:[{b:/=/,e:/;/},{b:/\(/,e:/\)/},{bK:"new throw return else",e:/;/}],k:c,c:n.concat([{b:/\(/,e:/\)/,k:c,c:n.concat(["self"]),r:0}]),r:0},{cN:"function",b:"("+t.IR+"[\\*&\\s]+)+"+a,rB:!0,e:/[{;=]/,eE:!0,k:c,i:/[^\w\s\*&]/,c:[{b:a,rB:!0,c:[t.TM],r:0},{cN:"params",b:/\(/,e:/\)/,k:c,r:0,c:[t.CLCM,t.CBCM,r,s,e]},t.CLCM,t.CBCM,i]},{cN:"class",bK:"class struct",e:/[{;:]/,c:[{b://,c:["self"]},t.TM]}]),exports:{preprocessor:i,strings:r,k:c}}});hljs.registerLanguage("x86asm",function(s){return{cI:!0,l:"[.%]?"+s.IR,k:{keyword:"lock rep repe repz repne repnz xaquire xrelease bnd nobnd aaa aad aam aas adc add and arpl bb0_reset bb1_reset bound bsf bsr bswap bt btc btr bts call cbw cdq cdqe clc cld cli clts cmc cmp cmpsb cmpsd cmpsq cmpsw cmpxchg cmpxchg486 cmpxchg8b cmpxchg16b cpuid cpu_read cpu_write cqo cwd cwde daa das dec div dmint emms enter equ f2xm1 fabs fadd faddp fbld fbstp fchs fclex fcmovb fcmovbe fcmove fcmovnb fcmovnbe fcmovne fcmovnu fcmovu fcom fcomi fcomip fcomp fcompp fcos fdecstp fdisi fdiv fdivp fdivr fdivrp femms feni ffree ffreep fiadd ficom ficomp fidiv fidivr fild fimul fincstp finit fist fistp fisttp fisub fisubr fld fld1 fldcw fldenv fldl2e fldl2t fldlg2 fldln2 fldpi fldz fmul fmulp fnclex fndisi fneni fninit fnop fnsave fnstcw fnstenv fnstsw fpatan fprem fprem1 fptan frndint frstor fsave fscale fsetpm fsin fsincos fsqrt fst fstcw fstenv fstp fstsw fsub fsubp fsubr fsubrp ftst fucom fucomi fucomip fucomp fucompp fxam fxch fxtract fyl2x fyl2xp1 hlt ibts icebp idiv imul in inc incbin insb insd insw int int01 int1 int03 int3 into invd invpcid invlpg invlpga iret iretd iretq iretw jcxz jecxz jrcxz jmp jmpe lahf lar lds lea leave les lfence lfs lgdt lgs lidt lldt lmsw loadall loadall286 lodsb lodsd lodsq lodsw loop loope loopne loopnz loopz lsl lss ltr mfence monitor mov movd movq movsb movsd movsq movsw movsx movsxd movzx mul mwait neg nop not or out outsb outsd outsw packssdw packsswb packuswb paddb paddd paddsb paddsiw paddsw paddusb paddusw paddw pand pandn pause paveb pavgusb pcmpeqb pcmpeqd pcmpeqw pcmpgtb pcmpgtd pcmpgtw pdistib pf2id pfacc pfadd pfcmpeq pfcmpge pfcmpgt pfmax pfmin pfmul pfrcp pfrcpit1 pfrcpit2 pfrsqit1 pfrsqrt pfsub pfsubr pi2fd pmachriw pmaddwd pmagw pmulhriw pmulhrwa pmulhrwc pmulhw pmullw pmvgezb pmvlzb pmvnzb pmvzb pop popa popad popaw popf popfd popfq popfw por prefetch prefetchw pslld psllq psllw psrad psraw psrld psrlq psrlw psubb psubd psubsb psubsiw psubsw psubusb psubusw psubw punpckhbw punpckhdq punpckhwd punpcklbw punpckldq punpcklwd push pusha pushad pushaw pushf pushfd pushfq pushfw pxor rcl rcr rdshr rdmsr rdpmc rdtsc rdtscp ret retf retn rol ror rdm rsdc rsldt rsm rsts sahf sal salc sar sbb scasb scasd scasq scasw sfence sgdt shl shld shr shrd sidt sldt skinit smi smint smintold smsw stc std sti stosb stosd stosq stosw str sub svdc svldt svts swapgs syscall sysenter sysexit sysret test ud0 ud1 ud2b ud2 ud2a umov verr verw fwait wbinvd wrshr wrmsr xadd xbts xchg xlatb xlat xor cmove cmovz cmovne cmovnz cmova cmovnbe cmovae cmovnb cmovb cmovnae cmovbe cmovna cmovg cmovnle cmovge cmovnl cmovl cmovnge cmovle cmovng cmovc cmovnc cmovo cmovno cmovs cmovns cmovp cmovpe cmovnp cmovpo je jz jne jnz ja jnbe jae jnb jb jnae jbe jna jg jnle jge jnl jl jnge jle jng jc jnc jo jno js jns jpo jnp jpe jp sete setz setne setnz seta setnbe setae setnb setnc setb setnae setcset setbe setna setg setnle setge setnl setl setnge setle setng sets setns seto setno setpe setp setpo setnp addps addss andnps andps cmpeqps cmpeqss cmpleps cmpless cmpltps cmpltss cmpneqps cmpneqss cmpnleps cmpnless cmpnltps cmpnltss cmpordps cmpordss cmpunordps cmpunordss cmpps cmpss comiss cvtpi2ps cvtps2pi cvtsi2ss cvtss2si cvttps2pi cvttss2si divps divss ldmxcsr maxps maxss minps minss movaps movhps movlhps movlps movhlps movmskps movntps movss movups mulps mulss orps rcpps rcpss rsqrtps rsqrtss shufps sqrtps sqrtss stmxcsr subps subss ucomiss unpckhps unpcklps xorps fxrstor fxrstor64 fxsave fxsave64 xgetbv xsetbv xsave xsave64 xsaveopt xsaveopt64 xrstor xrstor64 prefetchnta prefetcht0 prefetcht1 prefetcht2 maskmovq movntq pavgb pavgw pextrw pinsrw pmaxsw pmaxub pminsw pminub pmovmskb pmulhuw psadbw pshufw pf2iw pfnacc pfpnacc pi2fw pswapd maskmovdqu clflush movntdq movnti movntpd movdqa movdqu movdq2q movq2dq paddq pmuludq pshufd pshufhw pshuflw pslldq psrldq psubq punpckhqdq punpcklqdq addpd addsd andnpd andpd cmpeqpd cmpeqsd cmplepd cmplesd cmpltpd cmpltsd cmpneqpd cmpneqsd cmpnlepd cmpnlesd cmpnltpd cmpnltsd cmpordpd cmpordsd cmpunordpd cmpunordsd cmppd comisd cvtdq2pd cvtdq2ps cvtpd2dq cvtpd2pi cvtpd2ps cvtpi2pd cvtps2dq cvtps2pd cvtsd2si cvtsd2ss cvtsi2sd cvtss2sd cvttpd2pi cvttpd2dq cvttps2dq cvttsd2si divpd divsd maxpd maxsd minpd minsd movapd movhpd movlpd movmskpd movupd mulpd mulsd orpd shufpd sqrtpd sqrtsd subpd subsd ucomisd unpckhpd unpcklpd xorpd addsubpd addsubps haddpd haddps hsubpd hsubps lddqu movddup movshdup movsldup clgi stgi vmcall vmclear vmfunc vmlaunch vmload vmmcall vmptrld vmptrst vmread vmresume vmrun vmsave vmwrite vmxoff vmxon invept invvpid pabsb pabsw pabsd palignr phaddw phaddd phaddsw phsubw phsubd phsubsw pmaddubsw pmulhrsw pshufb psignb psignw psignd extrq insertq movntsd movntss lzcnt blendpd blendps blendvpd blendvps dppd dpps extractps insertps movntdqa mpsadbw packusdw pblendvb pblendw pcmpeqq pextrb pextrd pextrq phminposuw pinsrb pinsrd pinsrq pmaxsb pmaxsd pmaxud pmaxuw pminsb pminsd pminud pminuw pmovsxbw pmovsxbd pmovsxbq pmovsxwd pmovsxwq pmovsxdq pmovzxbw pmovzxbd pmovzxbq pmovzxwd pmovzxwq pmovzxdq pmuldq pmulld ptest roundpd roundps roundsd roundss crc32 pcmpestri pcmpestrm pcmpistri pcmpistrm pcmpgtq popcnt getsec pfrcpv pfrsqrtv movbe aesenc aesenclast aesdec aesdeclast aesimc aeskeygenassist vaesenc vaesenclast vaesdec vaesdeclast vaesimc vaeskeygenassist vaddpd vaddps vaddsd vaddss vaddsubpd vaddsubps vandpd vandps vandnpd vandnps vblendpd vblendps vblendvpd vblendvps vbroadcastss vbroadcastsd vbroadcastf128 vcmpeq_ospd vcmpeqpd vcmplt_ospd vcmpltpd vcmple_ospd vcmplepd vcmpunord_qpd vcmpunordpd vcmpneq_uqpd vcmpneqpd vcmpnlt_uspd vcmpnltpd vcmpnle_uspd vcmpnlepd vcmpord_qpd vcmpordpd vcmpeq_uqpd vcmpnge_uspd vcmpngepd vcmpngt_uspd vcmpngtpd vcmpfalse_oqpd vcmpfalsepd vcmpneq_oqpd vcmpge_ospd vcmpgepd vcmpgt_ospd vcmpgtpd vcmptrue_uqpd vcmptruepd vcmplt_oqpd vcmple_oqpd vcmpunord_spd vcmpneq_uspd vcmpnlt_uqpd vcmpnle_uqpd vcmpord_spd vcmpeq_uspd vcmpnge_uqpd vcmpngt_uqpd vcmpfalse_ospd vcmpneq_ospd vcmpge_oqpd vcmpgt_oqpd vcmptrue_uspd vcmppd vcmpeq_osps vcmpeqps vcmplt_osps vcmpltps vcmple_osps vcmpleps vcmpunord_qps vcmpunordps vcmpneq_uqps vcmpneqps vcmpnlt_usps vcmpnltps vcmpnle_usps vcmpnleps vcmpord_qps vcmpordps vcmpeq_uqps vcmpnge_usps vcmpngeps vcmpngt_usps vcmpngtps vcmpfalse_oqps vcmpfalseps vcmpneq_oqps vcmpge_osps vcmpgeps vcmpgt_osps vcmpgtps vcmptrue_uqps vcmptrueps vcmplt_oqps vcmple_oqps vcmpunord_sps vcmpneq_usps vcmpnlt_uqps vcmpnle_uqps vcmpord_sps vcmpeq_usps vcmpnge_uqps vcmpngt_uqps vcmpfalse_osps vcmpneq_osps vcmpge_oqps vcmpgt_oqps vcmptrue_usps vcmpps vcmpeq_ossd vcmpeqsd vcmplt_ossd vcmpltsd vcmple_ossd vcmplesd vcmpunord_qsd vcmpunordsd vcmpneq_uqsd vcmpneqsd vcmpnlt_ussd vcmpnltsd vcmpnle_ussd vcmpnlesd vcmpord_qsd vcmpordsd vcmpeq_uqsd vcmpnge_ussd vcmpngesd vcmpngt_ussd vcmpngtsd vcmpfalse_oqsd vcmpfalsesd vcmpneq_oqsd vcmpge_ossd vcmpgesd vcmpgt_ossd vcmpgtsd vcmptrue_uqsd vcmptruesd vcmplt_oqsd vcmple_oqsd vcmpunord_ssd vcmpneq_ussd vcmpnlt_uqsd vcmpnle_uqsd vcmpord_ssd vcmpeq_ussd vcmpnge_uqsd vcmpngt_uqsd vcmpfalse_ossd vcmpneq_ossd vcmpge_oqsd vcmpgt_oqsd vcmptrue_ussd vcmpsd vcmpeq_osss vcmpeqss vcmplt_osss vcmpltss vcmple_osss vcmpless vcmpunord_qss vcmpunordss vcmpneq_uqss vcmpneqss vcmpnlt_usss vcmpnltss vcmpnle_usss vcmpnless vcmpord_qss vcmpordss vcmpeq_uqss vcmpnge_usss vcmpngess vcmpngt_usss vcmpngtss vcmpfalse_oqss vcmpfalsess vcmpneq_oqss vcmpge_osss vcmpgess vcmpgt_osss vcmpgtss vcmptrue_uqss vcmptruess vcmplt_oqss vcmple_oqss vcmpunord_sss vcmpneq_usss vcmpnlt_uqss vcmpnle_uqss vcmpord_sss vcmpeq_usss vcmpnge_uqss vcmpngt_uqss vcmpfalse_osss vcmpneq_osss vcmpge_oqss vcmpgt_oqss vcmptrue_usss vcmpss vcomisd vcomiss vcvtdq2pd vcvtdq2ps vcvtpd2dq vcvtpd2ps vcvtps2dq vcvtps2pd vcvtsd2si vcvtsd2ss vcvtsi2sd vcvtsi2ss vcvtss2sd vcvtss2si vcvttpd2dq vcvttps2dq vcvttsd2si vcvttss2si vdivpd vdivps vdivsd vdivss vdppd vdpps vextractf128 vextractps vhaddpd vhaddps vhsubpd vhsubps vinsertf128 vinsertps vlddqu vldqqu vldmxcsr vmaskmovdqu vmaskmovps vmaskmovpd vmaxpd vmaxps vmaxsd vmaxss vminpd vminps vminsd vminss vmovapd vmovaps vmovd vmovq vmovddup vmovdqa vmovqqa vmovdqu vmovqqu vmovhlps vmovhpd vmovhps vmovlhps vmovlpd vmovlps vmovmskpd vmovmskps vmovntdq vmovntqq vmovntdqa vmovntpd vmovntps vmovsd vmovshdup vmovsldup vmovss vmovupd vmovups vmpsadbw vmulpd vmulps vmulsd vmulss vorpd vorps vpabsb vpabsw vpabsd vpacksswb vpackssdw vpackuswb vpackusdw vpaddb vpaddw vpaddd vpaddq vpaddsb vpaddsw vpaddusb vpaddusw vpalignr vpand vpandn vpavgb vpavgw vpblendvb vpblendw vpcmpestri vpcmpestrm vpcmpistri vpcmpistrm vpcmpeqb vpcmpeqw vpcmpeqd vpcmpeqq vpcmpgtb vpcmpgtw vpcmpgtd vpcmpgtq vpermilpd vpermilps vperm2f128 vpextrb vpextrw vpextrd vpextrq vphaddw vphaddd vphaddsw vphminposuw vphsubw vphsubd vphsubsw vpinsrb vpinsrw vpinsrd vpinsrq vpmaddwd vpmaddubsw vpmaxsb vpmaxsw vpmaxsd vpmaxub vpmaxuw vpmaxud vpminsb vpminsw vpminsd vpminub vpminuw vpminud vpmovmskb vpmovsxbw vpmovsxbd vpmovsxbq vpmovsxwd vpmovsxwq vpmovsxdq vpmovzxbw vpmovzxbd vpmovzxbq vpmovzxwd vpmovzxwq vpmovzxdq vpmulhuw vpmulhrsw vpmulhw vpmullw vpmulld vpmuludq vpmuldq vpor vpsadbw vpshufb vpshufd vpshufhw vpshuflw vpsignb vpsignw vpsignd vpslldq vpsrldq vpsllw vpslld vpsllq vpsraw vpsrad vpsrlw vpsrld vpsrlq vptest vpsubb vpsubw vpsubd vpsubq vpsubsb vpsubsw vpsubusb vpsubusw vpunpckhbw vpunpckhwd vpunpckhdq vpunpckhqdq vpunpcklbw vpunpcklwd vpunpckldq vpunpcklqdq vpxor vrcpps vrcpss vrsqrtps vrsqrtss vroundpd vroundps vroundsd vroundss vshufpd vshufps vsqrtpd vsqrtps vsqrtsd vsqrtss vstmxcsr vsubpd vsubps vsubsd vsubss vtestps vtestpd vucomisd vucomiss vunpckhpd vunpckhps vunpcklpd vunpcklps vxorpd vxorps vzeroall vzeroupper pclmullqlqdq pclmulhqlqdq pclmullqhqdq pclmulhqhqdq pclmulqdq vpclmullqlqdq vpclmulhqlqdq vpclmullqhqdq vpclmulhqhqdq vpclmulqdq vfmadd132ps vfmadd132pd vfmadd312ps vfmadd312pd vfmadd213ps vfmadd213pd vfmadd123ps vfmadd123pd vfmadd231ps vfmadd231pd vfmadd321ps vfmadd321pd vfmaddsub132ps vfmaddsub132pd vfmaddsub312ps vfmaddsub312pd vfmaddsub213ps vfmaddsub213pd vfmaddsub123ps vfmaddsub123pd vfmaddsub231ps vfmaddsub231pd vfmaddsub321ps vfmaddsub321pd vfmsub132ps vfmsub132pd vfmsub312ps vfmsub312pd vfmsub213ps vfmsub213pd vfmsub123ps vfmsub123pd vfmsub231ps vfmsub231pd vfmsub321ps vfmsub321pd vfmsubadd132ps vfmsubadd132pd vfmsubadd312ps vfmsubadd312pd vfmsubadd213ps vfmsubadd213pd vfmsubadd123ps vfmsubadd123pd vfmsubadd231ps vfmsubadd231pd vfmsubadd321ps vfmsubadd321pd vfnmadd132ps vfnmadd132pd vfnmadd312ps vfnmadd312pd vfnmadd213ps vfnmadd213pd vfnmadd123ps vfnmadd123pd vfnmadd231ps vfnmadd231pd vfnmadd321ps vfnmadd321pd vfnmsub132ps vfnmsub132pd vfnmsub312ps vfnmsub312pd vfnmsub213ps vfnmsub213pd vfnmsub123ps vfnmsub123pd vfnmsub231ps vfnmsub231pd vfnmsub321ps vfnmsub321pd vfmadd132ss vfmadd132sd vfmadd312ss vfmadd312sd vfmadd213ss vfmadd213sd vfmadd123ss vfmadd123sd vfmadd231ss vfmadd231sd vfmadd321ss vfmadd321sd vfmsub132ss vfmsub132sd vfmsub312ss vfmsub312sd vfmsub213ss vfmsub213sd vfmsub123ss vfmsub123sd vfmsub231ss vfmsub231sd vfmsub321ss vfmsub321sd vfnmadd132ss vfnmadd132sd vfnmadd312ss vfnmadd312sd vfnmadd213ss vfnmadd213sd vfnmadd123ss vfnmadd123sd vfnmadd231ss vfnmadd231sd vfnmadd321ss vfnmadd321sd vfnmsub132ss vfnmsub132sd vfnmsub312ss vfnmsub312sd vfnmsub213ss vfnmsub213sd vfnmsub123ss vfnmsub123sd vfnmsub231ss vfnmsub231sd vfnmsub321ss vfnmsub321sd rdfsbase rdgsbase rdrand wrfsbase wrgsbase vcvtph2ps vcvtps2ph adcx adox rdseed clac stac xstore xcryptecb xcryptcbc xcryptctr xcryptcfb xcryptofb montmul xsha1 xsha256 llwpcb slwpcb lwpval lwpins vfmaddpd vfmaddps vfmaddsd vfmaddss vfmaddsubpd vfmaddsubps vfmsubaddpd vfmsubaddps vfmsubpd vfmsubps vfmsubsd vfmsubss vfnmaddpd vfnmaddps vfnmaddsd vfnmaddss vfnmsubpd vfnmsubps vfnmsubsd vfnmsubss vfrczpd vfrczps vfrczsd vfrczss vpcmov vpcomb vpcomd vpcomq vpcomub vpcomud vpcomuq vpcomuw vpcomw vphaddbd vphaddbq vphaddbw vphadddq vphaddubd vphaddubq vphaddubw vphaddudq vphadduwd vphadduwq vphaddwd vphaddwq vphsubbw vphsubdq vphsubwd vpmacsdd vpmacsdqh vpmacsdql vpmacssdd vpmacssdqh vpmacssdql vpmacsswd vpmacssww vpmacswd vpmacsww vpmadcsswd vpmadcswd vpperm vprotb vprotd vprotq vprotw vpshab vpshad vpshaq vpshaw vpshlb vpshld vpshlq vpshlw vbroadcasti128 vpblendd vpbroadcastb vpbroadcastw vpbroadcastd vpbroadcastq vpermd vpermpd vpermps vpermq vperm2i128 vextracti128 vinserti128 vpmaskmovd vpmaskmovq vpsllvd vpsllvq vpsravd vpsrlvd vpsrlvq vgatherdpd vgatherqpd vgatherdps vgatherqps vpgatherdd vpgatherqd vpgatherdq vpgatherqq xabort xbegin xend xtest andn bextr blci blcic blsi blsic blcfill blsfill blcmsk blsmsk blsr blcs bzhi mulx pdep pext rorx sarx shlx shrx tzcnt tzmsk t1mskc valignd valignq vblendmpd vblendmps vbroadcastf32x4 vbroadcastf64x4 vbroadcasti32x4 vbroadcasti64x4 vcompresspd vcompressps vcvtpd2udq vcvtps2udq vcvtsd2usi vcvtss2usi vcvttpd2udq vcvttps2udq vcvttsd2usi vcvttss2usi vcvtudq2pd vcvtudq2ps vcvtusi2sd vcvtusi2ss vexpandpd vexpandps vextractf32x4 vextractf64x4 vextracti32x4 vextracti64x4 vfixupimmpd vfixupimmps vfixupimmsd vfixupimmss vgetexppd vgetexpps vgetexpsd vgetexpss vgetmantpd vgetmantps vgetmantsd vgetmantss vinsertf32x4 vinsertf64x4 vinserti32x4 vinserti64x4 vmovdqa32 vmovdqa64 vmovdqu32 vmovdqu64 vpabsq vpandd vpandnd vpandnq vpandq vpblendmd vpblendmq vpcmpltd vpcmpled vpcmpneqd vpcmpnltd vpcmpnled vpcmpd vpcmpltq vpcmpleq vpcmpneqq vpcmpnltq vpcmpnleq vpcmpq vpcmpequd vpcmpltud vpcmpleud vpcmpnequd vpcmpnltud vpcmpnleud vpcmpud vpcmpequq vpcmpltuq vpcmpleuq vpcmpnequq vpcmpnltuq vpcmpnleuq vpcmpuq vpcompressd vpcompressq vpermi2d vpermi2pd vpermi2ps vpermi2q vpermt2d vpermt2pd vpermt2ps vpermt2q vpexpandd vpexpandq vpmaxsq vpmaxuq vpminsq vpminuq vpmovdb vpmovdw vpmovqb vpmovqd vpmovqw vpmovsdb vpmovsdw vpmovsqb vpmovsqd vpmovsqw vpmovusdb vpmovusdw vpmovusqb vpmovusqd vpmovusqw vpord vporq vprold vprolq vprolvd vprolvq vprord vprorq vprorvd vprorvq vpscatterdd vpscatterdq vpscatterqd vpscatterqq vpsraq vpsravq vpternlogd vpternlogq vptestmd vptestmq vptestnmd vptestnmq vpxord vpxorq vrcp14pd vrcp14ps vrcp14sd vrcp14ss vrndscalepd vrndscaleps vrndscalesd vrndscaless vrsqrt14pd vrsqrt14ps vrsqrt14sd vrsqrt14ss vscalefpd vscalefps vscalefsd vscalefss vscatterdpd vscatterdps vscatterqpd vscatterqps vshuff32x4 vshuff64x2 vshufi32x4 vshufi64x2 kandnw kandw kmovw knotw kortestw korw kshiftlw kshiftrw kunpckbw kxnorw kxorw vpbroadcastmb2q vpbroadcastmw2d vpconflictd vpconflictq vplzcntd vplzcntq vexp2pd vexp2ps vrcp28pd vrcp28ps vrcp28sd vrcp28ss vrsqrt28pd vrsqrt28ps vrsqrt28sd vrsqrt28ss vgatherpf0dpd vgatherpf0dps vgatherpf0qpd vgatherpf0qps vgatherpf1dpd vgatherpf1dps vgatherpf1qpd vgatherpf1qps vscatterpf0dpd vscatterpf0dps vscatterpf0qpd vscatterpf0qps vscatterpf1dpd vscatterpf1dps vscatterpf1qpd vscatterpf1qps prefetchwt1 bndmk bndcl bndcu bndcn bndmov bndldx bndstx sha1rnds4 sha1nexte sha1msg1 sha1msg2 sha256rnds2 sha256msg1 sha256msg2 hint_nop0 hint_nop1 hint_nop2 hint_nop3 hint_nop4 hint_nop5 hint_nop6 hint_nop7 hint_nop8 hint_nop9 hint_nop10 hint_nop11 hint_nop12 hint_nop13 hint_nop14 hint_nop15 hint_nop16 hint_nop17 hint_nop18 hint_nop19 hint_nop20 hint_nop21 hint_nop22 hint_nop23 hint_nop24 hint_nop25 hint_nop26 hint_nop27 hint_nop28 hint_nop29 hint_nop30 hint_nop31 hint_nop32 hint_nop33 hint_nop34 hint_nop35 hint_nop36 hint_nop37 hint_nop38 hint_nop39 hint_nop40 hint_nop41 hint_nop42 hint_nop43 hint_nop44 hint_nop45 hint_nop46 hint_nop47 hint_nop48 hint_nop49 hint_nop50 hint_nop51 hint_nop52 hint_nop53 hint_nop54 hint_nop55 hint_nop56 hint_nop57 hint_nop58 hint_nop59 hint_nop60 hint_nop61 hint_nop62 hint_nop63",built_in:"ip eip rip al ah bl bh cl ch dl dh sil dil bpl spl r8b r9b r10b r11b r12b r13b r14b r15b ax bx cx dx si di bp sp r8w r9w r10w r11w r12w r13w r14w r15w eax ebx ecx edx esi edi ebp esp eip r8d r9d r10d r11d r12d r13d r14d r15d rax rbx rcx rdx rsi rdi rbp rsp r8 r9 r10 r11 r12 r13 r14 r15 cs ds es fs gs ss st st0 st1 st2 st3 st4 st5 st6 st7 mm0 mm1 mm2 mm3 mm4 mm5 mm6 mm7 xmm0 xmm1 xmm2 xmm3 xmm4 xmm5 xmm6 xmm7 xmm8 xmm9 xmm10 xmm11 xmm12 xmm13 xmm14 xmm15 xmm16 xmm17 xmm18 xmm19 xmm20 xmm21 xmm22 xmm23 xmm24 xmm25 xmm26 xmm27 xmm28 xmm29 xmm30 xmm31 ymm0 ymm1 ymm2 ymm3 ymm4 ymm5 ymm6 ymm7 ymm8 ymm9 ymm10 ymm11 ymm12 ymm13 ymm14 ymm15 ymm16 ymm17 ymm18 ymm19 ymm20 ymm21 ymm22 ymm23 ymm24 ymm25 ymm26 ymm27 ymm28 ymm29 ymm30 ymm31 zmm0 zmm1 zmm2 zmm3 zmm4 zmm5 zmm6 zmm7 zmm8 zmm9 zmm10 zmm11 zmm12 zmm13 zmm14 zmm15 zmm16 zmm17 zmm18 zmm19 zmm20 zmm21 zmm22 zmm23 zmm24 zmm25 zmm26 zmm27 zmm28 zmm29 zmm30 zmm31 k0 k1 k2 k3 k4 k5 k6 k7 bnd0 bnd1 bnd2 bnd3 cr0 cr1 cr2 cr3 cr4 cr8 dr0 dr1 dr2 dr3 dr8 tr3 tr4 tr5 tr6 tr7 r0 r1 r2 r3 r4 r5 r6 r7 r0b r1b r2b r3b r4b r5b r6b r7b r0w r1w r2w r3w r4w r5w r6w r7w r0d r1d r2d r3d r4d r5d r6d r7d r0h r1h r2h r3h r0l r1l r2l r3l r4l r5l r6l r7l r8l r9l r10l r11l r12l r13l r14l r15l db dw dd dq dt ddq do dy dz resb resw resd resq rest resdq reso resy resz incbin equ times byte word dword qword nosplit rel abs seg wrt strict near far a32 ptr",meta:"%define %xdefine %+ %undef %defstr %deftok %assign %strcat %strlen %substr %rotate %elif %else %endif %if %ifmacro %ifctx %ifidn %ifidni %ifid %ifnum %ifstr %iftoken %ifempty %ifenv %error %warning %fatal %rep %endrep %include %push %pop %repl %pathsearch %depend %use %arg %stacksize %local %line %comment %endcomment .nolist __FILE__ __LINE__ __SECT__ __BITS__ __OUTPUT_FORMAT__ __DATE__ __TIME__ __DATE_NUM__ __TIME_NUM__ __UTC_DATE__ __UTC_TIME__ __UTC_DATE_NUM__ __UTC_TIME_NUM__ __PASS__ struc endstruc istruc at iend align alignb sectalign daz nodaz up down zero default option assume public bits use16 use32 use64 default section segment absolute extern global common cpu float __utf16__ __utf16le__ __utf16be__ __utf32__ __utf32le__ __utf32be__ __float8__ __float16__ __float32__ __float64__ __float80m__ __float80e__ __float128l__ __float128h__ __Infinity__ __QNaN__ __SNaN__ Inf NaN QNaN SNaN float8 float16 float32 float64 float80m float80e float128l float128h __FLOAT_DAZ__ __FLOAT_ROUND__ __FLOAT__"},c:[s.C(";","$",{r:0}),{cN:"number",v:[{b:"\\b(?:([0-9][0-9_]*)?\\.[0-9_]*(?:[eE][+-]?[0-9_]+)?|(0[Xx])?[0-9][0-9_]*\\.?[0-9_]*(?:[pP](?:[+-]?[0-9_]+)?)?)\\b",r:0},{b:"\\$[0-9][0-9A-Fa-f]*",r:0},{b:"\\b(?:[0-9A-Fa-f][0-9A-Fa-f_]*[Hh]|[0-9][0-9_]*[DdTt]?|[0-7][0-7_]*[QqOo]|[0-1][0-1_]*[BbYy])\\b"},{b:"\\b(?:0[Xx][0-9A-Fa-f_]+|0[DdTt][0-9_]+|0[QqOo][0-7_]+|0[BbYy][0-1_]+)\\b"}]},s.QSM,{cN:"string",v:[{b:"'",e:"[^\\\\]'"},{b:"`",e:"[^\\\\]`"}],r:0},{cN:"symbol",v:[{b:"^\\s*[A-Za-z._?][A-Za-z0-9_$#@~.?]*(:|\\s+label)"},{b:"^\\s*%%[A-Za-z0-9_$#@~.?]*:"}],r:0},{cN:"subst",b:"%[0-9]+",r:0},{cN:"subst",b:"%!S+",r:0},{cN:"meta",b:/^\s*\.[\w_-]+/}]}});hljs.registerLanguage("bash",function(e){var t={cN:"variable",v:[{b:/\$[\w\d#@][\w\d_]*/},{b:/\$\{(.*?)}/}]},s={cN:"string",b:/"/,e:/"/,c:[e.BE,t,{cN:"variable",b:/\$\(/,e:/\)/,c:[e.BE]}]},a={cN:"string",b:/'/,e:/'/};return{aliases:["sh","zsh"],l:/\b-?[a-z\._]+\b/,k:{keyword:"if then else elif fi for while in do done case esac function",literal:"true false",built_in:"break cd continue eval exec exit export getopts hash pwd readonly return shift test times trap umask unset alias bind builtin caller command declare echo enable help let local logout mapfile printf read readarray source type typeset ulimit unalias set shopt autoload bg bindkey bye cap chdir clone comparguments compcall compctl compdescribe compfiles compgroups compquote comptags comptry compvalues dirs disable disown echotc echoti emulate fc fg float functions getcap getln history integer jobs kill limit log noglob popd print pushd pushln rehash sched setcap setopt stat suspend ttyctl unfunction unhash unlimit unsetopt vared wait whence where which zcompile zformat zftp zle zmodload zparseopts zprof zpty zregexparse zsocket zstyle ztcp",_:"-ne -eq -lt -gt -f -d -e -s -l -a"},c:[{cN:"meta",b:/^#![^\n]+sh\s*$/,r:10},{cN:"function",b:/\w[\w\d_]*\s*\(\s*\)\s*\{/,rB:!0,c:[e.inherit(e.TM,{b:/\w[\w\d_]*/})],r:0},e.HCM,s,a,t]}});hljs.registerLanguage("shell",function(s){return{aliases:["console"],c:[{cN:"meta",b:"^\\s{0,3}[\\w\\d\\[\\]()@-]*[>%$#]",starts:{e:"$",sL:"bash"}}]}});hljs.registerLanguage("http",function(e){var t="HTTP/[0-9\\.]+";return{aliases:["https"],i:"\\S",c:[{b:"^"+t,e:"$",c:[{cN:"number",b:"\\b\\d{3}\\b"}]},{b:"^[A-Z]+ (.*?) "+t+"$",rB:!0,e:"$",c:[{cN:"string",b:" ",e:" ",eB:!0,eE:!0},{b:t},{cN:"keyword",b:"[A-Z]+"}]},{cN:"attribute",b:"^\\w",e:": ",eE:!0,i:"\\n|\\s|=",starts:{e:"$",r:0}},{b:"\\n\\n",starts:{sL:[],eW:!0}}]}});hljs.registerLanguage("cs",function(e){var i={keyword:"abstract as base bool break byte case catch char checked const continue decimal default delegate do double enum event explicit extern finally fixed float for foreach goto if implicit in int interface internal is lock long nameof object operator out override params private protected public readonly ref sbyte sealed short sizeof stackalloc static string struct switch this try typeof uint ulong unchecked unsafe ushort using virtual void volatile while add alias ascending async await by descending dynamic equals from get global group into join let on orderby partial remove select set value var where yield",literal:"null false true"},t={cN:"string",b:'@"',e:'"',c:[{b:'""'}]},r=e.inherit(t,{i:/\n/}),a={cN:"subst",b:"{",e:"}",k:i},c=e.inherit(a,{i:/\n/}),n={cN:"string",b:/\$"/,e:'"',i:/\n/,c:[{b:"{{"},{b:"}}"},e.BE,c]},s={cN:"string",b:/\$@"/,e:'"',c:[{b:"{{"},{b:"}}"},{b:'""'},a]},o=e.inherit(s,{i:/\n/,c:[{b:"{{"},{b:"}}"},{b:'""'},c]});a.c=[s,n,t,e.ASM,e.QSM,e.CNM,e.CBCM],c.c=[o,n,r,e.ASM,e.QSM,e.CNM,e.inherit(e.CBCM,{i:/\n/})];var l={v:[s,n,t,e.ASM,e.QSM]},b=e.IR+"(<"+e.IR+"(\\s*,\\s*"+e.IR+")*>)?(\\[\\])?";return{aliases:["csharp"],k:i,i:/::/,c:[e.C("///","$",{rB:!0,c:[{cN:"doctag",v:[{b:"///",r:0},{b:""},{b:""}]}]}),e.CLCM,e.CBCM,{cN:"meta",b:"#",e:"$",k:{"meta-keyword":"if else elif endif define undef warning error line region endregion pragma checksum"}},l,e.CNM,{bK:"class interface",e:/[{;=]/,i:/[^\s:]/,c:[e.TM,e.CLCM,e.CBCM]},{bK:"namespace",e:/[{;=]/,i:/[^\s:]/,c:[e.inherit(e.TM,{b:"[a-zA-Z](\\.?\\w)*"}),e.CLCM,e.CBCM]},{cN:"meta",b:"^\\s*\\[",eB:!0,e:"\\]",eE:!0,c:[{cN:"meta-string",b:/"/,e:/"/}]},{bK:"new return throw await else",r:0},{cN:"function",b:"("+b+"\\s+)+"+e.IR+"\\s*\\(",rB:!0,e:/[{;=]/,eE:!0,k:i,c:[{b:e.IR+"\\s*\\(",rB:!0,c:[e.TM],r:0},{cN:"params",b:/\(/,e:/\)/,eB:!0,eE:!0,k:i,r:0,c:[l,e.CNM,e.CBCM]},e.CLCM,e.CBCM]}]}});hljs.registerLanguage("coffeescript",function(e){var c={keyword:"in if for while finally new do return else break catch instanceof throw try this switch continue typeof delete debugger super yield import export from as default await then unless until loop of by when and or is isnt not",literal:"true false null undefined yes no on off",built_in:"npm require console print module global window document"},n="[A-Za-z$_][0-9A-Za-z$_]*",r={cN:"subst",b:/#\{/,e:/}/,k:c},i=[e.BNM,e.inherit(e.CNM,{starts:{e:"(\\s*/)?",r:0}}),{cN:"string",v:[{b:/'''/,e:/'''/,c:[e.BE]},{b:/'/,e:/'/,c:[e.BE]},{b:/"""/,e:/"""/,c:[e.BE,r]},{b:/"/,e:/"/,c:[e.BE,r]}]},{cN:"regexp",v:[{b:"///",e:"///",c:[r,e.HCM]},{b:"//[gim]*",r:0},{b:/\/(?![ *])(\\\/|.)*?\/[gim]*(?=\W|$)/}]},{b:"@"+n},{sL:"javascript",eB:!0,eE:!0,v:[{b:"```",e:"```"},{b:"`",e:"`"}]}];r.c=i;var s=e.inherit(e.TM,{b:n}),t="(\\(.*\\))?\\s*\\B[-=]>",o={cN:"params",b:"\\([^\\(]",rB:!0,c:[{b:/\(/,e:/\)/,k:c,c:["self"].concat(i)}]};return{aliases:["coffee","cson","iced"],k:c,i:/\/\*/,c:i.concat([e.C("###","###"),e.HCM,{cN:"function",b:"^\\s*"+n+"\\s*=\\s*"+t,e:"[-=]>",rB:!0,c:[s,o]},{b:/[:\(,=]\s*/,r:0,c:[{cN:"function",b:t,e:"[-=]>",rB:!0,c:[o]}]},{cN:"class",bK:"class",e:"$",i:/[:="\[\]]/,c:[{bK:"extends",eW:!0,i:/[:="\[\]]/,c:[s]},s]},{b:n+":",e:":",rB:!0,rE:!0,r:0}])}});hljs.registerLanguage("sql",function(e){var t=e.C("--","$");return{cI:!0,i:/[<>{}*#]/,c:[{bK:"begin end start commit rollback savepoint lock alter create drop rename call delete do handler insert load replace select truncate update set show pragma grant merge describe use explain help declare prepare execute deallocate release unlock purge reset change stop analyze cache flush optimize repair kill install uninstall checksum restore check backup revoke comment",e:/;/,eW:!0,l:/[\w\.]+/,k:{keyword:"abort abs absolute acc acce accep accept access accessed accessible account acos action activate add addtime admin administer advanced advise aes_decrypt aes_encrypt after agent aggregate ali alia alias allocate allow alter always analyze ancillary and any anydata anydataset anyschema anytype apply archive archived archivelog are as asc ascii asin assembly assertion associate asynchronous at atan atn2 attr attri attrib attribu attribut attribute attributes audit authenticated authentication authid authors auto autoallocate autodblink autoextend automatic availability avg backup badfile basicfile before begin beginning benchmark between bfile bfile_base big bigfile bin binary_double binary_float binlog bit_and bit_count bit_length bit_or bit_xor bitmap blob_base block blocksize body both bound buffer_cache buffer_pool build bulk by byte byteordermark bytes cache caching call calling cancel capacity cascade cascaded case cast catalog category ceil ceiling chain change changed char_base char_length character_length characters characterset charindex charset charsetform charsetid check checksum checksum_agg child choose chr chunk class cleanup clear client clob clob_base clone close cluster_id cluster_probability cluster_set clustering coalesce coercibility col collate collation collect colu colum column column_value columns columns_updated comment commit compact compatibility compiled complete composite_limit compound compress compute concat concat_ws concurrent confirm conn connec connect connect_by_iscycle connect_by_isleaf connect_by_root connect_time connection consider consistent constant constraint constraints constructor container content contents context contributors controlfile conv convert convert_tz corr corr_k corr_s corresponding corruption cos cost count count_big counted covar_pop covar_samp cpu_per_call cpu_per_session crc32 create creation critical cross cube cume_dist curdate current current_date current_time current_timestamp current_user cursor curtime customdatum cycle data database databases datafile datafiles datalength date_add date_cache date_format date_sub dateadd datediff datefromparts datename datepart datetime2fromparts day day_to_second dayname dayofmonth dayofweek dayofyear days db_role_change dbtimezone ddl deallocate declare decode decompose decrement decrypt deduplicate def defa defau defaul default defaults deferred defi defin define degrees delayed delegate delete delete_all delimited demand dense_rank depth dequeue des_decrypt des_encrypt des_key_file desc descr descri describ describe descriptor deterministic diagnostics difference dimension direct_load directory disable disable_all disallow disassociate discardfile disconnect diskgroup distinct distinctrow distribute distributed div do document domain dotnet double downgrade drop dumpfile duplicate duration each edition editionable editions element ellipsis else elsif elt empty enable enable_all enclosed encode encoding encrypt end end-exec endian enforced engine engines enqueue enterprise entityescaping eomonth error errors escaped evalname evaluate event eventdata events except exception exceptions exchange exclude excluding execu execut execute exempt exists exit exp expire explain export export_set extended extent external external_1 external_2 externally extract failed failed_login_attempts failover failure far fast feature_set feature_value fetch field fields file file_name_convert filesystem_like_logging final finish first first_value fixed flash_cache flashback floor flush following follows for forall force form forma format found found_rows freelist freelists freepools fresh from from_base64 from_days ftp full function general generated get get_format get_lock getdate getutcdate global global_name globally go goto grant grants greatest group group_concat group_id grouping grouping_id groups gtid_subtract guarantee guard handler hash hashkeys having hea head headi headin heading heap help hex hierarchy high high_priority hosts hour http id ident_current ident_incr ident_seed identified identity idle_time if ifnull ignore iif ilike ilm immediate import in include including increment index indexes indexing indextype indicator indices inet6_aton inet6_ntoa inet_aton inet_ntoa infile initial initialized initially initrans inmemory inner innodb input insert install instance instantiable instr interface interleaved intersect into invalidate invisible is is_free_lock is_ipv4 is_ipv4_compat is_not is_not_null is_used_lock isdate isnull isolation iterate java join json json_exists keep keep_duplicates key keys kill language large last last_day last_insert_id last_value lax lcase lead leading least leaves left len lenght length less level levels library like like2 like4 likec limit lines link list listagg little ln load load_file lob lobs local localtime localtimestamp locate locator lock locked log log10 log2 logfile logfiles logging logical logical_reads_per_call logoff logon logs long loop low low_priority lower lpad lrtrim ltrim main make_set makedate maketime managed management manual map mapping mask master master_pos_wait match matched materialized max maxextents maximize maxinstances maxlen maxlogfiles maxloghistory maxlogmembers maxsize maxtrans md5 measures median medium member memcompress memory merge microsecond mid migration min minextents minimum mining minus minute minvalue missing mod mode model modification modify module monitoring month months mount move movement multiset mutex name name_const names nan national native natural nav nchar nclob nested never new newline next nextval no no_write_to_binlog noarchivelog noaudit nobadfile nocheck nocompress nocopy nocycle nodelay nodiscardfile noentityescaping noguarantee nokeep nologfile nomapping nomaxvalue nominimize nominvalue nomonitoring none noneditionable nonschema noorder nopr nopro noprom nopromp noprompt norely noresetlogs noreverse normal norowdependencies noschemacheck noswitch not nothing notice notrim novalidate now nowait nth_value nullif nulls num numb numbe nvarchar nvarchar2 object ocicoll ocidate ocidatetime ociduration ociinterval ociloblocator ocinumber ociref ocirefcursor ocirowid ocistring ocitype oct octet_length of off offline offset oid oidindex old on online only opaque open operations operator optimal optimize option optionally or oracle oracle_date oradata ord ordaudio orddicom orddoc order ordimage ordinality ordvideo organization orlany orlvary out outer outfile outline output over overflow overriding package pad parallel parallel_enable parameters parent parse partial partition partitions pascal passing password password_grace_time password_lock_time password_reuse_max password_reuse_time password_verify_function patch path patindex pctincrease pctthreshold pctused pctversion percent percent_rank percentile_cont percentile_disc performance period period_add period_diff permanent physical pi pipe pipelined pivot pluggable plugin policy position post_transaction pow power pragma prebuilt precedes preceding precision prediction prediction_cost prediction_details prediction_probability prediction_set prepare present preserve prior priority private private_sga privileges procedural procedure procedure_analyze processlist profiles project prompt protection public publishingservername purge quarter query quick quiesce quota quotename radians raise rand range rank raw read reads readsize rebuild record records recover recovery recursive recycle redo reduced ref reference referenced references referencing refresh regexp_like register regr_avgx regr_avgy regr_count regr_intercept regr_r2 regr_slope regr_sxx regr_sxy reject rekey relational relative relaylog release release_lock relies_on relocate rely rem remainder rename repair repeat replace replicate replication required reset resetlogs resize resource respect restore restricted result result_cache resumable resume retention return returning returns reuse reverse revoke right rlike role roles rollback rolling rollup round row row_count rowdependencies rowid rownum rows rtrim rules safe salt sample save savepoint sb1 sb2 sb4 scan schema schemacheck scn scope scroll sdo_georaster sdo_topo_geometry search sec_to_time second section securefile security seed segment select self sequence sequential serializable server servererror session session_user sessions_per_user set sets settings sha sha1 sha2 share shared shared_pool short show shrink shutdown si_averagecolor si_colorhistogram si_featurelist si_positionalcolor si_stillimage si_texture siblings sid sign sin size size_t sizes skip slave sleep smalldatetimefromparts smallfile snapshot some soname sort soundex source space sparse spfile split sql sql_big_result sql_buffer_result sql_cache sql_calc_found_rows sql_small_result sql_variant_property sqlcode sqldata sqlerror sqlname sqlstate sqrt square standalone standby start starting startup statement static statistics stats_binomial_test stats_crosstab stats_ks_test stats_mode stats_mw_test stats_one_way_anova stats_t_test_ stats_t_test_indep stats_t_test_one stats_t_test_paired stats_wsr_test status std stddev stddev_pop stddev_samp stdev stop storage store stored str str_to_date straight_join strcmp strict string struct stuff style subdate subpartition subpartitions substitutable substr substring subtime subtring_index subtype success sum suspend switch switchoffset switchover sync synchronous synonym sys sys_xmlagg sysasm sysaux sysdate sysdatetimeoffset sysdba sysoper system system_user sysutcdatetime table tables tablespace tan tdo template temporary terminated tertiary_weights test than then thread through tier ties time time_format time_zone timediff timefromparts timeout timestamp timestampadd timestampdiff timezone_abbr timezone_minute timezone_region to to_base64 to_date to_days to_seconds todatetimeoffset trace tracking transaction transactional translate translation treat trigger trigger_nestlevel triggers trim truncate try_cast try_convert try_parse type ub1 ub2 ub4 ucase unarchived unbounded uncompress under undo unhex unicode uniform uninstall union unique unix_timestamp unknown unlimited unlock unpivot unrecoverable unsafe unsigned until untrusted unusable unused update updated upgrade upped upper upsert url urowid usable usage use use_stored_outlines user user_data user_resources users using utc_date utc_timestamp uuid uuid_short validate validate_password_strength validation valist value values var var_samp varcharc vari varia variab variabl variable variables variance varp varraw varrawc varray verify version versions view virtual visible void wait wallet warning warnings week weekday weekofyear wellformed when whene whenev wheneve whenever where while whitespace with within without work wrapped xdb xml xmlagg xmlattributes xmlcast xmlcolattval xmlelement xmlexists xmlforest xmlindex xmlnamespaces xmlpi xmlquery xmlroot xmlschema xmlserialize xmltable xmltype xor year year_to_month years yearweek",literal:"true false null",built_in:"array bigint binary bit blob boolean char character date dec decimal float int int8 integer interval number numeric real record serial serial8 smallint text varchar varying void"},c:[{cN:"string",b:"'",e:"'",c:[e.BE,{b:"''"}]},{cN:"string",b:'"',e:'"',c:[e.BE,{b:'""'}]},{cN:"string",b:"`",e:"`",c:[e.BE]},e.CNM,e.CBCM,t]},e.CBCM,t]}});hljs.registerLanguage("apache",function(e){var r={cN:"number",b:"[\\$%]\\d+"};return{aliases:["apacheconf"],cI:!0,c:[e.HCM,{cN:"section",b:""},{cN:"attribute",b:/\w+/,r:0,k:{nomarkup:"order deny allow setenv rewriterule rewriteengine rewritecond documentroot sethandler errordocument loadmodule options header listen serverroot servername"},starts:{e:/$/,r:0,k:{literal:"on off all"},c:[{cN:"meta",b:"\\s\\[",e:"\\]$"},{cN:"variable",b:"[\\$%]\\{",e:"\\}",c:["self",r]},r,e.QSM]}}],i:/\S/}});hljs.registerLanguage("haskell",function(e){var i={v:[e.C("--","$"),e.C("{-","-}",{c:["self"]})]},a={cN:"meta",b:"{-#",e:"#-}"},l={cN:"meta",b:"^#",e:"$"},c={cN:"type",b:"\\b[A-Z][\\w']*",r:0},n={b:"\\(",e:"\\)",i:'"',c:[a,l,{cN:"type",b:"\\b[A-Z][\\w]*(\\((\\.\\.|,|\\w+)\\))?"},e.inherit(e.TM,{b:"[_a-z][\\w']*"}),i]},s={b:"{",e:"}",c:n.c};return{aliases:["hs"],k:"let in if then else case of where do module import hiding qualified type data newtype deriving class instance as default infix infixl infixr foreign export ccall stdcall cplusplus jvm dotnet safe unsafe family forall mdo proc rec",c:[{bK:"module",e:"where",k:"module where",c:[n,i],i:"\\W\\.|;"},{b:"\\bimport\\b",e:"$",k:"import qualified as hiding",c:[n,i],i:"\\W\\.|;"},{cN:"class",b:"^(\\s*)?(class|instance)\\b",e:"where",k:"class family instance where",c:[c,n,i]},{cN:"class",b:"\\b(data|(new)?type)\\b",e:"$",k:"data family type newtype deriving",c:[a,c,n,s,i]},{bK:"default",e:"$",c:[c,n,i]},{bK:"infix infixl infixr",e:"$",c:[e.CNM,i]},{b:"\\bforeign\\b",e:"$",k:"foreign import export ccall stdcall cplusplus jvm dotnet safe unsafe",c:[c,e.QSM,i]},{cN:"meta",b:"#!\\/usr\\/bin\\/env runhaskell",e:"$"},a,l,e.QSM,e.CNM,c,e.inherit(e.TM,{b:"^[_a-z][\\w']*"}),i,{b:"->|<-"}]}});hljs.registerLanguage("scala",function(e){var t={cN:"meta",b:"@[A-Za-z]+"},a={cN:"subst",v:[{b:"\\$[A-Za-z0-9_]+"},{b:"\\${",e:"}"}]},r={cN:"string",v:[{b:'"',e:'"',i:"\\n",c:[e.BE]},{b:'"""',e:'"""',r:10},{b:'[a-z]+"',e:'"',i:"\\n",c:[e.BE,a]},{cN:"string",b:'[a-z]+"""',e:'"""',c:[a],r:10}]},c={cN:"symbol",b:"'\\w[\\w\\d_]*(?!')"},i={cN:"type",b:"\\b[A-Z][A-Za-z0-9_]*",r:0},s={cN:"title",b:/[^0-9\n\t "'(),.`{}\[\]:;][^\n\t "'(),.`{}\[\]:;]+|[^0-9\n\t "'(),.`{}\[\]:;=]/,r:0},n={cN:"class",bK:"class object trait type",e:/[:={\[\n;]/,eE:!0,c:[{bK:"extends with",r:10},{b:/\[/,e:/\]/,eB:!0,eE:!0,r:0,c:[i]},{cN:"params",b:/\(/,e:/\)/,eB:!0,eE:!0,r:0,c:[i]},s]},l={cN:"function",bK:"def",e:/[:={\[(\n;]/,eE:!0,c:[s]};return{k:{literal:"true false null",keyword:"type yield lazy override def with val var sealed abstract private trait object if forSome for while throw finally protected extends import final return else break new catch super class case package default try this match continue throws implicit"},c:[e.CLCM,e.CBCM,r,c,i,l,n,e.CNM,t]}}); \ No newline at end of file diff --git a/docs/guide/index.html b/docs/guide/index.html new file mode 100644 index 0000000..d518369 --- /dev/null +++ b/docs/guide/index.html @@ -0,0 +1,184 @@ + + + + + + Chapter 1 - Elefren Guide + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ +
+ + + + + + + + + + +
+
+

Chapter 1

+ +
+ + +
+
+ + + +
+ + + + + + + + + + + + + + + + + + + + + + + + diff --git a/docs/guide/mark.min.js b/docs/guide/mark.min.js new file mode 100644 index 0000000..1636231 --- /dev/null +++ b/docs/guide/mark.min.js @@ -0,0 +1,7 @@ +/*!*************************************************** +* mark.js v8.11.1 +* https://markjs.io/ +* Copyright (c) 2014–2018, Julian Kühnel +* Released under the MIT license https://git.io/vwTVl +*****************************************************/ +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t():"function"==typeof define&&define.amd?define(t):e.Mark=t()}(this,function(){"use strict";var e="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e},t=function(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")},n=function(){function e(e,t){for(var n=0;n1&&void 0!==arguments[1])||arguments[1],i=arguments.length>2&&void 0!==arguments[2]?arguments[2]:[],o=arguments.length>3&&void 0!==arguments[3]?arguments[3]:5e3;t(this,e),this.ctx=n,this.iframes=r,this.exclude=i,this.iframesTimeout=o}return n(e,[{key:"getContexts",value:function(){var e=[];return(void 0!==this.ctx&&this.ctx?NodeList.prototype.isPrototypeOf(this.ctx)?Array.prototype.slice.call(this.ctx):Array.isArray(this.ctx)?this.ctx:"string"==typeof this.ctx?Array.prototype.slice.call(document.querySelectorAll(this.ctx)):[this.ctx]:[]).forEach(function(t){var n=e.filter(function(e){return e.contains(t)}).length>0;-1!==e.indexOf(t)||n||e.push(t)}),e}},{key:"getIframeContents",value:function(e,t){var n=arguments.length>2&&void 0!==arguments[2]?arguments[2]:function(){},r=void 0;try{var i=e.contentWindow;if(r=i.document,!i||!r)throw new Error("iframe inaccessible")}catch(e){n()}r&&t(r)}},{key:"isIframeBlank",value:function(e){var t="about:blank",n=e.getAttribute("src").trim();return e.contentWindow.location.href===t&&n!==t&&n}},{key:"observeIframeLoad",value:function(e,t,n){var r=this,i=!1,o=null,a=function a(){if(!i){i=!0,clearTimeout(o);try{r.isIframeBlank(e)||(e.removeEventListener("load",a),r.getIframeContents(e,t,n))}catch(e){n()}}};e.addEventListener("load",a),o=setTimeout(a,this.iframesTimeout)}},{key:"onIframeReady",value:function(e,t,n){try{"complete"===e.contentWindow.document.readyState?this.isIframeBlank(e)?this.observeIframeLoad(e,t,n):this.getIframeContents(e,t,n):this.observeIframeLoad(e,t,n)}catch(e){n()}}},{key:"waitForIframes",value:function(e,t){var n=this,r=0;this.forEachIframe(e,function(){return!0},function(e){r++,n.waitForIframes(e.querySelector("html"),function(){--r||t()})},function(e){e||t()})}},{key:"forEachIframe",value:function(t,n,r){var i=this,o=arguments.length>3&&void 0!==arguments[3]?arguments[3]:function(){},a=t.querySelectorAll("iframe"),s=a.length,c=0;a=Array.prototype.slice.call(a);var u=function(){--s<=0&&o(c)};s||u(),a.forEach(function(t){e.matches(t,i.exclude)?u():i.onIframeReady(t,function(e){n(t)&&(c++,r(e)),u()},u)})}},{key:"createIterator",value:function(e,t,n){return document.createNodeIterator(e,t,n,!1)}},{key:"createInstanceOnIframe",value:function(t){return new e(t.querySelector("html"),this.iframes)}},{key:"compareNodeIframe",value:function(e,t,n){if(e.compareDocumentPosition(n)&Node.DOCUMENT_POSITION_PRECEDING){if(null===t)return!0;if(t.compareDocumentPosition(n)&Node.DOCUMENT_POSITION_FOLLOWING)return!0}return!1}},{key:"getIteratorNode",value:function(e){var t=e.previousNode();return{prevNode:t,node:null===t?e.nextNode():e.nextNode()&&e.nextNode()}}},{key:"checkIframeFilter",value:function(e,t,n,r){var i=!1,o=!1;return r.forEach(function(e,t){e.val===n&&(i=t,o=e.handled)}),this.compareNodeIframe(e,t,n)?(!1!==i||o?!1===i||o||(r[i].handled=!0):r.push({val:n,handled:!0}),!0):(!1===i&&r.push({val:n,handled:!1}),!1)}},{key:"handleOpenIframes",value:function(e,t,n,r){var i=this;e.forEach(function(e){e.handled||i.getIframeContents(e.val,function(e){i.createInstanceOnIframe(e).forEachNode(t,n,r)})})}},{key:"iterateThroughNodes",value:function(e,t,n,r,i){for(var o,a=this,s=this.createIterator(t,e,r),c=[],u=[],l=void 0,h=void 0;void 0,o=a.getIteratorNode(s),h=o.prevNode,l=o.node;)this.iframes&&this.forEachIframe(t,function(e){return a.checkIframeFilter(l,h,e,c)},function(t){a.createInstanceOnIframe(t).forEachNode(e,function(e){return u.push(e)},r)}),u.push(l);u.forEach(function(e){n(e)}),this.iframes&&this.handleOpenIframes(c,e,n,r),i()}},{key:"forEachNode",value:function(e,t,n){var r=this,i=arguments.length>3&&void 0!==arguments[3]?arguments[3]:function(){},o=this.getContexts(),a=o.length;a||i(),o.forEach(function(o){var s=function(){r.iterateThroughNodes(e,o,t,n,function(){--a<=0&&i()})};r.iframes?r.waitForIframes(o,s):s()})}}],[{key:"matches",value:function(e,t){var n="string"==typeof t?[t]:t,r=e.matches||e.matchesSelector||e.msMatchesSelector||e.mozMatchesSelector||e.oMatchesSelector||e.webkitMatchesSelector;if(r){var i=!1;return n.every(function(t){return!r.call(e,t)||(i=!0,!1)}),i}return!1}}]),e}(),o=function(){function e(n){t(this,e),this.opt=r({},{diacritics:!0,synonyms:{},accuracy:"partially",caseSensitive:!1,ignoreJoiners:!1,ignorePunctuation:[],wildcards:"disabled"},n)}return n(e,[{key:"create",value:function(e){return"disabled"!==this.opt.wildcards&&(e=this.setupWildcardsRegExp(e)),e=this.escapeStr(e),Object.keys(this.opt.synonyms).length&&(e=this.createSynonymsRegExp(e)),(this.opt.ignoreJoiners||this.opt.ignorePunctuation.length)&&(e=this.setupIgnoreJoinersRegExp(e)),this.opt.diacritics&&(e=this.createDiacriticsRegExp(e)),e=this.createMergedBlanksRegExp(e),(this.opt.ignoreJoiners||this.opt.ignorePunctuation.length)&&(e=this.createJoinersRegExp(e)),"disabled"!==this.opt.wildcards&&(e=this.createWildcardsRegExp(e)),e=this.createAccuracyRegExp(e),new RegExp(e,"gm"+(this.opt.caseSensitive?"":"i"))}},{key:"escapeStr",value:function(e){return e.replace(/[\-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g,"\\$&")}},{key:"createSynonymsRegExp",value:function(e){var t=this.opt.synonyms,n=this.opt.caseSensitive?"":"i",r=this.opt.ignoreJoiners||this.opt.ignorePunctuation.length?"\0":"";for(var i in t)if(t.hasOwnProperty(i)){var o=t[i],a="disabled"!==this.opt.wildcards?this.setupWildcardsRegExp(i):this.escapeStr(i),s="disabled"!==this.opt.wildcards?this.setupWildcardsRegExp(o):this.escapeStr(o);""!==a&&""!==s&&(e=e.replace(new RegExp("("+this.escapeStr(a)+"|"+this.escapeStr(s)+")","gm"+n),r+"("+this.processSynonyms(a)+"|"+this.processSynonyms(s)+")"+r))}return e}},{key:"processSynonyms",value:function(e){return(this.opt.ignoreJoiners||this.opt.ignorePunctuation.length)&&(e=this.setupIgnoreJoinersRegExp(e)),e}},{key:"setupWildcardsRegExp",value:function(e){return(e=e.replace(/(?:\\)*\?/g,function(e){return"\\"===e.charAt(0)?"?":""})).replace(/(?:\\)*\*/g,function(e){return"\\"===e.charAt(0)?"*":""})}},{key:"createWildcardsRegExp",value:function(e){var t="withSpaces"===this.opt.wildcards;return e.replace(/\u0001/g,t?"[\\S\\s]?":"\\S?").replace(/\u0002/g,t?"[\\S\\s]*?":"\\S*")}},{key:"setupIgnoreJoinersRegExp",value:function(e){return e.replace(/[^(|)\\]/g,function(e,t,n){var r=n.charAt(t+1);return/[(|)\\]/.test(r)||""===r?e:e+"\0"})}},{key:"createJoinersRegExp",value:function(e){var t=[],n=this.opt.ignorePunctuation;return Array.isArray(n)&&n.length&&t.push(this.escapeStr(n.join(""))),this.opt.ignoreJoiners&&t.push("\\u00ad\\u200b\\u200c\\u200d"),t.length?e.split(/\u0000+/).join("["+t.join("")+"]*"):e}},{key:"createDiacriticsRegExp",value:function(e){var t=this.opt.caseSensitive?"":"i",n=this.opt.caseSensitive?["aàáảãạăằắẳẵặâầấẩẫậäåāą","AÀÁẢÃẠĂẰẮẲẴẶÂẦẤẨẪẬÄÅĀĄ","cçćč","CÇĆČ","dđď","DĐĎ","eèéẻẽẹêềếểễệëěēę","EÈÉẺẼẸÊỀẾỂỄỆËĚĒĘ","iìíỉĩịîïī","IÌÍỈĨỊÎÏĪ","lł","LŁ","nñňń","NÑŇŃ","oòóỏõọôồốổỗộơởỡớờợöøō","OÒÓỎÕỌÔỒỐỔỖỘƠỞỠỚỜỢÖØŌ","rř","RŘ","sšśșş","SŠŚȘŞ","tťțţ","TŤȚŢ","uùúủũụưừứửữựûüůū","UÙÚỦŨỤƯỪỨỬỮỰÛÜŮŪ","yýỳỷỹỵÿ","YÝỲỶỸỴŸ","zžżź","ZŽŻŹ"]:["aàáảãạăằắẳẵặâầấẩẫậäåāąAÀÁẢÃẠĂẰẮẲẴẶÂẦẤẨẪẬÄÅĀĄ","cçćčCÇĆČ","dđďDĐĎ","eèéẻẽẹêềếểễệëěēęEÈÉẺẼẸÊỀẾỂỄỆËĚĒĘ","iìíỉĩịîïīIÌÍỈĨỊÎÏĪ","lłLŁ","nñňńNÑŇŃ","oòóỏõọôồốổỗộơởỡớờợöøōOÒÓỎÕỌÔỒỐỔỖỘƠỞỠỚỜỢÖØŌ","rřRŘ","sšśșşSŠŚȘŞ","tťțţTŤȚŢ","uùúủũụưừứửữựûüůūUÙÚỦŨỤƯỪỨỬỮỰÛÜŮŪ","yýỳỷỹỵÿYÝỲỶỸỴŸ","zžżźZŽŻŹ"],r=[];return e.split("").forEach(function(i){n.every(function(n){if(-1!==n.indexOf(i)){if(r.indexOf(n)>-1)return!1;e=e.replace(new RegExp("["+n+"]","gm"+t),"["+n+"]"),r.push(n)}return!0})}),e}},{key:"createMergedBlanksRegExp",value:function(e){return e.replace(/[\s]+/gim,"[\\s]+")}},{key:"createAccuracyRegExp",value:function(e){var t=this,n=this.opt.accuracy,r="string"==typeof n?n:n.value,i="";switch(("string"==typeof n?[]:n.limiters).forEach(function(e){i+="|"+t.escapeStr(e)}),r){case"partially":default:return"()("+e+")";case"complementary":return"()([^"+(i="\\s"+(i||this.escapeStr("!\"#$%&'()*+,-./:;<=>?@[\\]^_`{|}~¡¿")))+"]*"+e+"[^"+i+"]*)";case"exactly":return"(^|\\s"+i+")("+e+")(?=$|\\s"+i+")"}}}]),e}(),a=function(){function a(e){t(this,a),this.ctx=e,this.ie=!1;var n=window.navigator.userAgent;(n.indexOf("MSIE")>-1||n.indexOf("Trident")>-1)&&(this.ie=!0)}return n(a,[{key:"log",value:function(t){var n=arguments.length>1&&void 0!==arguments[1]?arguments[1]:"debug",r=this.opt.log;this.opt.debug&&"object"===(void 0===r?"undefined":e(r))&&"function"==typeof r[n]&&r[n]("mark.js: "+t)}},{key:"getSeparatedKeywords",value:function(e){var t=this,n=[];return e.forEach(function(e){t.opt.separateWordSearch?e.split(" ").forEach(function(e){e.trim()&&-1===n.indexOf(e)&&n.push(e)}):e.trim()&&-1===n.indexOf(e)&&n.push(e)}),{keywords:n.sort(function(e,t){return t.length-e.length}),length:n.length}}},{key:"isNumeric",value:function(e){return Number(parseFloat(e))==e}},{key:"checkRanges",value:function(e){var t=this;if(!Array.isArray(e)||"[object Object]"!==Object.prototype.toString.call(e[0]))return this.log("markRanges() will only accept an array of objects"),this.opt.noMatch(e),[];var n=[],r=0;return e.sort(function(e,t){return e.start-t.start}).forEach(function(e){var i=t.callNoMatchOnInvalidRanges(e,r),o=i.start,a=i.end;i.valid&&(e.start=o,e.length=a-o,n.push(e),r=a)}),n}},{key:"callNoMatchOnInvalidRanges",value:function(e,t){var n=void 0,r=void 0,i=!1;return e&&void 0!==e.start?(r=(n=parseInt(e.start,10))+parseInt(e.length,10),this.isNumeric(e.start)&&this.isNumeric(e.length)&&r-t>0&&r-n>0?i=!0:(this.log("Ignoring invalid or overlapping range: "+JSON.stringify(e)),this.opt.noMatch(e))):(this.log("Ignoring invalid range: "+JSON.stringify(e)),this.opt.noMatch(e)),{start:n,end:r,valid:i}}},{key:"checkWhitespaceRanges",value:function(e,t,n){var r=void 0,i=!0,o=n.length,a=t-o,s=parseInt(e.start,10)-a;return(r=(s=s>o?o:s)+parseInt(e.length,10))>o&&(r=o,this.log("End range automatically set to the max value of "+o)),s<0||r-s<0||s>o||r>o?(i=!1,this.log("Invalid range: "+JSON.stringify(e)),this.opt.noMatch(e)):""===n.substring(s,r).replace(/\s+/g,"")&&(i=!1,this.log("Skipping whitespace only range: "+JSON.stringify(e)),this.opt.noMatch(e)),{start:s,end:r,valid:i}}},{key:"getTextNodes",value:function(e){var t=this,n="",r=[];this.iterator.forEachNode(NodeFilter.SHOW_TEXT,function(e){r.push({start:n.length,end:(n+=e.textContent).length,node:e})},function(e){return t.matchesExclude(e.parentNode)?NodeFilter.FILTER_REJECT:NodeFilter.FILTER_ACCEPT},function(){e({value:n,nodes:r})})}},{key:"matchesExclude",value:function(e){return i.matches(e,this.opt.exclude.concat(["script","style","title","head","html"]))}},{key:"wrapRangeInTextNode",value:function(e,t,n){var r=this.opt.element?this.opt.element:"mark",i=e.splitText(t),o=i.splitText(n-t),a=document.createElement(r);return a.setAttribute("data-markjs","true"),this.opt.className&&a.setAttribute("class",this.opt.className),a.textContent=i.textContent,i.parentNode.replaceChild(a,i),o}},{key:"wrapRangeInMappedTextNode",value:function(e,t,n,r,i){var o=this;e.nodes.every(function(a,s){var c=e.nodes[s+1];if(void 0===c||c.start>t){if(!r(a.node))return!1;var u=t-a.start,l=(n>a.end?a.end:n)-a.start,h=e.value.substr(0,a.start),f=e.value.substr(l+a.start);if(a.node=o.wrapRangeInTextNode(a.node,u,l),e.value=h+f,e.nodes.forEach(function(t,n){n>=s&&(e.nodes[n].start>0&&n!==s&&(e.nodes[n].start-=l),e.nodes[n].end-=l)}),n-=l,i(a.node.previousSibling,a.start),!(n>a.end))return!1;t=a.end}return!0})}},{key:"wrapGroups",value:function(e,t,n,r){return r((e=this.wrapRangeInTextNode(e,t,t+n)).previousSibling),e}},{key:"separateGroups",value:function(e,t,n,r,i){for(var o=t.length,a=1;a-1&&r(t[a],e)&&(e=this.wrapGroups(e,s,t[a].length,i))}return e}},{key:"wrapMatches",value:function(e,t,n,r,i){var o=this,a=0===t?0:t+1;this.getTextNodes(function(t){t.nodes.forEach(function(t){t=t.node;for(var i=void 0;null!==(i=e.exec(t.textContent))&&""!==i[a];){if(o.opt.separateGroups)t=o.separateGroups(t,i,a,n,r);else{if(!n(i[a],t))continue;var s=i.index;if(0!==a)for(var c=1;c + + + + + Elefren Guide + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ +
+ + + + + + + + + + +
+
+

Chapter 1

+ +
+ + +
+
+ + + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/docs/guide/searcher.js b/docs/guide/searcher.js new file mode 100644 index 0000000..7fd97d4 --- /dev/null +++ b/docs/guide/searcher.js @@ -0,0 +1,477 @@ +"use strict"; +window.search = window.search || {}; +(function search(search) { + // Search functionality + // + // You can use !hasFocus() to prevent keyhandling in your key + // event handlers while the user is typing their search. + + if (!Mark || !elasticlunr) { + return; + } + + //IE 11 Compatibility from https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/startsWith + if (!String.prototype.startsWith) { + String.prototype.startsWith = function(search, pos) { + return this.substr(!pos || pos < 0 ? 0 : +pos, search.length) === search; + }; + } + + var search_wrap = document.getElementById('search-wrapper'), + searchbar = document.getElementById('searchbar'), + searchbar_outer = document.getElementById('searchbar-outer'), + searchresults = document.getElementById('searchresults'), + searchresults_outer = document.getElementById('searchresults-outer'), + searchresults_header = document.getElementById('searchresults-header'), + searchicon = document.getElementById('search-toggle'), + content = document.getElementById('content'), + + searchindex = null, + doc_urls = [], + results_options = { + teaser_word_count: 30, + limit_results: 30, + }, + search_options = { + bool: "AND", + expand: true, + fields: { + title: {boost: 1}, + body: {boost: 1}, + breadcrumbs: {boost: 0} + } + }, + mark_exclude = [], + marker = new Mark(content), + current_searchterm = "", + URL_SEARCH_PARAM = 'search', + URL_MARK_PARAM = 'highlight', + teaser_count = 0, + + SEARCH_HOTKEY_KEYCODE = 83, + ESCAPE_KEYCODE = 27, + DOWN_KEYCODE = 40, + UP_KEYCODE = 38, + SELECT_KEYCODE = 13; + + function hasFocus() { + return searchbar === document.activeElement; + } + + function removeChildren(elem) { + while (elem.firstChild) { + elem.removeChild(elem.firstChild); + } + } + + // Helper to parse a url into its building blocks. + function parseURL(url) { + var a = document.createElement('a'); + a.href = url; + return { + source: url, + protocol: a.protocol.replace(':',''), + host: a.hostname, + port: a.port, + params: (function(){ + var ret = {}; + var seg = a.search.replace(/^\?/,'').split('&'); + var len = seg.length, i = 0, s; + for (;i': '>', + '"': '"', + "'": ''' + }; + var repl = function(c) { return MAP[c]; }; + return function(s) { + return s.replace(/[&<>'"]/g, repl); + }; + })(); + + function formatSearchMetric(count, searchterm) { + if (count == 1) { + return count + " search result for '" + searchterm + "':"; + } else if (count == 0) { + return "No search results for '" + searchterm + "'."; + } else { + return count + " search results for '" + searchterm + "':"; + } + } + + function formatSearchResult(result, searchterms) { + var teaser = makeTeaser(escapeHTML(result.doc.body), searchterms); + teaser_count++; + + // The ?URL_MARK_PARAM= parameter belongs inbetween the page and the #heading-anchor + var url = doc_urls[result.ref].split("#"); + if (url.length == 1) { // no anchor found + url.push(""); + } + + return '' + result.doc.breadcrumbs + '' + + '' + + teaser + ''; + } + + function makeTeaser(body, searchterms) { + // The strategy is as follows: + // First, assign a value to each word in the document: + // Words that correspond to search terms (stemmer aware): 40 + // Normal words: 2 + // First word in a sentence: 8 + // Then use a sliding window with a constant number of words and count the + // sum of the values of the words within the window. Then use the window that got the + // maximum sum. If there are multiple maximas, then get the last one. + // Enclose the terms in . + var stemmed_searchterms = searchterms.map(function(w) { + return elasticlunr.stemmer(w.toLowerCase()); + }); + var searchterm_weight = 40; + var weighted = []; // contains elements of ["word", weight, index_in_document] + // split in sentences, then words + var sentences = body.toLowerCase().split('. '); + var index = 0; + var value = 0; + var searchterm_found = false; + for (var sentenceindex in sentences) { + var words = sentences[sentenceindex].split(' '); + value = 8; + for (var wordindex in words) { + var word = words[wordindex]; + if (word.length > 0) { + for (var searchtermindex in stemmed_searchterms) { + if (elasticlunr.stemmer(word).startsWith(stemmed_searchterms[searchtermindex])) { + value = searchterm_weight; + searchterm_found = true; + } + }; + weighted.push([word, value, index]); + value = 2; + } + index += word.length; + index += 1; // ' ' or '.' if last word in sentence + }; + index += 1; // because we split at a two-char boundary '. ' + }; + + if (weighted.length == 0) { + return body; + } + + var window_weight = []; + var window_size = Math.min(weighted.length, results_options.teaser_word_count); + + var cur_sum = 0; + for (var wordindex = 0; wordindex < window_size; wordindex++) { + cur_sum += weighted[wordindex][1]; + }; + window_weight.push(cur_sum); + for (var wordindex = 0; wordindex < weighted.length - window_size; wordindex++) { + cur_sum -= weighted[wordindex][1]; + cur_sum += weighted[wordindex + window_size][1]; + window_weight.push(cur_sum); + }; + + if (searchterm_found) { + var max_sum = 0; + var max_sum_window_index = 0; + // backwards + for (var i = window_weight.length - 1; i >= 0; i--) { + if (window_weight[i] > max_sum) { + max_sum = window_weight[i]; + max_sum_window_index = i; + } + }; + } else { + max_sum_window_index = 0; + } + + // add around searchterms + var teaser_split = []; + var index = weighted[max_sum_window_index][2]; + for (var i = max_sum_window_index; i < max_sum_window_index+window_size; i++) { + var word = weighted[i]; + if (index < word[2]) { + // missing text from index to start of `word` + teaser_split.push(body.substring(index, word[2])); + index = word[2]; + } + if (word[1] == searchterm_weight) { + teaser_split.push("") + } + index = word[2] + word[0].length; + teaser_split.push(body.substring(word[2], index)); + if (word[1] == searchterm_weight) { + teaser_split.push("") + } + }; + + return teaser_split.join(''); + } + + function init(config) { + results_options = config.results_options; + search_options = config.search_options; + searchbar_outer = config.searchbar_outer; + doc_urls = config.doc_urls; + searchindex = elasticlunr.Index.load(config.index); + + // Set up events + searchicon.addEventListener('click', function(e) { searchIconClickHandler(); }, false); + searchbar.addEventListener('keyup', function(e) { searchbarKeyUpHandler(); }, false); + document.addEventListener('keydown', function(e) { globalKeyHandler(e); }, false); + // If the user uses the browser buttons, do the same as if a reload happened + window.onpopstate = function(e) { doSearchOrMarkFromUrl(); }; + // Suppress "submit" events so the page doesn't reload when the user presses Enter + document.addEventListener('submit', function(e) { e.preventDefault(); }, false); + + // If reloaded, do the search or mark again, depending on the current url parameters + doSearchOrMarkFromUrl(); + } + + function unfocusSearchbar() { + // hacky, but just focusing a div only works once + var tmp = document.createElement('input'); + tmp.setAttribute('style', 'position: absolute; opacity: 0;'); + searchicon.appendChild(tmp); + tmp.focus(); + tmp.remove(); + } + + // On reload or browser history backwards/forwards events, parse the url and do search or mark + function doSearchOrMarkFromUrl() { + // Check current URL for search request + var url = parseURL(window.location.href); + if (url.params.hasOwnProperty(URL_SEARCH_PARAM) + && url.params[URL_SEARCH_PARAM] != "") { + showSearch(true); + searchbar.value = decodeURIComponent( + (url.params[URL_SEARCH_PARAM]+'').replace(/\+/g, '%20')); + searchbarKeyUpHandler(); // -> doSearch() + } else { + showSearch(false); + } + + if (url.params.hasOwnProperty(URL_MARK_PARAM)) { + var words = url.params[URL_MARK_PARAM].split(' '); + marker.mark(words, { + exclude: mark_exclude + }); + + var markers = document.querySelectorAll("mark"); + function hide() { + for (var i = 0; i < markers.length; i++) { + markers[i].classList.add("fade-out"); + window.setTimeout(function(e) { marker.unmark(); }, 300); + } + } + for (var i = 0; i < markers.length; i++) { + markers[i].addEventListener('click', hide); + } + } + } + + // Eventhandler for keyevents on `document` + function globalKeyHandler(e) { + if (e.altKey || e.ctrlKey || e.metaKey || e.shiftKey || e.target.type === 'textarea') { return; } + + if (e.keyCode === ESCAPE_KEYCODE) { + e.preventDefault(); + searchbar.classList.remove("active"); + setSearchUrlParameters("", + (searchbar.value.trim() !== "") ? "push" : "replace"); + if (hasFocus()) { + unfocusSearchbar(); + } + showSearch(false); + marker.unmark(); + } else if (!hasFocus() && e.keyCode === SEARCH_HOTKEY_KEYCODE) { + e.preventDefault(); + showSearch(true); + window.scrollTo(0, 0); + searchbar.select(); + } else if (hasFocus() && e.keyCode === DOWN_KEYCODE) { + e.preventDefault(); + unfocusSearchbar(); + searchresults.firstElementChild.classList.add("focus"); + } else if (!hasFocus() && (e.keyCode === DOWN_KEYCODE + || e.keyCode === UP_KEYCODE + || e.keyCode === SELECT_KEYCODE)) { + // not `:focus` because browser does annoying scrolling + var focused = searchresults.querySelector("li.focus"); + if (!focused) return; + e.preventDefault(); + if (e.keyCode === DOWN_KEYCODE) { + var next = focused.nextElementSibling; + if (next) { + focused.classList.remove("focus"); + next.classList.add("focus"); + } + } else if (e.keyCode === UP_KEYCODE) { + focused.classList.remove("focus"); + var prev = focused.previousElementSibling; + if (prev) { + prev.classList.add("focus"); + } else { + searchbar.select(); + } + } else { // SELECT_KEYCODE + window.location.assign(focused.querySelector('a')); + } + } + } + + function showSearch(yes) { + if (yes) { + search_wrap.classList.remove('hidden'); + searchicon.setAttribute('aria-expanded', 'true'); + } else { + search_wrap.classList.add('hidden'); + searchicon.setAttribute('aria-expanded', 'false'); + var results = searchresults.children; + for (var i = 0; i < results.length; i++) { + results[i].classList.remove("focus"); + } + } + } + + function showResults(yes) { + if (yes) { + searchresults_outer.classList.remove('hidden'); + } else { + searchresults_outer.classList.add('hidden'); + } + } + + // Eventhandler for search icon + function searchIconClickHandler() { + if (search_wrap.classList.contains('hidden')) { + showSearch(true); + window.scrollTo(0, 0); + searchbar.select(); + } else { + showSearch(false); + } + } + + // Eventhandler for keyevents while the searchbar is focused + function searchbarKeyUpHandler() { + var searchterm = searchbar.value.trim(); + if (searchterm != "") { + searchbar.classList.add("active"); + doSearch(searchterm); + } else { + searchbar.classList.remove("active"); + showResults(false); + removeChildren(searchresults); + } + + setSearchUrlParameters(searchterm, "push_if_new_search_else_replace"); + + // Remove marks + marker.unmark(); + } + + // Update current url with ?URL_SEARCH_PARAM= parameter, remove ?URL_MARK_PARAM and #heading-anchor . + // `action` can be one of "push", "replace", "push_if_new_search_else_replace" + // and replaces or pushes a new browser history item. + // "push_if_new_search_else_replace" pushes if there is no `?URL_SEARCH_PARAM=abc` yet. + function setSearchUrlParameters(searchterm, action) { + var url = parseURL(window.location.href); + var first_search = ! url.params.hasOwnProperty(URL_SEARCH_PARAM); + if (searchterm != "" || action == "push_if_new_search_else_replace") { + url.params[URL_SEARCH_PARAM] = searchterm; + delete url.params[URL_MARK_PARAM]; + url.hash = ""; + } else { + delete url.params[URL_SEARCH_PARAM]; + } + // A new search will also add a new history item, so the user can go back + // to the page prior to searching. A updated search term will only replace + // the url. + if (action == "push" || (action == "push_if_new_search_else_replace" && first_search) ) { + history.pushState({}, document.title, renderURL(url)); + } else if (action == "replace" || (action == "push_if_new_search_else_replace" && !first_search) ) { + history.replaceState({}, document.title, renderURL(url)); + } + } + + function doSearch(searchterm) { + + // Don't search the same twice + if (current_searchterm == searchterm) { return; } + else { current_searchterm = searchterm; } + + if (searchindex == null) { return; } + + // Do the actual search + var results = searchindex.search(searchterm, search_options); + var resultcount = Math.min(results.length, results_options.limit_results); + + // Display search metrics + searchresults_header.innerText = formatSearchMetric(resultcount, searchterm); + + // Clear and insert results + var searchterms = searchterm.split(' '); + removeChildren(searchresults); + for(var i = 0; i < resultcount ; i++){ + var resultElem = document.createElement('li'); + resultElem.innerHTML = formatSearchResult(results[i], searchterms); + searchresults.appendChild(resultElem); + } + + // Display results + showResults(true); + } + + fetch(path_to_root + 'searchindex.json') + .then(response => response.json()) + .then(json => init(json)) + .catch(error => { // Try to load searchindex.js if fetch failed + var script = document.createElement('script'); + script.src = path_to_root + 'searchindex.js'; + script.onload = () => init(window.search); + document.head.appendChild(script); + }); + + // Exported functions + search.hasFocus = hasFocus; +})(window.search); diff --git a/docs/guide/searchindex.js b/docs/guide/searchindex.js new file mode 100644 index 0000000..3b2b2c1 --- /dev/null +++ b/docs/guide/searchindex.js @@ -0,0 +1 @@ +window.search = {"doc_urls":["chapter_1.html#chapter-1"],"index":{"documentStore":{"docInfo":{"0":{"body":0,"breadcrumbs":2,"title":2}},"docs":{"0":{"body":"","breadcrumbs":"Chapter 1","id":"0","title":"Chapter 1"}},"length":1,"save":true},"fields":["title","body","breadcrumbs"],"index":{"body":{"root":{"1":{"df":1,"docs":{"0":{"tf":1.0}}},"c":{"df":0,"docs":{},"h":{"a":{"df":0,"docs":{},"p":{"df":0,"docs":{},"t":{"df":0,"docs":{},"e":{"df":0,"docs":{},"r":{"df":1,"docs":{"0":{"tf":1.0}}}}}}},"df":0,"docs":{}}},"df":0,"docs":{}}},"breadcrumbs":{"root":{"1":{"df":1,"docs":{"0":{"tf":1.4142135623730951}}},"c":{"df":0,"docs":{},"h":{"a":{"df":0,"docs":{},"p":{"df":0,"docs":{},"t":{"df":0,"docs":{},"e":{"df":0,"docs":{},"r":{"df":1,"docs":{"0":{"tf":1.4142135623730951}}}}}}},"df":0,"docs":{}}},"df":0,"docs":{}}},"title":{"root":{"1":{"df":1,"docs":{"0":{"tf":1.0}}},"c":{"df":0,"docs":{},"h":{"a":{"df":0,"docs":{},"p":{"df":0,"docs":{},"t":{"df":0,"docs":{},"e":{"df":0,"docs":{},"r":{"df":1,"docs":{"0":{"tf":1.0}}}}}}},"df":0,"docs":{}}},"df":0,"docs":{}}}},"pipeline":["trimmer","stopWordFilter","stemmer"],"ref":"id","version":"0.9.5"},"results_options":{"limit_results":30,"teaser_word_count":30},"search_options":{"bool":"OR","expand":true,"fields":{"body":{"boost":1},"breadcrumbs":{"boost":1},"title":{"boost":2}}}}; \ No newline at end of file diff --git a/docs/guide/searchindex.json b/docs/guide/searchindex.json new file mode 100644 index 0000000..8e3c2fc --- /dev/null +++ b/docs/guide/searchindex.json @@ -0,0 +1 @@ +{"doc_urls":["chapter_1.html#chapter-1"],"index":{"documentStore":{"docInfo":{"0":{"body":0,"breadcrumbs":2,"title":2}},"docs":{"0":{"body":"","breadcrumbs":"Chapter 1","id":"0","title":"Chapter 1"}},"length":1,"save":true},"fields":["title","body","breadcrumbs"],"index":{"body":{"root":{"1":{"df":1,"docs":{"0":{"tf":1.0}}},"c":{"df":0,"docs":{},"h":{"a":{"df":0,"docs":{},"p":{"df":0,"docs":{},"t":{"df":0,"docs":{},"e":{"df":0,"docs":{},"r":{"df":1,"docs":{"0":{"tf":1.0}}}}}}},"df":0,"docs":{}}},"df":0,"docs":{}}},"breadcrumbs":{"root":{"1":{"df":1,"docs":{"0":{"tf":1.4142135623730951}}},"c":{"df":0,"docs":{},"h":{"a":{"df":0,"docs":{},"p":{"df":0,"docs":{},"t":{"df":0,"docs":{},"e":{"df":0,"docs":{},"r":{"df":1,"docs":{"0":{"tf":1.4142135623730951}}}}}}},"df":0,"docs":{}}},"df":0,"docs":{}}},"title":{"root":{"1":{"df":1,"docs":{"0":{"tf":1.0}}},"c":{"df":0,"docs":{},"h":{"a":{"df":0,"docs":{},"p":{"df":0,"docs":{},"t":{"df":0,"docs":{},"e":{"df":0,"docs":{},"r":{"df":1,"docs":{"0":{"tf":1.0}}}}}}},"df":0,"docs":{}}},"df":0,"docs":{}}}},"pipeline":["trimmer","stopWordFilter","stemmer"],"ref":"id","version":"0.9.5"},"results_options":{"limit_results":30,"teaser_word_count":30},"search_options":{"bool":"OR","expand":true,"fields":{"body":{"boost":1},"breadcrumbs":{"boost":1},"title":{"boost":2}}}} \ No newline at end of file diff --git a/docs/guide/tomorrow-night.css b/docs/guide/tomorrow-night.css new file mode 100644 index 0000000..9788e08 --- /dev/null +++ b/docs/guide/tomorrow-night.css @@ -0,0 +1,96 @@ +/* Tomorrow Night Theme */ +/* http://jmblog.github.com/color-themes-for-google-code-highlightjs */ +/* Original theme - https://github.com/chriskempson/tomorrow-theme */ +/* http://jmblog.github.com/color-themes-for-google-code-highlightjs */ + +/* Tomorrow Comment */ +.hljs-comment { + color: #969896; +} + +/* Tomorrow Red */ +.hljs-variable, +.hljs-attribute, +.hljs-tag, +.hljs-regexp, +.ruby .hljs-constant, +.xml .hljs-tag .hljs-title, +.xml .hljs-pi, +.xml .hljs-doctype, +.html .hljs-doctype, +.css .hljs-id, +.css .hljs-class, +.css .hljs-pseudo { + color: #cc6666; +} + +/* Tomorrow Orange */ +.hljs-number, +.hljs-preprocessor, +.hljs-pragma, +.hljs-built_in, +.hljs-literal, +.hljs-params, +.hljs-constant { + color: #de935f; +} + +/* Tomorrow Yellow */ +.ruby .hljs-class .hljs-title, +.css .hljs-rule .hljs-attribute { + color: #f0c674; +} + +/* Tomorrow Green */ +.hljs-string, +.hljs-value, +.hljs-inheritance, +.hljs-header, +.hljs-name, +.ruby .hljs-symbol, +.xml .hljs-cdata { + color: #b5bd68; +} + +/* Tomorrow Aqua */ +.hljs-title, +.css .hljs-hexcolor { + color: #8abeb7; +} + +/* Tomorrow Blue */ +.hljs-function, +.python .hljs-decorator, +.python .hljs-title, +.ruby .hljs-function .hljs-title, +.ruby .hljs-title .hljs-keyword, +.perl .hljs-sub, +.javascript .hljs-title, +.coffeescript .hljs-title { + color: #81a2be; +} + +/* Tomorrow Purple */ +.hljs-keyword, +.javascript .hljs-function { + color: #b294bb; +} + +.hljs { + display: block; + overflow-x: auto; + background: #1d1f21; + color: #c5c8c6; + padding: 0.5em; + -webkit-text-size-adjust: none; +} + +.coffeescript .javascript, +.javascript .xml, +.tex .hljs-formula, +.xml .javascript, +.xml .vbscript, +.xml .css, +.xml .hljs-cdata { + opacity: 0.5; +} diff --git a/docs/src/SUMMARY.md b/docs/src/SUMMARY.md new file mode 100644 index 0000000..7390c82 --- /dev/null +++ b/docs/src/SUMMARY.md @@ -0,0 +1,3 @@ +# Summary + +- [Chapter 1](./chapter_1.md) diff --git a/docs/src/chapter_1.md b/docs/src/chapter_1.md new file mode 100644 index 0000000..b743fda --- /dev/null +++ b/docs/src/chapter_1.md @@ -0,0 +1 @@ +# Chapter 1 diff --git a/examples/follow_profile.rs b/examples/follow_profile.rs new file mode 100644 index 0000000..84b2998 --- /dev/null +++ b/examples/follow_profile.rs @@ -0,0 +1,27 @@ +#![cfg_attr(not(feature = "toml"), allow(dead_code))] +#![cfg_attr(not(feature = "toml"), allow(unused_imports))] +#[macro_use] +extern crate pretty_env_logger; +extern crate elefren; +mod register; + +use register::MastodonClient; +use std::error; + +#[cfg(feature = "toml")] +fn main() -> Result<(), Box> { + let mastodon = register::get_mastodon_data()?; + let input = register::read_line("Enter the account id you'd like to follow: ")?; + let new_follow = mastodon.follow(input.trim())?; + + println!("{:#?}", new_follow); + Ok(()) +} + +#[cfg(not(feature = "toml"))] +fn main() { + println!( + "examples require the `toml` feature, run this command for this example:\n\ncargo run \ + --example follow_profile --features toml\n" + ); +} diff --git a/examples/follows_me.rs b/examples/follows_me.rs new file mode 100644 index 0000000..0df4414 --- /dev/null +++ b/examples/follows_me.rs @@ -0,0 +1,27 @@ +#![cfg_attr(not(feature = "toml"), allow(dead_code))] +#![cfg_attr(not(feature = "toml"), allow(unused_imports))] +#[macro_use] +extern crate pretty_env_logger; +extern crate elefren; +mod register; + +use register::MastodonClient; +use std::error; + +#[cfg(feature = "toml")] +fn main() -> Result<(), Box> { + let mastodon = register::get_mastodon_data()?; + for account in mastodon.follows_me()?.items_iter() { + println!("{}", account.acct); + } + + Ok(()) +} + +#[cfg(not(feature = "toml"))] +fn main() { + println!( + "examples require the `toml` feature, run this command for this example:\n\ncargo run \ + --example print_your_profile --features toml\n" + ); +} diff --git a/examples/home_timeline.rs b/examples/home_timeline.rs new file mode 100644 index 0000000..46519e3 --- /dev/null +++ b/examples/home_timeline.rs @@ -0,0 +1,27 @@ +#![cfg_attr(not(feature = "toml"), allow(dead_code))] +#![cfg_attr(not(feature = "toml"), allow(unused_imports))] +#[macro_use] +extern crate pretty_env_logger; +extern crate elefren; +mod register; + +use register::MastodonClient; +use std::error; + +#[cfg(feature = "toml")] +fn main() -> Result<(), Box> { + let mastodon = register::get_mastodon_data()?; + let tl = mastodon.get_home_timeline()?; + + println!("{:#?}", tl); + + Ok(()) +} + +#[cfg(not(feature = "toml"))] +fn main() { + println!( + "examples require the `toml` feature, run this command for this example:\n\ncargo run \ + --example print_your_profile --features toml\n" + ); +} diff --git a/examples/print_your_profile.rs b/examples/print_your_profile.rs new file mode 100644 index 0000000..cbcd372 --- /dev/null +++ b/examples/print_your_profile.rs @@ -0,0 +1,27 @@ +#![cfg_attr(not(feature = "toml"), allow(dead_code))] +#![cfg_attr(not(feature = "toml"), allow(unused_imports))] +#[macro_use] +extern crate pretty_env_logger; +extern crate elefren; +mod register; + +use register::MastodonClient; +use std::error; + +#[cfg(feature = "toml")] +fn main() -> Result<(), Box> { + let mastodon = register::get_mastodon_data()?; + let you = mastodon.verify_credentials()?; + + println!("{:#?}", you); + + Ok(()) +} + +#[cfg(not(feature = "toml"))] +fn main() { + println!( + "examples require the `toml` feature, run this command for this example:\n\ncargo run \ + --example print_your_profile --features toml\n" + ); +} diff --git a/examples/register/mod.rs b/examples/register/mod.rs new file mode 100644 index 0000000..d8dbaff --- /dev/null +++ b/examples/register/mod.rs @@ -0,0 +1,56 @@ +#![cfg_attr(not(feature = "toml"), allow(dead_code))] +#![cfg_attr(not(feature = "toml"), allow(unused_imports))] + +pub use elefren::prelude::*; + +use std::{error::Error, io}; + +use elefren::helpers::cli; +#[cfg(feature = "toml")] +use elefren::helpers::toml; + +#[allow(dead_code)] +#[cfg(feature = "toml")] +fn main() -> Result<(), Box> { + register()?; + Ok(()) +} + +#[allow(dead_code)] +#[cfg(feature = "toml")] +pub fn get_mastodon_data() -> Result> { + if let Ok(data) = toml::from_file("mastodon-data.toml") { + Ok(Mastodon::from(data)) + } else { + register() + } +} + +#[cfg(feature = "toml")] +pub fn register() -> Result> { + let website = read_line("Please enter your mastodon instance url:")?; + let registration = Registration::new(website.trim()) + .client_name("elefren-examples") + .scopes(Scopes::all()) + .website("https://github.com/pwoolcoc/elefren") + .build()?; + let mastodon = cli::authenticate(registration)?; + + // Save app data for using on the next run. + toml::to_file(&*mastodon, "mastodon-data.toml")?; + + Ok(mastodon) +} + +#[cfg(feature = "toml")] +pub fn read_line(message: &str) -> Result> { + println!("{}", message); + + let mut input = String::new(); + io::stdin().read_line(&mut input)?; + + Ok(input.trim().to_string()) +} + +#[cfg(not(feature = "toml"))] +fn main() {} diff --git a/examples/search.rs b/examples/search.rs new file mode 100644 index 0000000..ec13f6d --- /dev/null +++ b/examples/search.rs @@ -0,0 +1,28 @@ +#![cfg_attr(not(feature = "toml"), allow(dead_code))] +#![cfg_attr(not(feature = "toml"), allow(unused_imports))] +#[macro_use] +extern crate pretty_env_logger; +extern crate elefren; +mod register; + +use register::MastodonClient; +use std::error; + +#[cfg(feature = "toml")] +fn main() -> Result<(), Box> { + let mastodon = register::get_mastodon_data()?; + let input = register::read_line("Enter the term you'd like to search: ")?; + let result = mastodon.search(&input, false)?; + + println!("{:#?}", result); + + Ok(()) +} + +#[cfg(not(feature = "toml"))] +fn main() { + println!( + "examples require the `toml` feature, run this command for this example:\n\ncargo run \ + --example search --features toml\n" + ); +} diff --git a/examples/upload_photo.rs b/examples/upload_photo.rs new file mode 100644 index 0000000..a2e6688 --- /dev/null +++ b/examples/upload_photo.rs @@ -0,0 +1,27 @@ +#![cfg_attr(not(feature = "toml"), allow(dead_code))] +#![cfg_attr(not(feature = "toml"), allow(unused_imports))] +#[macro_use] +extern crate pretty_env_logger; +extern crate elefren; +mod register; + +use register::MastodonClient; +use std::error; + +#[cfg(feature = "toml")] +fn main() -> Result<(), Box> { + let mastodon = register::get_mastodon_data()?; + let input = register::read_line("Enter the path to the photo you'd like to post: ")?; + + mastodon.media(input.into())?; + + Ok(()) +} + +#[cfg(not(feature = "toml"))] +fn main() { + println!( + "examples require the `toml` feature, run this command for this example:\n\ncargo run \ + --example upload_photo --features toml\n" + ); +} diff --git a/rustfmt.toml b/rustfmt.toml new file mode 100644 index 0000000..e2b5e5b --- /dev/null +++ b/rustfmt.toml @@ -0,0 +1,16 @@ +format_strings = true +format_macro_matchers = true +imports_layout = "HorizontalVertical" +merge_imports = true +match_block_trailing_comma = true +merge_derives = true +newline_style = "Unix" +normalize_comments = true +reorder_impl_items = true +space_after_colon = true +space_before_colon = false +struct_lit_single_line = false +type_punctuation_density = "Wide" +use_field_init_shorthand = true +use_try_shorthand = true +wrap_comments = true diff --git a/src/apps.rs b/src/apps.rs new file mode 100644 index 0000000..7032932 --- /dev/null +++ b/src/apps.rs @@ -0,0 +1,233 @@ +use std::borrow::Cow; + +use try_from::TryInto; + +use errors::{Error, Result}; +use scopes::Scopes; + +/// Represents an application that can be registered with a mastodon instance +#[derive(Clone, Debug, Default, Serialize, PartialEq)] +pub struct App { + client_name: String, + redirect_uris: String, + scopes: Scopes, + #[serde(skip_serializing_if = "Option::is_none")] + website: Option, +} + +impl App { + /// Get an AppBuilder object + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::apps::App; + /// + /// let mut builder = App::builder(); + /// ``` + pub fn builder<'a>() -> AppBuilder<'a> { + AppBuilder::new() + } + + /// Retrieve the list of scopes that apply to this App + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// # use elefren::Error; + /// use elefren::{apps::App, scopes::Scopes}; + /// + /// # fn main() -> Result<(), Error> { + /// let mut builder = App::builder(); + /// builder.client_name("elefren-test"); + /// let app = builder.build()?; + /// let scopes = app.scopes(); + /// assert_eq!(scopes, &Scopes::read_all()); + /// # Ok(()) + /// # } + /// ``` + pub fn scopes(&self) -> &Scopes { + &self.scopes + } +} + +/// Builder struct for defining your application. +/// ``` +/// use elefren::apps::App; +/// use std::error::Error; +/// +/// # fn main() -> Result<(), Box> { +/// let mut builder = App::builder(); +/// builder.client_name("elefren_test"); +/// let app = builder.build()?; +/// # Ok(()) +/// # } +/// ``` +#[derive(Clone, Debug, Default, PartialEq, Serialize)] +pub struct AppBuilder<'a> { + client_name: Option>, + redirect_uris: Option>, + scopes: Option, + website: Option>, +} + +impl<'a> AppBuilder<'a> { + /// Creates a new AppBuilder object + pub fn new() -> Self { + Default::default() + } + + /// Name of the application. Will be displayed when the user is deciding to + /// grant permission. + /// + /// In order to turn this builder into an App, this needs to be provided + pub fn client_name>>(&mut self, name: I) -> &mut Self { + self.client_name = Some(name.into()); + self + } + + /// Where the user should be redirected after authorization + /// + /// If none is specified, the default is `urn:ietf:wg:oauth:2.0:oob` + pub fn redirect_uris>>(&mut self, uris: I) -> &mut Self { + self.redirect_uris = Some(uris.into()); + self + } + + /// Permission scope of the application. + /// + /// IF none is specified, the default is Scopes::read_all() + pub fn scopes(&mut self, scopes: Scopes) -> &mut Self { + self.scopes = Some(scopes); + self + } + + /// URL to the homepage of your application. + pub fn website>>(&mut self, website: I) -> &mut Self { + self.website = Some(website.into()); + self + } + + /// Attempts to convert this build into an `App` + /// + /// Will fail if no `client_name` was provided + pub fn build(self) -> Result { + Ok(App { + client_name: self + .client_name + .ok_or_else(|| Error::MissingField("client_name"))? + .into(), + redirect_uris: self + .redirect_uris + .unwrap_or_else(|| "urn:ietf:wg:oauth:2.0:oob".into()) + .into(), + scopes: self.scopes.unwrap_or_else(|| Scopes::read_all()), + website: self.website.map(|s| s.into()), + }) + } +} + +impl TryInto for App { + type Err = Error; + + fn try_into(self) -> Result { + Ok(self) + } +} + +impl<'a> TryInto for AppBuilder<'a> { + type Err = Error; + + fn try_into(self) -> Result { + Ok(self.build()?) + } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn test_app_builder() { + let builder = App::builder(); + assert_eq!(builder, AppBuilder::new()); + } + + #[test] + fn test_app_scopes() { + let mut builder = App::builder(); + builder.client_name("test").scopes(Scopes::all()); + let app = builder.build().expect("Couldn't build App"); + assert_eq!(app.scopes(), &Scopes::all()); + } + + #[test] + fn test_app_builder_all_methods() { + let mut builder = AppBuilder::new(); + builder.client_name("foo-test"); + builder.redirect_uris("http://example.com"); + builder.scopes(Scopes::read_all() | Scopes::write_all()); + builder.website("https://example.com"); + let app = builder.build().expect("Couldn't build App"); + assert_eq!( + app, + App { + client_name: "foo-test".to_string(), + redirect_uris: "http://example.com".to_string(), + scopes: Scopes::read_all() | Scopes::write_all(), + website: Some("https://example.com".to_string()), + } + ); + } + + #[test] + #[should_panic] + fn test_app_builder_build_fails_if_no_client_name_1() { + App::builder().build().expect("no client-name"); + } + + #[test] + #[should_panic] + fn test_app_builder_build_fails_if_no_client_name_2() { + let mut builder = App::builder(); + builder + .website("https://example.com") + .redirect_uris("https://example.com") + .scopes(Scopes::all()); + builder.build().expect("no client-name"); + } + + #[test] + fn test_app_try_into_app() { + let app = App { + client_name: "foo-test".to_string(), + redirect_uris: "http://example.com".to_string(), + scopes: Scopes::all(), + website: None, + }; + let expected = app.clone(); + let result = app.try_into().expect("Couldn't make App into App"); + assert_eq!(expected, result); + } + + #[test] + fn test_app_builder_try_into_app() { + let mut builder = App::builder(); + builder + .client_name("foo-test") + .redirect_uris("http://example.com") + .scopes(Scopes::all()); + let expected = App { + client_name: "foo-test".to_string(), + redirect_uris: "http://example.com".to_string(), + scopes: Scopes::all(), + website: None, + }; + let result = builder + .try_into() + .expect("Couldn't make AppBuilder into App"); + assert_eq!(expected, result); + } +} diff --git a/src/data.rs b/src/data.rs new file mode 100644 index 0000000..20dfd69 --- /dev/null +++ b/src/data.rs @@ -0,0 +1,17 @@ +use std::borrow::Cow; + +/// Raw data about mastodon app. Save `Data` using `serde` to prevent needing +/// to authenticate on every run. +#[derive(Clone, Debug, PartialEq, Deserialize, Serialize)] +pub struct Data { + /// Base url of instance eg. `https://mastodon.social`. + pub base: Cow<'static, str>, + /// The client's id given by the instance. + pub client_id: Cow<'static, str>, + /// The client's secret given by the instance. + pub client_secret: Cow<'static, str>, + /// Url to redirect back to your application from the instance signup. + pub redirect: Cow<'static, str>, + /// The client's access token. + pub token: Cow<'static, str>, +} diff --git a/src/entities/account.rs b/src/entities/account.rs new file mode 100644 index 0000000..499fb0a --- /dev/null +++ b/src/entities/account.rs @@ -0,0 +1,144 @@ +//! A module containing everything relating to a account returned from the api. + +use chrono::prelude::*; +use serde::de::{self, Deserialize, Deserializer, Unexpected}; +use status_builder; +use std::path::PathBuf; + +/// A struct representing an Account. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Account { + /// Equals `username` for local users, includes `@domain` for remote ones. + pub acct: String, + /// URL to the avatar image + pub avatar: String, + /// URL to the avatar static image (gif) + pub avatar_static: String, + /// The time the account was created. + pub created_at: DateTime, + /// The account's display name. + pub display_name: String, + /// The number of followers for the account. + pub followers_count: u64, + /// The number of accounts the given account is following. + pub following_count: u64, + /// URL to the header image. + pub header: String, + /// URL to the header static image (gif). + pub header_static: String, + /// The ID of the account. + pub id: String, + /// Boolean for when the account cannot be followed without waiting for + /// approval first. + pub locked: bool, + /// Biography of user. + pub note: String, + /// The number of statuses the account has made. + pub statuses_count: u64, + /// URL of the user's profile page (can be remote). + pub url: String, + /// The username of the account. + pub username: String, + /// An extra attribute given from `verify_credentials` giving defaults about + /// a user + pub source: Option, + /// If the owner decided to switch accounts, new account is in + /// this attribute + pub moved: Option>, + /// List of profile metadata fields + pub fields: Option>, + /// Boolean indicating whether this account is a bot or not + pub bot: Option, +} + +/// A single name: value pair from a user's profile +#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, Default)] +pub struct MetadataField { + /// name part of metadata + pub name: String, + /// value part of metadata + pub value: String, +} + +impl MetadataField { + pub(crate) fn new(name: &str, value: &str) -> MetadataField { + MetadataField { + name: name.into(), + value: value.into(), + } + } +} + +/// An extra object given from `verify_credentials` giving defaults about a user +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Source { + privacy: Option, + #[serde(deserialize_with = "string_or_bool")] + sensitive: bool, + note: Option, + fields: Option>, +} + +fn string_or_bool<'de, D: Deserializer<'de>>(val: D) -> ::std::result::Result { + #[derive(Clone, Debug, Deserialize, PartialEq)] + #[serde(untagged)] + pub enum BoolOrString { + Bool(bool), + Str(String), + } + + Ok(match BoolOrString::deserialize(val)? { + BoolOrString::Bool(b) => b, + BoolOrString::Str(ref s) => { + if s == "true" { + true + } else if s == "false" { + false + } else { + return Err(de::Error::invalid_value( + Unexpected::Str(s), + &"true or false", + )); + } + }, + }) +} + +#[derive(Debug, Default, Clone, Serialize, PartialEq)] +pub(crate) struct UpdateSource { + #[serde(skip_serializing_if = "Option::is_none")] + pub(crate) privacy: Option, + #[serde(skip_serializing_if = "Option::is_none")] + pub(crate) sensitive: Option, +} + +#[derive(Debug, Default, Serialize, PartialEq)] +pub(crate) struct Credentials { + #[serde(skip_serializing_if = "Option::is_none")] + pub(crate) display_name: Option, + #[serde(skip_serializing_if = "Option::is_none")] + pub(crate) note: Option, + #[serde(skip_serializing_if = "Option::is_none")] + pub(crate) avatar: Option, + #[serde(skip_serializing_if = "Option::is_none")] + pub(crate) header: Option, + #[serde(skip_serializing_if = "Option::is_none")] + pub(crate) source: Option, + #[serde(serialize_with = "fields_attributes_ser::ser")] + pub(crate) fields_attributes: Vec, +} + +mod fields_attributes_ser { + use super::*; + use serde::ser::{SerializeMap, Serializer}; + pub(crate) fn ser(attrs: &Vec, serializer: S) -> Result + where + S: Serializer, + { + let mut map = serializer.serialize_map(Some(attrs.len()))?; + for (i, field) in attrs.iter().enumerate() { + map.serialize_entry(&i, &field)?; + } + map.end() + } +} diff --git a/src/entities/attachment.rs b/src/entities/attachment.rs new file mode 100644 index 0000000..740db19 --- /dev/null +++ b/src/entities/attachment.rs @@ -0,0 +1,63 @@ +//! Module containing everything related to media attachements. + +/// A struct representing a media attachment. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Attachment { + /// ID of the attachment. + pub id: String, + /// The media type of an attachment. + #[serde(rename = "type")] + pub media_type: MediaType, + /// URL of the locally hosted version of the image. + pub url: String, + /// For remote images, the remote URL of the original image. + pub remote_url: Option, + /// URL of the preview image. + pub preview_url: String, + /// Shorter URL for the image, for insertion into text + /// (only present on local images) + pub text_url: Option, + /// Meta information about the attachment. + pub meta: Option, + /// Noop will be removed. + pub description: Option, +} + +/// Information about the attachment itself. +#[derive(Debug, Deserialize, Clone, PartialEq)] +pub struct Meta { + /// Original version. + pub original: Option, + /// Smaller version. + pub small: Option, +} + +/// Dimensions of an attachement. +#[derive(Debug, Deserialize, Clone, PartialEq)] +pub struct ImageDetails { + /// width of attachment. + width: u64, + /// height of attachment. + height: u64, + /// A string of `widthxheight`. + size: Option, + /// The aspect ratio of the attachment. + aspect: Option, +} + +/// The type of media attachment. +#[derive(Debug, Deserialize, Clone, Copy, PartialEq)] +pub enum MediaType { + /// An image. + #[serde(rename = "image")] + Image, + /// A video file. + #[serde(rename = "video")] + Video, + /// A gifv format file. + #[serde(rename = "gifv")] + Gifv, + /// Unknown format. + #[serde(rename = "unknown")] + Unknown, +} diff --git a/src/entities/card.rs b/src/entities/card.rs new file mode 100644 index 0000000..ec9471c --- /dev/null +++ b/src/entities/card.rs @@ -0,0 +1,28 @@ +//! Module representing cards of statuses. + +/// A card of a status. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Card { + /// The url associated with the card. + pub url: String, + /// The title of the card. + pub title: String, + /// The card description. + pub description: String, + /// The image associated with the card, if any. + pub image: Option, + /// OEmbed data + author_name: Option, + /// OEmbed data + author_url: Option, + /// OEmbed data + provider_name: Option, + /// OEmbed data + provider_url: Option, + /// OEmbed data + html: Option, + /// OEmbed data + width: Option, + /// OEmbed data + height: Option, +} diff --git a/src/entities/context.rs b/src/entities/context.rs new file mode 100644 index 0000000..60a4ea2 --- /dev/null +++ b/src/entities/context.rs @@ -0,0 +1,13 @@ +//! A module about contexts of statuses. + +use super::status::Status; + +/// A context of a status returning a list of statuses it replied to and +/// statuses replied to it. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Context { + /// Statuses that were replied to. + pub ancestors: Vec, + /// Statuses that replied to this status. + pub descendants: Vec, +} diff --git a/src/entities/event.rs b/src/entities/event.rs new file mode 100644 index 0000000..101c677 --- /dev/null +++ b/src/entities/event.rs @@ -0,0 +1,14 @@ +use entities::{notification::Notification, status::Status}; + +#[derive(Debug, Clone)] +/// Events that come from the /streaming/user API call +pub enum Event { + /// Update event + Update(Status), + /// Notification event + Notification(Notification), + /// Delete event + Delete(String), + /// FiltersChanged event + FiltersChanged, +} diff --git a/src/entities/filter.rs b/src/entities/filter.rs new file mode 100644 index 0000000..29dcb27 --- /dev/null +++ b/src/entities/filter.rs @@ -0,0 +1,27 @@ +/// Represents a single Filter +#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)] +pub struct Filter { + id: String, + phrase: String, + context: Vec, + expires_at: Option, // TODO: timestamp + irreversible: bool, + whole_word: bool, +} + +/// Represents the various types of Filter contexts +#[derive(Debug, Clone, Copy, PartialEq, Serialize, Deserialize)] +pub enum FilterContext { + /// Represents the "home" context + #[serde(rename = "home")] + Home, + /// Represents the "notifications" context + #[serde(rename = "notifications")] + Notifications, + /// Represents the "public" context + #[serde(rename = "public")] + Public, + /// Represents the "thread" context + #[serde(rename = "thread")] + Thread, +} diff --git a/src/entities/instance.rs b/src/entities/instance.rs new file mode 100644 index 0000000..66d60ca --- /dev/null +++ b/src/entities/instance.rs @@ -0,0 +1,45 @@ +//! Module containing everything related to an instance. +use super::account::Account; + +/// A struct containing info of an instance. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Instance { + /// URI of the current instance + pub uri: String, + /// The instance's title. + pub title: String, + /// A description for the instance. + pub description: String, + /// An email address which can be used to contact the + /// instance administrator. + pub email: String, + /// The Mastodon version used by instance. + pub version: String, + /// Urls to the streaming api. + pub urls: Option, + /// Stats about the instance. + pub stats: Option, + /// Thumbnail of the server image. + pub thumbnail: Option, + /// List of languages used on the server. + pub languages: Option>, + /// Contact account for the server. + pub contact_account: Option, + /// The maximum number of characters allowed in a status + pub max_toot_chars: Option, +} + +/// Object containing url for streaming api. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct StreamingApi { + /// Url for streaming API, typically a `wss://` url. + pub streaming_api: String, +} + +/// Statistics about the Mastodon instance. +#[derive(Debug, Clone, Copy, Deserialize, PartialEq)] +pub struct Stats { + user_count: u64, + status_count: u64, + domain_count: u64, +} diff --git a/src/entities/itemsiter.rs b/src/entities/itemsiter.rs new file mode 100644 index 0000000..e8b3e76 --- /dev/null +++ b/src/entities/itemsiter.rs @@ -0,0 +1,95 @@ +use http_send::HttpSend; +use page::Page; +use serde::Deserialize; + +/// Abstracts away the `next_page` logic into a single stream of items +/// +/// ```no_run +/// # extern crate elefren; +/// # use elefren::prelude::*; +/// # use std::error::Error; +/// # fn main() -> Result<(), Box> { +/// # let data = Data { +/// # base: "".into(), +/// # client_id: "".into(), +/// # client_secret: "".into(), +/// # redirect: "".into(), +/// # token: "".into(), +/// # }; +/// let client = Mastodon::from(data); +/// let statuses = client.statuses("user-id", None)?; +/// for status in statuses.items_iter() { +/// // do something with `status` +/// } +/// # Ok(()) +/// # } +/// ``` +#[derive(Debug, Clone)] +pub(crate) struct ItemsIter<'a, T: Clone + for<'de> Deserialize<'de>, H: 'a + HttpSend> { + page: Page<'a, T, H>, + buffer: Vec, + cur_idx: usize, + use_initial: bool, +} + +impl<'a, T: Clone + for<'de> Deserialize<'de>, H: HttpSend> ItemsIter<'a, T, H> { + pub(crate) fn new(page: Page<'a, T, H>) -> ItemsIter<'a, T, H> { + ItemsIter { + page, + buffer: vec![], + cur_idx: 0, + use_initial: true, + } + } + + fn need_next_page(&self) -> bool { + self.buffer.is_empty() || self.cur_idx == self.buffer.len() + } + + fn fill_next_page(&mut self) -> Option<()> { + let items = if let Ok(items) = self.page.next_page() { + items + } else { + return None; + }; + if let Some(items) = items { + if items.is_empty() { + return None; + } + self.buffer = items; + self.cur_idx = 0; + Some(()) + } else { + None + } + } +} + +impl<'a, T: Clone + for<'de> Deserialize<'de>, H: HttpSend> Iterator for ItemsIter<'a, T, H> { + type Item = T; + + fn next(&mut self) -> Option { + if self.use_initial { + if self.page.initial_items.is_empty() || self.cur_idx == self.page.initial_items.len() { + return None; + } + let idx = self.cur_idx; + if self.cur_idx == self.page.initial_items.len() - 1 { + self.cur_idx = 0; + self.use_initial = false; + } else { + self.cur_idx += 1; + } + Some(self.page.initial_items[idx].clone()) + } else { + if self.need_next_page() { + if self.fill_next_page().is_none() { + return None; + } + } + let idx = self.cur_idx; + self.cur_idx += 1; + Some(self.buffer[idx].clone()) + } + } +} diff --git a/src/entities/list.rs b/src/entities/list.rs new file mode 100644 index 0000000..8d640c8 --- /dev/null +++ b/src/entities/list.rs @@ -0,0 +1,6 @@ +/// Used for ser/de of list resources +#[derive(Clone, Debug, Deserialize, PartialEq)] +pub struct List { + id: String, + title: String, +} diff --git a/src/entities/mention.rs b/src/entities/mention.rs new file mode 100644 index 0000000..8c5a392 --- /dev/null +++ b/src/entities/mention.rs @@ -0,0 +1,12 @@ +/// Represents a `mention` used in a status +#[derive(Debug, Clone, PartialEq)] +pub struct Mention { + /// URL of user's profile (can be remote) + pub url: String, + /// The username of the account + pub username: String, + /// Equals username for local users, includes `@domain` for remote ones + pub acct: String, + /// Account ID + pub id: String, +} diff --git a/src/entities/mod.rs b/src/entities/mod.rs new file mode 100644 index 0000000..d5ad373 --- /dev/null +++ b/src/entities/mod.rs @@ -0,0 +1,59 @@ +/// Data structures for ser/de of account-related resources +pub mod account; +/// Data structures for ser/de of attachment-related resources +pub mod attachment; +/// Data structures for ser/de of card-related resources +pub mod card; +/// Data structures for ser/de of contetx-related resources +pub mod context; +/// Data structures for ser/de of streaming events +pub mod event; +/// Data structures for ser/de of filter-related resources +pub mod filter; +/// Data structures for ser/de of instance-related resources +pub mod instance; +pub(crate) mod itemsiter; +/// Data structures for ser/de of list-related resources +pub mod list; +/// Data structures for ser/de of mention-related resources +pub mod mention; +/// Data structures for ser/de of notification-related resources +pub mod notification; +/// Data structures for ser/de of push-subscription-related resources +pub mod push; +/// Data structures for ser/de of relationship-related resources +pub mod relationship; +/// Data structures for ser/de of report-related resources +pub mod report; +/// Data structures for ser/de of search-related resources +pub mod search_result; +/// Data structures for ser/de of status-related resources +pub mod status; + +/// An empty JSON object. +#[derive(Deserialize, Debug, Copy, Clone, PartialEq)] +pub struct Empty {} + +/// The purpose of this module is to alleviate imports of many common +/// structs by adding a glob import to the top of mastodon heavy +/// modules: +pub mod prelude { + pub use super::{ + account::{Account, Source}, + attachment::{Attachment, MediaType}, + card::Card, + context::Context, + event::Event, + filter::{Filter, FilterContext}, + instance::*, + list::List, + mention::Mention, + notification::Notification, + push::Subscription, + relationship::Relationship, + report::Report, + search_result::{SearchResult, SearchResultV2}, + status::{Application, Emoji, Status}, + Empty, + }; +} diff --git a/src/entities/notification.rs b/src/entities/notification.rs new file mode 100644 index 0000000..6ad4c37 --- /dev/null +++ b/src/entities/notification.rs @@ -0,0 +1,34 @@ +//! Module containing all info about notifications. + +use super::{account::Account, status::Status}; +use chrono::prelude::*; + +/// A struct containing info about a notification. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Notification { + /// The notification ID. + pub id: String, + /// The type of notification. + #[serde(rename = "type")] + pub notification_type: NotificationType, + /// The time the notification was created. + pub created_at: DateTime, + /// The Account sending the notification to the user. + pub account: Account, + /// The Status associated with the notification, if applicable. + pub status: Option, +} + +/// The type of notification. +#[derive(Debug, Clone, Copy, Deserialize, PartialEq)] +#[serde(rename_all = "lowercase")] +pub enum NotificationType { + /// Someone mentioned the application client in another status. + Mention, + /// Someone reblogged one of the application client's statuses. + Reblog, + /// Someone favourited one of the application client's statuses. + Favourite, + /// Someone followed the application client. + Follow, +} diff --git a/src/entities/push.rs b/src/entities/push.rs new file mode 100644 index 0000000..bcf826a --- /dev/null +++ b/src/entities/push.rs @@ -0,0 +1,67 @@ +/// Represents the `alerts` key of the `Subscription` object +#[derive(Debug, Clone, Copy, PartialEq, Serialize, Deserialize, Default)] +pub struct Alerts { + /// flag for follow alerts + pub follow: Option, + /// flag for favourite alerts + pub favourite: Option, + /// flag for reblog alerts + pub reblog: Option, + /// flag for mention alerts + pub mention: Option, +} + +/// Represents a new Push subscription +#[derive(Debug, Clone, PartialEq, Deserialize)] +pub struct Subscription { + /// The `id` of the subscription + pub id: String, + /// The endpoint of the subscription + pub endpoint: String, + /// The server key of the subscription + pub server_key: String, + /// The status of the alerts for this subscription + pub alerts: Option, +} + +pub(crate) mod add_subscription { + use super::Alerts; + + #[derive(Debug, Clone, PartialEq, Serialize, Default)] + pub(crate) struct Form { + pub(crate) subscription: Subscription, + pub(crate) data: Option, + } + + #[derive(Debug, Clone, PartialEq, Serialize, Default)] + pub(crate) struct Subscription { + pub(crate) endpoint: String, + pub(crate) keys: Keys, + } + + #[derive(Debug, Clone, PartialEq, Serialize, Default)] + pub(crate) struct Keys { + pub(crate) p256dh: String, + pub(crate) auth: String, + } + + #[derive(Debug, Clone, PartialEq, Serialize, Default)] + pub(crate) struct Data { + pub(crate) alerts: Option, + } +} + +pub(crate) mod update_data { + use super::Alerts; + + #[derive(Debug, Clone, PartialEq, Serialize, Default)] + pub(crate) struct Data { + pub(crate) alerts: Option, + } + + #[derive(Debug, Clone, PartialEq, Serialize, Default)] + pub(crate) struct Form { + pub(crate) id: String, + pub(crate) data: Data, + } +} diff --git a/src/entities/relationship.rs b/src/entities/relationship.rs new file mode 100644 index 0000000..35f3c99 --- /dev/null +++ b/src/entities/relationship.rs @@ -0,0 +1,31 @@ +//! module containing everything relating to a relationship with +//! another account. + +/// A struct containing information about a relationship with another account. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Relationship { + /// Target account id + pub id: String, + /// Whether the application client follows the account. + pub following: bool, + /// Whether the account follows the application client. + pub followed_by: bool, + /// Whether the application client blocks the account. + pub blocking: bool, + /// Whether the application client blocks the account. + pub muting: bool, + /// Whether the application client has requested to follow the account. + pub requested: bool, + /// Whether the user is also muting notifications + pub muting_notifications: bool, + /// Whether the user is currently blocking the accounts's domain + pub domain_blocking: bool, + /// Whether the user's reblogs will show up in the home timeline + pub showing_reblogs: bool, + /// Whether the user is currently endorsing the account + /// + /// This field is not techincally nullable with mastodon >= 2.5.0, but + /// making it `Option` here means we shouldn't get deser errors when + /// making calls to pleroma or mastodon<2.5.0 instances + pub endorsed: Option, +} diff --git a/src/entities/report.rs b/src/entities/report.rs new file mode 100644 index 0000000..2c3411b --- /dev/null +++ b/src/entities/report.rs @@ -0,0 +1,10 @@ +//! module containing information about a finished report of a user. + +/// A struct containing info about a report. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Report { + /// The ID of the report. + pub id: String, + /// The action taken in response to the report. + pub action_taken: String, +} diff --git a/src/entities/search_result.rs b/src/entities/search_result.rs new file mode 100644 index 0000000..d2df4c3 --- /dev/null +++ b/src/entities/search_result.rs @@ -0,0 +1,29 @@ +//! A module containing info relating to a search result. + +use super::{ + prelude::{Account, Status}, + status::Tag, +}; + +/// A struct containing results of a search. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct SearchResult { + /// An array of matched Accounts. + pub accounts: Vec, + /// An array of matched Statuses. + pub statuses: Vec, + /// An array of matched hashtags, as strings. + pub hashtags: Vec, +} + +/// A struct containing results of a search, with `Tag` objects in the +/// `hashtags` field +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct SearchResultV2 { + /// An array of matched Accounts. + pub accounts: Vec, + /// An array of matched Statuses. + pub statuses: Vec, + /// An array of matched hashtags, as `Tag` objects. + pub hashtags: Vec, +} diff --git a/src/entities/status.rs b/src/entities/status.rs new file mode 100644 index 0000000..65c6df8 --- /dev/null +++ b/src/entities/status.rs @@ -0,0 +1,107 @@ +//! Module containing all info relating to a status. + +use super::prelude::*; +use chrono::prelude::*; +use entities::card::Card; +use status_builder::Visibility; + +/// A status from the instance. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Status { + /// The ID of the status. + pub id: String, + /// A Fediverse-unique resource ID. + pub uri: String, + /// URL to the status page (can be remote) + pub url: Option, + /// The Account which posted the status. + pub account: Account, + /// The ID of the status this status is replying to, if the status is + /// a reply. + pub in_reply_to_id: Option, + /// The ID of the account this status is replying to, if the status is + /// a reply. + pub in_reply_to_account_id: Option, + /// If this status is a reblogged Status of another User. + pub reblog: Option>, + /// Body of the status; this will contain HTML + /// (remote HTML already sanitized) + pub content: String, + /// The time the status was created. + pub created_at: DateTime, + /// An array of Emoji + pub emojis: Vec, + /// The numbef or replies to this status. + pub replies_count: Option, + /// The number of reblogs for the status. + pub reblogs_count: u64, + /// The number of favourites for the status. + pub favourites_count: u64, + /// Whether the application client has reblogged the status. + pub reblogged: Option, + /// Whether the application client has favourited the status. + pub favourited: Option, + /// Whether media attachments should be hidden by default. + pub sensitive: bool, + /// If not empty, warning text that should be displayed before the actual + /// content. + pub spoiler_text: String, + /// The visibilty of the status. + pub visibility: Visibility, + /// An array of attachments. + pub media_attachments: Vec, + /// An array of mentions. + pub mentions: Vec, + /// An array of tags. + pub tags: Vec, + /// The associated card + pub card: Option, + /// Name of application used to post status. + pub application: Option, + /// The detected language for the status, if detected. + pub language: Option, + /// Whether this is the pinned status for the account that posted it. + pub pinned: Option, +} + +/// A mention of another user. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Mention { + /// URL of user's profile (can be remote). + pub url: String, + /// The username of the account. + pub username: String, + /// Equals `username` for local users, includes `@domain` for remote ones. + pub acct: String, + /// Account ID. + pub id: String, +} + +/// Struct representing an emoji within text. +#[derive(Clone, Debug, Deserialize, PartialEq)] +pub struct Emoji { + /// The shortcode of the emoji + pub shortcode: String, + /// URL to the emoji static image + pub static_url: String, + /// URL to the emoji image + pub url: String, +} + +/// Hashtags in the status. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Tag { + /// The hashtag, not including the preceding `#`. + pub name: String, + /// The URL of the hashtag. + pub url: String, +} + +/// Application details. +#[derive(Debug, Clone, Deserialize, PartialEq)] +pub struct Application { + /// Name of the application. + pub name: String, + /// Homepage URL of the application. + pub website: Option, +} diff --git a/src/errors.rs b/src/errors.rs new file mode 100644 index 0000000..f4d62b4 --- /dev/null +++ b/src/errors.rs @@ -0,0 +1,245 @@ +use std::{error, fmt, io::Error as IoError}; + +#[cfg(feature = "env")] +use envy::Error as EnvyError; +use hyper_old_types::Error as HeaderParseError; +use reqwest::{header::ToStrError as HeaderStrError, Error as HttpError, StatusCode}; +use serde_json::Error as SerdeError; +use serde_qs::Error as SerdeQsError; +use serde_urlencoded::ser::Error as UrlEncodedError; +#[cfg(feature = "toml")] +use tomlcrate::de::Error as TomlDeError; +#[cfg(feature = "toml")] +use tomlcrate::ser::Error as TomlSerError; +use url::ParseError as UrlError; +use tungstenite::error::Error as WebSocketError; + +/// Convience type over `std::result::Result` with `Error` as the error type. +pub type Result = ::std::result::Result; + +/// enum of possible errors encountered using the mastodon API. +#[derive(Debug)] +pub enum Error { + /// Error from the Mastodon API. This typically means something went + /// wrong with your authentication or data. + Api(ApiError), + /// Error deserialising to json. Typically represents a breaking change in + /// the Mastodon API + Serde(SerdeError), + /// Error serializing to url-encoded string + UrlEncoded(UrlEncodedError), + /// Error encountered in the HTTP backend while requesting a route. + Http(HttpError), + /// Wrapper around the `std::io::Error` struct. + Io(IoError), + /// Wrapper around the `url::ParseError` struct. + Url(UrlError), + /// Missing Client Id. + ClientIdRequired, + /// Missing Client Secret. + ClientSecretRequired, + /// Missing Access Token. + AccessTokenRequired, + /// Generic client error. + Client(StatusCode), + /// Generic server error. + Server(StatusCode), + /// MastodonBuilder & AppBuilder error + MissingField(&'static str), + #[cfg(feature = "toml")] + /// Error serializing to toml + TomlSer(TomlSerError), + #[cfg(feature = "toml")] + /// Error deserializing from toml + TomlDe(TomlDeError), + /// Error converting an http header to a string + HeaderStrError(HeaderStrError), + /// Error parsing the http Link header + HeaderParseError(HeaderParseError), + #[cfg(feature = "env")] + /// Error deserializing from the environment + Envy(EnvyError), + /// Error serializing to a query string + SerdeQs(SerdeQsError), + /// WebSocket error + WebSocket(WebSocketError), + /// Other errors + Other(String), +} + +impl fmt::Display for Error { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + write!(f, "{:?}", self) + } +} + +impl error::Error for Error { + fn source(&self) -> Option<&(dyn error::Error + 'static)> { + Some(match *self { + Error::Api(ref e) => e, + Error::Serde(ref e) => e, + Error::UrlEncoded(ref e) => e, + Error::Http(ref e) => e, + Error::Io(ref e) => e, + Error::Url(ref e) => e, + #[cfg(feature = "toml")] + Error::TomlSer(ref e) => e, + #[cfg(feature = "toml")] + Error::TomlDe(ref e) => e, + Error::HeaderStrError(ref e) => e, + Error::HeaderParseError(ref e) => e, + #[cfg(feature = "env")] + Error::Envy(ref e) => e, + Error::SerdeQs(ref e) => e, + Error::WebSocket(ref e) => e, + + Error::Client(..) | Error::Server(..) => { + return None + }, + Error::ClientIdRequired => return None, + Error::ClientSecretRequired => return None, + Error::AccessTokenRequired => return None, + Error::MissingField(_) => return None, + Error::Other(..) => return None, + }) + } +} + +/// Error returned from the Mastodon API. +#[derive(Clone, Debug, Deserialize)] +pub struct ApiError { + /// The type of error. + pub error: Option, + /// The description of the error. + pub error_description: Option, +} + +impl fmt::Display for ApiError { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + write!(f, "{:?}", self) + } +} + +impl error::Error for ApiError {} + +macro_rules! from { + ($($(#[$met:meta])* $typ:ident, $variant:ident,)*) => { + $( + $(#[$met])* + impl From<$typ> for Error { + fn from(from: $typ) -> Self { + use Error::*; + $variant(from) + } + } + )* + } +} + +from! { + HttpError, Http, + IoError, Io, + SerdeError, Serde, + UrlEncodedError, UrlEncoded, + UrlError, Url, + ApiError, Api, + #[cfg(feature = "toml")] TomlSerError, TomlSer, + #[cfg(feature = "toml")] TomlDeError, TomlDe, + HeaderStrError, HeaderStrError, + HeaderParseError, HeaderParseError, + #[cfg(feature = "env")] EnvyError, Envy, + SerdeQsError, SerdeQs, + WebSocketError, WebSocket, + String, Other, +} + +#[macro_export] +/// Used to easily create errors from strings +macro_rules! format_err { + ( $( $arg:tt )* ) => { + { + use elefren::Error; + Error::Other(format!($($arg)*)) + } + } +} + +#[cfg(test)] +mod tests { + use super::*; + use reqwest; + use serde_json; + use serde_urlencoded; + use std::io; + + macro_rules! assert_is { + ($err:ident, $variant:pat) => { + assert!(match $err { + $variant => true, + _ => false, + }); + }; + } + + #[test] + fn from_http_error() { + let err: HttpError = reqwest::get("not an actual URL").unwrap_err(); + let err: Error = Error::from(err); + assert_is!(err, Error::Http(..)); + } + + #[test] + fn from_io_error() { + let err: IoError = io::Error::new(io::ErrorKind::Other, "other error"); + let err: Error = Error::from(err); + assert_is!(err, Error::Io(..)); + } + + #[test] + fn from_serde_error() { + let err: SerdeError = serde_json::from_str::<()>("not valid json").unwrap_err(); + let err: Error = Error::from(err); + assert_is!(err, Error::Serde(..)); + } + + #[test] + fn from_url_encoded_error() { + let err: UrlEncodedError = serde_urlencoded::ser::Error::Custom("error".into()); + let err: Error = Error::from(err); + assert_is!(err, Error::UrlEncoded(..)); + } + + #[test] + fn from_url_error() { + let err: UrlError = UrlError::EmptyHost; + let err: Error = Error::from(err); + assert_is!(err, Error::Url(..)); + } + + #[test] + fn from_api_error() { + let err: ApiError = ApiError { + error: None, + error_description: None, + }; + let err: Error = Error::from(err); + assert_is!(err, Error::Api(..)); + } + + #[cfg(feature = "toml")] + #[test] + fn from_toml_ser_error() { + let err: TomlSerError = TomlSerError::DateInvalid; + let err: Error = Error::from(err); + assert_is!(err, Error::TomlSer(..)); + } + + #[cfg(feature = "toml")] + #[test] + fn from_toml_de_error() { + use tomlcrate; + let err: TomlDeError = tomlcrate::from_str::<()>("not valid toml").unwrap_err(); + let err: Error = Error::from(err); + assert_is!(err, Error::TomlDe(..)); + } +} diff --git a/src/helpers/cli.rs b/src/helpers/cli.rs new file mode 100644 index 0000000..a1321d5 --- /dev/null +++ b/src/helpers/cli.rs @@ -0,0 +1,27 @@ +use std::io::{self, BufRead, Write}; + +use errors::Result; +use http_send::HttpSend; +use registration::Registered; +use Mastodon; + +/// Finishes the authentication process for the given `Registered` object, +/// using the command-line +pub fn authenticate(registration: Registered) -> Result> { + let url = registration.authorize_url()?; + + let stdout = io::stdout(); + let stdin = io::stdin(); + + let mut stdout = stdout.lock(); + let mut stdin = stdin.lock(); + + writeln!(&mut stdout, "Click this link to authorize: {}", url)?; + write!(&mut stdout, "Paste the returned authorization code: ")?; + stdout.flush()?; + + let mut input = String::new(); + stdin.read_line(&mut input)?; + let code = input.trim(); + Ok(registration.complete(code)?) +} diff --git a/src/helpers/env.rs b/src/helpers/env.rs new file mode 100644 index 0000000..1f13acb --- /dev/null +++ b/src/helpers/env.rs @@ -0,0 +1,81 @@ +use envy; + +use data::Data; +use Result; + +/// Attempts to deserialize a Data struct from the environment +pub fn from_env() -> Result { + Ok(envy::from_env()?) +} + +/// Attempts to deserialize a Data struct from the environment. All keys are +/// prefixed with the given prefix +pub fn from_env_prefixed(prefix: &str) -> Result { + Ok(envy::prefixed(prefix).from_env()?) +} + +#[cfg(test)] +mod tests { + use super::*; + use std::{ + env, + ops::FnOnce, + panic::{catch_unwind, UnwindSafe}, + }; + + fn withenv R + UnwindSafe, R>(prefix: Option<&'static str>, test: F) -> R { + env::set_var(makekey(prefix, "BASE"), "https://example.com"); + env::set_var(makekey(prefix, "CLIENT_ID"), "adbc01234"); + env::set_var(makekey(prefix, "CLIENT_SECRET"), "0987dcba"); + env::set_var(makekey(prefix, "REDIRECT"), "urn:ietf:wg:oauth:2.0:oob"); + env::set_var(makekey(prefix, "TOKEN"), "fedc5678"); + + let result = catch_unwind(test); + + env::remove_var(makekey(prefix, "BASE")); + env::remove_var(makekey(prefix, "CLIENT_ID")); + env::remove_var(makekey(prefix, "CLIENT_SECRET")); + env::remove_var(makekey(prefix, "REDIRECT")); + env::remove_var(makekey(prefix, "TOKEN")); + + fn makekey(prefix: Option<&'static str>, key: &str) -> String { + if let Some(prefix) = prefix { + format!("{}{}", prefix, key) + } else { + key.to_string() + } + } + + result.expect("failed") + } + + #[test] + fn test_from_env_no_prefix() { + let desered = withenv(None, || from_env()).expect("Couldn't deser"); + assert_eq!( + desered, + Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + } + ); + } + + #[test] + fn test_from_env_prefixed() { + let desered = withenv(Some("APP_"), || from_env_prefixed("APP_")).expect("Couldn't deser"); + assert_eq!( + desered, + Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + } + ); + } +} diff --git a/src/helpers/json.rs b/src/helpers/json.rs new file mode 100644 index 0000000..ab69890 --- /dev/null +++ b/src/helpers/json.rs @@ -0,0 +1,231 @@ +use std::{ + fs::{File, OpenOptions}, + io::{BufWriter, Read, Write}, + path::Path, +}; + +use serde_json; + +use data::Data; +use Result; + +/// Attempts to deserialize a Data struct from a string +pub fn from_str(s: &str) -> Result { + Ok(serde_json::from_str(s)?) +} + +/// Attempts to deserialize a Data struct from a slice of bytes +pub fn from_slice(s: &[u8]) -> Result { + Ok(serde_json::from_slice(s)?) +} + +/// Attempts to deserialize a Data struct from something that implements +/// the std::io::Read trait +pub fn from_reader(mut r: R) -> Result { + let mut buffer = Vec::new(); + r.read_to_end(&mut buffer)?; + from_slice(&buffer) +} + +/// Attempts to deserialize a Data struct from a file +pub fn from_file>(path: P) -> Result { + let path = path.as_ref(); + let file = File::open(path)?; + Ok(from_reader(file)?) +} + +/// Attempts to serialize a Data struct to a String +pub fn to_string(data: &Data) -> Result { + Ok(serde_json::to_string_pretty(data)?) +} + +/// Attempts to serialize a Data struct to a Vec of bytes +pub fn to_vec(data: &Data) -> Result> { + Ok(serde_json::to_vec(data)?) +} + +/// Attempts to serialize a Data struct to something that implements the +/// std::io::Write trait +pub fn to_writer(data: &Data, writer: W) -> Result<()> { + let mut buf_writer = BufWriter::new(writer); + let vec = to_vec(data)?; + buf_writer.write(&vec)?; + Ok(()) +} + +/// Attempts to serialize a Data struct to a file +/// +/// When opening the file, this will set the `.write(true)` and +/// `.truncate(true)` options, use the next method for more +/// fine-grained control +pub fn to_file>(data: &Data, path: P) -> Result<()> { + let mut options = OpenOptions::new(); + options.create(true).write(true).truncate(true); + to_file_with_options(data, path, options)?; + Ok(()) +} + +/// Attempts to serialize a Data struct to a file +pub fn to_file_with_options>( + data: &Data, + path: P, + options: OpenOptions, +) -> Result<()> { + let path = path.as_ref(); + let file = options.open(path)?; + to_writer(data, file)?; + Ok(()) +} + +#[cfg(test)] +mod tests { + use super::*; + use std::{fs::OpenOptions, io::Cursor}; + use tempfile::{tempdir, NamedTempFile}; + + const DOC: &'static str = indoc!( + r#" + { + "base": "https://example.com", + "client_id": "adbc01234", + "client_secret": "0987dcba", + "redirect": "urn:ietf:wg:oauth:2.0:oob", + "token": "fedc5678" + } + "# + ); + + #[test] + fn test_from_str() { + let desered = from_str(DOC).expect("Couldn't deserialize Data"); + assert_eq!( + desered, + Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + } + ); + } + #[test] + fn test_from_slice() { + let doc = DOC.as_bytes(); + let desered = from_slice(&doc).expect("Couldn't deserialize Data"); + assert_eq!( + desered, + Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + } + ); + } + #[test] + fn test_from_reader() { + let doc = DOC.as_bytes(); + let doc = Cursor::new(doc); + let desered = from_reader(doc).expect("Couldn't deserialize Data"); + assert_eq!( + desered, + Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + } + ); + } + #[test] + fn test_from_file() { + let mut datafile = NamedTempFile::new().expect("Couldn't create tempfile"); + write!(&mut datafile, "{}", DOC).expect("Couldn't write Data to file"); + let desered = from_file(datafile.path()).expect("Couldn't deserialize Data"); + assert_eq!( + desered, + Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + } + ); + } + #[test] + fn test_to_string() { + let data = Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + }; + let s = to_string(&data).expect("Couldn't serialize Data"); + let desered = from_str(&s).expect("Couldn't deserialize Data"); + assert_eq!(data, desered); + } + #[test] + fn test_to_vec() { + let data = Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + }; + let v = to_vec(&data).expect("Couldn't write to vec"); + let desered = from_slice(&v).expect("Couldn't deserialize data"); + assert_eq!(data, desered); + } + #[test] + fn test_to_writer() { + let data = Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + }; + let mut buffer = Vec::new(); + to_writer(&data, &mut buffer).expect("Couldn't write to writer"); + let reader = Cursor::new(buffer); + let desered = from_reader(reader).expect("Couldn't deserialize Data"); + assert_eq!(data, desered); + } + #[test] + fn test_to_file() { + let data = Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + }; + let tempdir = tempdir().expect("Couldn't create tempdir"); + let filename = tempdir.path().join("mastodon-data.json"); + to_file(&data, &filename).expect("Couldn't write to file"); + let desered = from_file(&filename).expect("Couldn't deserialize Data"); + assert_eq!(data, desered); + } + #[test] + fn test_to_file_with_options() { + let data = Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + }; + let file = NamedTempFile::new().expect("Couldn't create tempfile"); + let mut options = OpenOptions::new(); + options.write(true).create(false).truncate(true); + to_file_with_options(&data, file.path(), options).expect("Couldn't write to file"); + let desered = from_file(file.path()).expect("Couldn't deserialize Data"); + assert_eq!(data, desered); + } +} diff --git a/src/helpers/mod.rs b/src/helpers/mod.rs new file mode 100644 index 0000000..57357b7 --- /dev/null +++ b/src/helpers/mod.rs @@ -0,0 +1,38 @@ +#[cfg(feature = "toml")] +/// Helpers for serializing to/deserializing from toml +/// +/// In order to use this module, set the "toml" feature in your Cargo.toml: +/// +/// ```toml,ignore +/// [dependencies.elefren] +/// version = "0.22" +/// features = ["toml"] +/// ``` +pub mod toml; + +#[cfg(feature = "json")] +/// Helpers for serializing to/deserializing from json +/// +/// In order to use this module, set the "json" feature in your Cargo.toml: +/// +/// ```toml,ignore +/// [dependencies.elefen] +/// version = "0.22" +/// features = ["json"] +/// ``` +pub mod json; + +#[cfg(feature = "env")] +/// Helpers for deserializing a `Data` struct from the environment +/// +/// In order to use this module, set the "env" feature in your Cargo.toml: +/// +/// ```toml,ignore +/// [dependencies.elefren] +/// version = "0.22" +/// features = ["env"] +/// ``` +pub mod env; + +/// Helpers for working with the command line +pub mod cli; diff --git a/src/helpers/toml.rs b/src/helpers/toml.rs new file mode 100644 index 0000000..be17783 --- /dev/null +++ b/src/helpers/toml.rs @@ -0,0 +1,229 @@ +use std::{ + fs::{File, OpenOptions}, + io::{BufWriter, Read, Write}, + path::Path, +}; + +use tomlcrate; + +use data::Data; +use Result; + +/// Attempts to deserialize a Data struct from a string +pub fn from_str(s: &str) -> Result { + Ok(tomlcrate::from_str(s)?) +} + +/// Attempts to deserialize a Data struct from a slice of bytes +pub fn from_slice(s: &[u8]) -> Result { + Ok(tomlcrate::from_slice(s)?) +} + +/// Attempts to deserialize a Data struct from something that implements +/// the std::io::Read trait +pub fn from_reader(mut r: R) -> Result { + let mut buffer = Vec::new(); + r.read_to_end(&mut buffer)?; + from_slice(&buffer) +} + +/// Attempts to deserialize a Data struct from a file +pub fn from_file>(path: P) -> Result { + let path = path.as_ref(); + let file = File::open(path)?; + Ok(from_reader(file)?) +} + +/// Attempts to serialize a Data struct to a String +pub fn to_string(data: &Data) -> Result { + Ok(tomlcrate::to_string_pretty(data)?) +} + +/// Attempts to serialize a Data struct to a Vec of bytes +pub fn to_vec(data: &Data) -> Result> { + Ok(tomlcrate::to_vec(data)?) +} + +/// Attempts to serialize a Data struct to something that implements the +/// std::io::Write trait +pub fn to_writer(data: &Data, writer: W) -> Result<()> { + let mut buf_writer = BufWriter::new(writer); + let vec = to_vec(data)?; + buf_writer.write(&vec)?; + Ok(()) +} + +/// Attempts to serialize a Data struct to a file +/// +/// When opening the file, this will set the `.write(true)` and +/// `.truncate(true)` options, use the next method for more +/// fine-grained control +pub fn to_file>(data: &Data, path: P) -> Result<()> { + let mut options = OpenOptions::new(); + options.create(true).write(true).truncate(true); + to_file_with_options(data, path, options)?; + Ok(()) +} + +/// Attempts to serialize a Data struct to a file +pub fn to_file_with_options>( + data: &Data, + path: P, + options: OpenOptions, +) -> Result<()> { + let path = path.as_ref(); + let file = options.open(path)?; + to_writer(data, file)?; + Ok(()) +} + +#[cfg(test)] +mod tests { + use super::*; + use std::{fs::OpenOptions, io::Cursor}; + use tempfile::{tempdir, NamedTempFile}; + + const DOC: &'static str = indoc!( + r#" + base = "https://example.com" + client_id = "adbc01234" + client_secret = "0987dcba" + redirect = "urn:ietf:wg:oauth:2.0:oob" + token = "fedc5678" + "# + ); + + #[test] + fn test_from_str() { + let desered = from_str(DOC).expect("Couldn't deserialize Data"); + assert_eq!( + desered, + Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + } + ); + } + #[test] + fn test_from_slice() { + let doc = DOC.as_bytes(); + let desered = from_slice(&doc).expect("Couldn't deserialize Data"); + assert_eq!( + desered, + Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + } + ); + } + #[test] + fn test_from_reader() { + let doc = DOC.as_bytes(); + let doc = Cursor::new(doc); + let desered = from_reader(doc).expect("Couldn't deserialize Data"); + assert_eq!( + desered, + Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + } + ); + } + #[test] + fn test_from_file() { + let mut datafile = NamedTempFile::new().expect("Couldn't create tempfile"); + write!(&mut datafile, "{}", DOC).expect("Couldn't write Data to file"); + let desered = from_file(datafile.path()).expect("Couldn't deserialize Data"); + assert_eq!( + desered, + Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + } + ); + } + #[test] + fn test_to_string() { + let data = Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + }; + let s = to_string(&data).expect("Couldn't serialize Data"); + let desered = from_str(&s).expect("Couldn't deserialize Data"); + assert_eq!(data, desered); + } + #[test] + fn test_to_vec() { + let data = Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + }; + let v = to_vec(&data).expect("Couldn't write to vec"); + let desered = from_slice(&v).expect("Couldn't deserialize data"); + assert_eq!(data, desered); + } + #[test] + fn test_to_writer() { + let data = Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + }; + let mut buffer = Vec::new(); + to_writer(&data, &mut buffer).expect("Couldn't write to writer"); + let reader = Cursor::new(buffer); + let desered = from_reader(reader).expect("Couldn't deserialize Data"); + assert_eq!(data, desered); + } + #[test] + fn test_to_file() { + let data = Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + }; + let tempdir = tempdir().expect("Couldn't create tempdir"); + let filename = tempdir.path().join("mastodon-data.toml"); + to_file(&data, &filename).expect("Couldn't write to file"); + let desered = from_file(&filename).expect("Couldn't deserialize Data"); + assert_eq!(data, desered); + } + #[test] + fn test_to_file_with_options() { + let data = Data { + base: "https://example.com".into(), + client_id: "adbc01234".into(), + client_secret: "0987dcba".into(), + redirect: "urn:ietf:wg:oauth:2.0:oob".into(), + token: "fedc5678".into(), + }; + let file = NamedTempFile::new().expect("Couldn't create tempfile"); + let mut options = OpenOptions::new(); + options.write(true).create(false).truncate(true); + to_file_with_options(&data, file.path(), options).expect("Couldn't write to file"); + let desered = from_file(file.path()).expect("Couldn't deserialize Data"); + assert_eq!(data, desered); + } +} diff --git a/src/http_send.rs b/src/http_send.rs new file mode 100644 index 0000000..9c161f8 --- /dev/null +++ b/src/http_send.rs @@ -0,0 +1,26 @@ +use reqwest::{Client, Request, RequestBuilder, Response}; +use std::fmt::Debug; +use Result; + +/// Abstracts away the process of turning an HTTP request into an HTTP response +pub trait HttpSend: Clone + Debug { + /// Converts an HTTP request into an HTTP response + fn execute(&self, client: &Client, request: Request) -> Result; + + /// Convenience method so that .build() doesn't have to be called at every + /// call site + fn send(&self, client: &Client, builder: RequestBuilder) -> Result { + let request = builder.build()?; + self.execute(client, request) + } +} + +#[doc(hidden)] +#[derive(Clone, Copy, Debug, PartialEq)] +pub struct HttpSender; + +impl HttpSend for HttpSender { + fn execute(&self, client: &Client, request: Request) -> Result { + Ok(client.execute(request)?) + } +} diff --git a/src/lib.rs b/src/lib.rs new file mode 100644 index 0000000..0f3d9d3 --- /dev/null +++ b/src/lib.rs @@ -0,0 +1,858 @@ +//! # Elefren: API Wrapper around the Mastodon API. +//! +//! Most of the api is documented on [Mastodon's website](https://docs.joinmastodon.org/client/intro/) +//! +//! ```no_run +//! # extern crate elefren; +//! # fn main() { +//! # try().unwrap(); +//! # } +//! # fn try() -> elefren::Result<()> { +//! use elefren::{helpers::cli, prelude::*}; +//! +//! let registration = Registration::new("https://mastodon.social") +//! .client_name("elefren_test") +//! .build()?; +//! let mastodon = cli::authenticate(registration)?; +//! +//! println!( +//! "{:?}", +//! mastodon +//! .get_home_timeline()? +//! .items_iter() +//! .take(100) +//! .collect::>() +//! ); +//! # Ok(()) +//! # } +//! ``` +//! +//! Elefren also supports Mastodon's Streaming API: +//! +//! # Example +//! +//! ```no_run +//! # extern crate elefren; +//! # use elefren::prelude::*; +//! # use std::error::Error; +//! use elefren::entities::event::Event; +//! # fn main() -> Result<(), Box> { +//! # let data = Data { +//! # base: "".into(), +//! # client_id: "".into(), +//! # client_secret: "".into(), +//! # redirect: "".into(), +//! # token: "".into(), +//! # }; +//! let client = Mastodon::from(data); +//! for event in client.streaming_user()? { +//! match event { +//! Event::Update(ref status) => { /* .. */ }, +//! Event::Notification(ref notification) => { /* .. */ }, +//! Event::Delete(ref id) => { /* .. */ }, +//! Event::FiltersChanged => { /* .. */ }, +//! } +//! } +//! # Ok(()) +//! # } +//! ``` + +#![deny( + missing_docs, + warnings, + missing_debug_implementations, + missing_copy_implementations, + trivial_casts, + trivial_numeric_casts, + unsafe_code, + unstable_features, + unused_import_braces, + unused_qualifications +)] + +#[macro_use] +extern crate log; +#[macro_use] +extern crate serde_derive; +#[macro_use] +extern crate doc_comment; +extern crate hyper_old_types; +extern crate isolang; +#[macro_use] +extern crate serde_json; +extern crate chrono; +extern crate reqwest; +extern crate serde; +extern crate serde_qs; +extern crate serde_urlencoded; +extern crate tap_reader; +extern crate try_from; +extern crate tungstenite; +extern crate url; + +#[cfg(feature = "env")] +extern crate envy; + +#[cfg(feature = "toml")] +extern crate toml as tomlcrate; + +#[cfg(test)] +extern crate tempfile; + +#[cfg(test)] +#[cfg_attr(all(test, any(feature = "toml", feature = "json")), macro_use)] +extern crate indoc; + +use std::{borrow::Cow, io::BufRead, ops}; + +use reqwest::{Client, RequestBuilder, Response}; +use tap_reader::Tap; +use tungstenite::client::AutoStream; + +use entities::prelude::*; +use http_send::{HttpSend, HttpSender}; +use page::Page; + +pub use data::Data; +pub use errors::{ApiError, Error, Result}; +pub use isolang::Language; +pub use mastodon_client::{MastodonClient, MastodonUnauthenticated}; +pub use registration::Registration; +pub use requests::{ + AddFilterRequest, + AddPushRequest, + StatusesRequest, + UpdateCredsRequest, + UpdatePushRequest, +}; +pub use status_builder::{NewStatus, StatusBuilder}; + +/// Registering your App +pub mod apps; +/// Contains the struct that holds the client auth data +pub mod data; +/// Entities returned from the API +pub mod entities; +/// Errors +pub mod errors; +/// Collection of helpers for serializing/deserializing `Data` objects +pub mod helpers; +/// Contains trait for converting `reqwest::Request`s to `reqwest::Response`s +pub mod http_send; +mod mastodon_client; +/// Handling multiple pages of entities. +pub mod page; +/// Registering your app. +pub mod registration; +/// Requests +pub mod requests; +/// OAuth Scopes +pub mod scopes; +/// Constructing a status +pub mod status_builder; +#[macro_use] +mod macros; +/// Automatically import the things you need +pub mod prelude { + pub use scopes::Scopes; + pub use Data; + pub use Mastodon; + pub use MastodonClient; + pub use NewStatus; + pub use Registration; + pub use StatusBuilder; + pub use StatusesRequest; +} + +/// Your mastodon application client, handles all requests to and from Mastodon. +#[derive(Clone, Debug)] +pub struct Mastodon { + client: Client, + http_sender: H, + /// Raw data about your mastodon instance. + pub data: Data, +} + +impl Mastodon { + methods![get, post, delete,]; + + fn route(&self, url: &str) -> String { + format!("{}{}", self.base, url) + } + + pub(crate) fn send(&self, req: RequestBuilder) -> Result { + Ok(self + .http_sender + .send(&self.client, req.bearer_auth(&self.token))?) + } +} + +impl From for Mastodon { + /// Creates a mastodon instance from the data struct. + fn from(data: Data) -> Mastodon { + let mut builder = MastodonBuilder::new(HttpSender); + builder.data(data); + builder + .build() + .expect("We know `data` is present, so this should be fine") + } +} + +impl MastodonClient for Mastodon { + type Stream = EventReader; + + paged_routes! { + (get) favourites: "favourites" => Status, + (get) blocks: "blocks" => Account, + (get) domain_blocks: "domain_blocks" => String, + (get) follow_requests: "follow_requests" => Account, + (get) get_home_timeline: "timelines/home" => Status, + (get) get_emojis: "custom_emojis" => Emoji, + (get) mutes: "mutes" => Account, + (get) notifications: "notifications" => Notification, + (get) reports: "reports" => Report, + (get (q: &'a str, #[serde(skip_serializing_if = "Option::is_none")] limit: Option, following: bool,)) search_accounts: "accounts/search" => Account, + (get) get_endorsements: "endorsements" => Account, + } + + paged_routes_with_id! { + (get) followers: "accounts/{}/followers" => Account, + (get) following: "accounts/{}/following" => Account, + (get) reblogged_by: "statuses/{}/reblogged_by" => Account, + (get) favourited_by: "statuses/{}/favourited_by" => Account, + } + + route! { + (delete (domain: String,)) unblock_domain: "domain_blocks" => Empty, + (get) instance: "instance" => Instance, + (get) verify_credentials: "accounts/verify_credentials" => Account, + (post (account_id: &str, status_ids: Vec<&str>, comment: String,)) report: "reports" => Report, + (post (domain: String,)) block_domain: "domain_blocks" => Empty, + (post (id: &str,)) authorize_follow_request: "accounts/follow_requests/authorize" => Empty, + (post (id: &str,)) reject_follow_request: "accounts/follow_requests/reject" => Empty, + (get (q: &'a str, resolve: bool,)) search: "search" => SearchResult, + (get (local: bool,)) get_public_timeline: "timelines/public" => Vec, + (post (uri: Cow<'static, str>,)) follows: "follows" => Account, + (post multipart (file: Cow<'static, str>,)) media: "media" => Attachment, + (post) clear_notifications: "notifications/clear" => Empty, + (post (id: &str,)) dismiss_notification: "notifications/dismiss" => Empty, + (get) get_push_subscription: "push/subscription" => Subscription, + (delete) delete_push_subscription: "push/subscription" => Empty, + (get) get_filters: "filters" => Vec, + (get) get_follow_suggestions: "suggestions" => Vec, + } + + route_v2! { + (get (q: &'a str, resolve: bool,)) search_v2: "search" => SearchResultV2, + } + + route_id! { + (get) get_account: "accounts/{}" => Account, + (post) follow: "accounts/{}/follow" => Relationship, + (post) unfollow: "accounts/{}/unfollow" => Relationship, + (post) block: "accounts/{}/block" => Relationship, + (post) unblock: "accounts/{}/unblock" => Relationship, + (get) mute: "accounts/{}/mute" => Relationship, + (get) unmute: "accounts/{}/unmute" => Relationship, + (get) get_notification: "notifications/{}" => Notification, + (get) get_status: "statuses/{}" => Status, + (get) get_context: "statuses/{}/context" => Context, + (get) get_card: "statuses/{}/card" => Card, + (post) reblog: "statuses/{}/reblog" => Status, + (post) unreblog: "statuses/{}/unreblog" => Status, + (post) favourite: "statuses/{}/favourite" => Status, + (post) unfavourite: "statuses/{}/unfavourite" => Status, + (delete) delete_status: "statuses/{}" => Empty, + (get) get_filter: "filters/{}" => Filter, + (delete) delete_filter: "filters/{}" => Empty, + (delete) delete_from_suggestions: "suggestions/{}" => Empty, + (post) endorse_user: "accounts/{}/pin" => Relationship, + (post) unendorse_user: "accounts/{}/unpin" => Relationship, + } + + fn add_filter(&self, request: &mut AddFilterRequest) -> Result { + let url = self.route("/api/v1/filters"); + let response = self.send(self.client.post(&url).json(&request))?; + + let status = response.status(); + + if status.is_client_error() { + return Err(Error::Client(status.clone())); + } else if status.is_server_error() { + return Err(Error::Server(status.clone())); + } + + deserialise(response) + } + + /// PUT /api/v1/filters/:id + fn update_filter(&self, id: &str, request: &mut AddFilterRequest) -> Result { + let url = self.route(&format!("/api/v1/filters/{}", id)); + let response = self.send(self.client.put(&url).json(&request))?; + + let status = response.status(); + + if status.is_client_error() { + return Err(Error::Client(status.clone())); + } else if status.is_server_error() { + return Err(Error::Server(status.clone())); + } + + deserialise(response) + } + + fn update_credentials(&self, builder: &mut UpdateCredsRequest) -> Result { + let changes = builder.build()?; + let url = self.route("/api/v1/accounts/update_credentials"); + let response = self.send(self.client.patch(&url).json(&changes))?; + + let status = response.status(); + + if status.is_client_error() { + return Err(Error::Client(status.clone())); + } else if status.is_server_error() { + return Err(Error::Server(status.clone())); + } + + deserialise(response) + } + + /// Post a new status to the account. + fn new_status(&self, status: NewStatus) -> Result { + let response = self.send( + self.client + .post(&self.route("/api/v1/statuses")) + .json(&status), + )?; + + deserialise(response) + } + + /// Get timeline filtered by a hashtag(eg. `#coffee`) either locally or + /// federated. + fn get_tagged_timeline(&self, hashtag: String, local: bool) -> Result> { + let base = "/api/v1/timelines/tag/"; + let url = if local { + self.route(&format!("{}{}?local=1", base, hashtag)) + } else { + self.route(&format!("{}{}", base, hashtag)) + }; + + self.get(url) + } + + /// Get statuses of a single account by id. Optionally only with pictures + /// and or excluding replies. + /// + /// # Example + /// + /// ```no_run + /// # extern crate elefren; + /// # use elefren::prelude::*; + /// # use std::error::Error; + /// # fn main() -> Result<(), Box> { + /// # let data = Data { + /// # base: "".into(), + /// # client_id: "".into(), + /// # client_secret: "".into(), + /// # redirect: "".into(), + /// # token: "".into(), + /// # }; + /// let client = Mastodon::from(data); + /// let statuses = client.statuses("user-id", None)?; + /// # Ok(()) + /// # } + /// ``` + /// + /// ```no_run + /// # extern crate elefren; + /// # use elefren::prelude::*; + /// # use std::error::Error; + /// # fn main() -> Result<(), Box> { + /// # let data = Data { + /// # base: "".into(), + /// # client_id: "".into(), + /// # client_secret: "".into(), + /// # redirect: "".into(), + /// # token: "".into(), + /// # }; + /// let client = Mastodon::from(data); + /// let mut request = StatusesRequest::new(); + /// request.only_media(); + /// let statuses = client.statuses("user-id", request)?; + /// # Ok(()) + /// # } + /// ``` + fn statuses<'a, 'b: 'a, S>(&'b self, id: &'b str, request: S) -> Result> + where + S: Into>>, + { + let mut url = format!("{}/api/v1/accounts/{}/statuses", self.base, id); + + if let Some(request) = request.into() { + url = format!("{}{}", url, request.to_querystring()?); + } + + let response = self.send(self.client.get(&url))?; + + Page::new(self, response) + } + + /// Returns the client account's relationship to a list of other accounts. + /// Such as whether they follow them or vice versa. + fn relationships(&self, ids: &[&str]) -> Result> { + let mut url = self.route("/api/v1/accounts/relationships?"); + + if ids.len() == 1 { + url += "id="; + url += &ids[0]; + } else { + for id in ids { + url += "id[]="; + url += &id; + url += "&"; + } + url.pop(); + } + + let response = self.send(self.client.get(&url))?; + + Page::new(self, response) + } + + /// Add a push notifications subscription + fn add_push_subscription(&self, request: &AddPushRequest) -> Result { + let request = request.build()?; + let response = self.send( + self.client + .post(&self.route("/api/v1/push/subscription")) + .json(&request), + )?; + + deserialise(response) + } + + /// Update the `data` portion of the push subscription associated with this + /// access token + fn update_push_data(&self, request: &UpdatePushRequest) -> Result { + let request = request.build(); + let response = self.send( + self.client + .put(&self.route("/api/v1/push/subscription")) + .json(&request), + )?; + + deserialise(response) + } + + /// Get all accounts that follow the authenticated user + fn follows_me(&self) -> Result> { + let me = self.verify_credentials()?; + Ok(self.followers(&me.id)?) + } + + /// Get all accounts that the authenticated user follows + fn followed_by_me(&self) -> Result> { + let me = self.verify_credentials()?; + Ok(self.following(&me.id)?) + } + + /// returns events that are relevant to the authorized user, i.e. home + /// timeline & notifications + /// + /// # Example + /// + /// ```no_run + /// # extern crate elefren; + /// # use elefren::prelude::*; + /// # use std::error::Error; + /// use elefren::entities::event::Event; + /// # fn main() -> Result<(), Box> { + /// # let data = Data { + /// # base: "".into(), + /// # client_id: "".into(), + /// # client_secret: "".into(), + /// # redirect: "".into(), + /// # token: "".into(), + /// # }; + /// let client = Mastodon::from(data); + /// for event in client.streaming_user()? { + /// match event { + /// Event::Update(ref status) => { /* .. */ }, + /// Event::Notification(ref notification) => { /* .. */ }, + /// Event::Delete(ref id) => { /* .. */ }, + /// Event::FiltersChanged => { /* .. */ }, + /// } + /// } + /// # Ok(()) + /// # } + /// ``` + fn streaming_user(&self) -> Result { + let mut url: url::Url = self.route("/api/v1/streaming").parse()?; + url.query_pairs_mut() + .append_pair("access_token", &self.token) + .append_pair("stream", "user"); + let mut url: url::Url = reqwest::get(url.as_str())?.url().as_str().parse()?; + let new_scheme = match url.scheme() { + "http" => "ws", + "https" => "wss", + x => return Err(Error::Other(format!("Bad URL scheme: {}", x))), + }; + url.set_scheme(new_scheme) + .map_err(|_| Error::Other("Bad URL scheme!".to_string()))?; + + let client = tungstenite::connect(url.as_str())?.0; + + Ok(EventReader(WebSocket(client))) + } + + /// returns all public statuses + fn streaming_public(&self) -> Result { + let mut url: url::Url = self.route("/api/v1/streaming").parse()?; + url.query_pairs_mut() + .append_pair("access_token", &self.token) + .append_pair("stream", "public"); + let mut url: url::Url = reqwest::get(url.as_str())?.url().as_str().parse()?; + let new_scheme = match url.scheme() { + "http" => "ws", + "https" => "wss", + x => return Err(Error::Other(format!("Bad URL scheme: {}", x))), + }; + url.set_scheme(new_scheme) + .map_err(|_| Error::Other("Bad URL scheme!".to_string()))?; + + let client = tungstenite::connect(url.as_str())?.0; + + Ok(EventReader(WebSocket(client))) + } + + /// Returns all local statuses + fn streaming_local(&self) -> Result { + let mut url: url::Url = self.route("/api/v1/streaming").parse()?; + url.query_pairs_mut() + .append_pair("access_token", &self.token) + .append_pair("stream", "public:local"); + let mut url: url::Url = reqwest::get(url.as_str())?.url().as_str().parse()?; + let new_scheme = match url.scheme() { + "http" => "ws", + "https" => "wss", + x => return Err(Error::Other(format!("Bad URL scheme: {}", x))), + }; + url.set_scheme(new_scheme) + .map_err(|_| Error::Other("Bad URL scheme!".to_string()))?; + + let client = tungstenite::connect(url.as_str())?.0; + + Ok(EventReader(WebSocket(client))) + } + + /// Returns all public statuses for a particular hashtag + fn streaming_public_hashtag(&self, hashtag: &str) -> Result { + let mut url: url::Url = self.route("/api/v1/streaming").parse()?; + url.query_pairs_mut() + .append_pair("access_token", &self.token) + .append_pair("stream", "hashtag") + .append_pair("tag", hashtag); + let mut url: url::Url = reqwest::get(url.as_str())?.url().as_str().parse()?; + let new_scheme = match url.scheme() { + "http" => "ws", + "https" => "wss", + x => return Err(Error::Other(format!("Bad URL scheme: {}", x))), + }; + url.set_scheme(new_scheme) + .map_err(|_| Error::Other("Bad URL scheme!".to_string()))?; + + let client = tungstenite::connect(url.as_str())?.0; + + Ok(EventReader(WebSocket(client))) + } + + /// Returns all local statuses for a particular hashtag + fn streaming_local_hashtag(&self, hashtag: &str) -> Result { + let mut url: url::Url = self.route("/api/v1/streaming").parse()?; + url.query_pairs_mut() + .append_pair("access_token", &self.token) + .append_pair("stream", "hashtag:local") + .append_pair("tag", hashtag); + let mut url: url::Url = reqwest::get(url.as_str())?.url().as_str().parse()?; + let new_scheme = match url.scheme() { + "http" => "ws", + "https" => "wss", + x => return Err(Error::Other(format!("Bad URL scheme: {}", x))), + }; + url.set_scheme(new_scheme) + .map_err(|_| Error::Other("Bad URL scheme!".to_string()))?; + + let client = tungstenite::connect(url.as_str())?.0; + + Ok(EventReader(WebSocket(client))) + } + + /// Returns statuses for a list + fn streaming_list(&self, list_id: &str) -> Result { + let mut url: url::Url = self.route("/api/v1/streaming").parse()?; + url.query_pairs_mut() + .append_pair("access_token", &self.token) + .append_pair("stream", "list") + .append_pair("list", list_id); + let mut url: url::Url = reqwest::get(url.as_str())?.url().as_str().parse()?; + let new_scheme = match url.scheme() { + "http" => "ws", + "https" => "wss", + x => return Err(Error::Other(format!("Bad URL scheme: {}", x))), + }; + url.set_scheme(new_scheme) + .map_err(|_| Error::Other("Bad URL scheme!".to_string()))?; + + let client = tungstenite::connect(url.as_str())?.0; + + Ok(EventReader(WebSocket(client))) + } + + /// Returns all direct messages + fn streaming_direct(&self) -> Result { + let mut url: url::Url = self.route("/api/v1/streaming").parse()?; + url.query_pairs_mut() + .append_pair("access_token", &self.token) + .append_pair("stream", "direct"); + let mut url: url::Url = reqwest::get(url.as_str())?.url().as_str().parse()?; + let new_scheme = match url.scheme() { + "http" => "ws", + "https" => "wss", + x => return Err(Error::Other(format!("Bad URL scheme: {}", x))), + }; + url.set_scheme(new_scheme) + .map_err(|_| Error::Other("Bad URL scheme!".to_string()))?; + + let client = tungstenite::connect(url.as_str())?.0; + + Ok(EventReader(WebSocket(client))) + } +} + +#[derive(Debug)] +/// WebSocket newtype so that EventStream can be implemented without coherency +/// issues +pub struct WebSocket(tungstenite::protocol::WebSocket); + +/// A type that streaming events can be read from +pub trait EventStream { + /// Read a message from this stream + fn read_message(&mut self) -> Result; +} + +impl EventStream for R { + fn read_message(&mut self) -> Result { + let mut buf = String::new(); + self.read_line(&mut buf)?; + Ok(buf) + } +} + +impl EventStream for WebSocket { + fn read_message(&mut self) -> Result { + Ok(self.0.read_message()?.into_text()?) + } +} + +#[derive(Debug)] +/// Iterator that produces events from a mastodon streaming API event stream +pub struct EventReader(R); +impl Iterator for EventReader { + type Item = Event; + + fn next(&mut self) -> Option { + let mut lines = Vec::new(); + loop { + if let Ok(line) = self.0.read_message() { + let line = line.trim().to_string(); + if line.starts_with(":") || line.is_empty() { + continue; + } + lines.push(line); + if let Ok(event) = self.make_event(&lines) { + lines.clear(); + return Some(event); + } else { + continue; + } + } + } + } +} + +impl EventReader { + fn make_event(&self, lines: &[String]) -> Result { + let event; + let data; + if let Some(event_line) = lines.iter().find(|line| line.starts_with("event:")) { + event = event_line[6..].trim().to_string(); + data = lines + .iter() + .find(|line| line.starts_with("data:")) + .map(|x| x[5..].trim().to_string()); + } else { + #[derive(Deserialize)] + struct Message { + pub event: String, + pub payload: Option, + } + let message = serde_json::from_str::(&lines[0])?; + event = message.event; + data = message.payload; + } + let event: &str = &event; + Ok(match event { + "notification" => { + let data = data.ok_or_else(|| { + Error::Other("Missing `data` line for notification".to_string()) + })?; + let notification = serde_json::from_str::(&data)?; + Event::Notification(notification) + }, + "update" => { + let data = + data.ok_or_else(|| Error::Other("Missing `data` line for update".to_string()))?; + let status = serde_json::from_str::(&data)?; + Event::Update(status) + }, + "delete" => { + let data = + data.ok_or_else(|| Error::Other("Missing `data` line for delete".to_string()))?; + Event::Delete(data) + }, + "filters_changed" => Event::FiltersChanged, + _ => return Err(Error::Other(format!("Unknown event `{}`", event))), + }) + } +} + +impl ops::Deref for Mastodon { + type Target = Data; + + fn deref(&self) -> &Self::Target { + &self.data + } +} + +struct MastodonBuilder { + client: Option, + http_sender: H, + data: Option, +} + +impl MastodonBuilder { + pub fn new(sender: H) -> Self { + MastodonBuilder { + http_sender: sender, + client: None, + data: None, + } + } + + pub fn client(&mut self, client: Client) -> &mut Self { + self.client = Some(client); + self + } + + pub fn data(&mut self, data: Data) -> &mut Self { + self.data = Some(data); + self + } + + pub fn build(self) -> Result> { + Ok(if let Some(data) = self.data { + Mastodon { + client: self.client.unwrap_or_else(|| Client::new()), + http_sender: self.http_sender, + data, + } + } else { + return Err(Error::MissingField("missing field 'data'")); + }) + } +} + +/// Client that can make unauthenticated calls to a mastodon instance +#[derive(Clone, Debug)] +pub struct MastodonUnauth { + client: Client, + http_sender: H, + base: url::Url, +} + +impl MastodonUnauth { + /// Create a new unauthenticated client + pub fn new(base: &str) -> Result> { + let base = if base.starts_with("https://") { + base.to_string() + } else { + format!("https://{}", base) + }; + Ok(MastodonUnauth { + client: Client::new(), + http_sender: HttpSender, + base: url::Url::parse(&base)?, + }) + } +} + +impl MastodonUnauth { + fn route(&self, url: &str) -> Result { + Ok(self.base.join(url)?) + } + + fn send(&self, req: RequestBuilder) -> Result { + Ok(self.http_sender.send(&self.client, req)?) + } +} + +impl MastodonUnauthenticated for MastodonUnauth { + /// GET /api/v1/statuses/:id + fn get_status(&self, id: &str) -> Result { + let route = self.route("/api/v1/statuses")?; + let route = route.join(id)?; + let response = self.send(self.client.get(route))?; + deserialise(response) + } + + /// GET /api/v1/statuses/:id/context + fn get_context(&self, id: &str) -> Result { + let route = self.route("/api/v1/statuses")?; + let route = route.join(id)?; + let route = route.join("context")?; + let response = self.send(self.client.get(route))?; + deserialise(response) + } + + /// GET /api/v1/statuses/:id/card + fn get_card(&self, id: &str) -> Result { + let route = self.route("/api/v1/statuses")?; + let route = route.join(id)?; + let route = route.join("card")?; + let response = self.send(self.client.get(route))?; + deserialise(response) + } +} + +// Convert the HTTP response body from JSON. Pass up deserialization errors +// transparently. +fn deserialise serde::Deserialize<'de>>(response: Response) -> Result { + let mut reader = Tap::new(response); + + match serde_json::from_reader(&mut reader) { + Ok(t) => { + debug!("{}", String::from_utf8_lossy(&reader.bytes)); + Ok(t) + }, + // If deserializing into the desired type fails try again to + // see if this is an error response. + Err(e) => { + error!("{}", String::from_utf8_lossy(&reader.bytes)); + if let Ok(error) = serde_json::from_slice(&reader.bytes) { + return Err(Error::Api(error)); + } + Err(e.into()) + }, + } +} diff --git a/src/macros.rs b/src/macros.rs new file mode 100644 index 0000000..0f9f1a2 --- /dev/null +++ b/src/macros.rs @@ -0,0 +1,364 @@ +macro_rules! methods { + ($($method:ident,)+) => { + $( + fn $method serde::Deserialize<'de>>(&self, url: String) + -> Result + { + let response = self.send( + self.client.$method(&url) + )?; + + deserialise(response) + } + )+ + }; +} + +macro_rules! paged_routes { + + (($method:ident) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { + doc_comment! { + concat!( + "Equivalent to `", stringify!($method), " /api/v1/", + $url, + "`\n# Errors\nIf `access_token` is not set.", + "\n", + "```no_run", + "# extern crate elefren;\n", + "# use elefren::prelude::*;\n", + "# fn main() -> Result<(), Box<::std::error::Error>> {\n", + "# let data = Data {\n", + "# base: \"https://example.com\".into(),\n", + "# client_id: \"taosuah\".into(),\n", + "# client_secret: \"htnjdiuae\".into(),\n", + "# redirect: \"https://example.com\".into(),\n", + "# token: \"tsaohueaheis\".into(),\n", + "# };\n", + "let client = Mastodon::from(data);\n", + "client.", stringify!($name), "();\n", + "# Ok(())\n", + "# }\n", + "```" + ), + fn $name(&self) -> Result> { + let url = self.route(concat!("/api/v1/", $url)); + let response = self.send( + self.client.$method(&url) + )?; + + Page::new(self, response) + } + + } + + paged_routes!{$($rest)*} + }; + + ((get ($($(#[$m:meta])* $param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { + doc_comment! { + concat!( + "Equivalent to `get /api/v1/", + $url, + "`\n# Errors\nIf `access_token` is not set." + ), + fn $name<'a>(&self, $($param: $typ,)*) -> Result> { + use serde_urlencoded; + + #[derive(Serialize)] + struct Data<'a> { + $( + $( + #[$m] + )* + $param: $typ, + )* + #[serde(skip)] + _marker: ::std::marker::PhantomData<&'a ()>, + } + + let qs_data = Data { + $( + $param: $param, + )* + _marker: ::std::marker::PhantomData, + }; + + let qs = serde_urlencoded::to_string(&qs_data)?; + + let url = format!(concat!("/api/v1/", $url, "?{}"), &qs); + + let response = self.send( + self.client.get(&url) + )?; + + Page::new(self, response) + } + } + + paged_routes!{$($rest)*} + }; + + () => {} +} + +macro_rules! route_v2 { + ((get ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { + doc_comment! { + concat!( + "Equivalent to `get /api/v2/", + $url, + "`\n# Errors\nIf `access_token` is not set." + ), + fn $name<'a>(&self, $($param: $typ,)*) -> Result<$ret> { + use serde_urlencoded; + + #[derive(Serialize)] + struct Data<'a> { + $( + $param: $typ, + )* + #[serde(skip)] + _marker: ::std::marker::PhantomData<&'a ()>, + } + + let qs_data = Data { + $( + $param: $param, + )* + _marker: ::std::marker::PhantomData, + }; + + let qs = serde_urlencoded::to_string(&qs_data)?; + + let url = format!(concat!("/api/v2/", $url, "?{}"), &qs); + + Ok(self.get(self.route(&url))?) + } + } + + route_v2!{$($rest)*} + }; + + () => {} +} + +macro_rules! route { + + ((post multipart ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { + doc_comment! { + concat!( + "Equivalent to `post /api/v1/", + $url, + "`\n# Errors\nIf `access_token` is not set."), + fn $name(&self, $($param: $typ,)*) -> Result<$ret> { + use reqwest::multipart::Form; + + let form_data = Form::new() + $( + .file(stringify!($param), $param.as_ref())? + )*; + + let response = self.send( + self.client + .post(&self.route(concat!("/api/v1/", $url))) + .multipart(form_data) + )?; + + let status = response.status().clone(); + + if status.is_client_error() { + return Err(Error::Client(status)); + } else if status.is_server_error() { + return Err(Error::Server(status)); + } + + deserialise(response) + } + } + + route!{$($rest)*} + }; + + ((get ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { + doc_comment! { + concat!( + "Equivalent to `get /api/v1/", + $url, + "`\n# Errors\nIf `access_token` is not set." + ), + fn $name<'a>(&self, $($param: $typ,)*) -> Result<$ret> { + use serde_urlencoded; + + #[derive(Serialize)] + struct Data<'a> { + $( + $param: $typ, + )* + #[serde(skip)] + _marker: ::std::marker::PhantomData<&'a ()>, + } + + let qs_data = Data { + $( + $param: $param, + )* + _marker: ::std::marker::PhantomData, + }; + + let qs = serde_urlencoded::to_string(&qs_data)?; + + let url = format!(concat!("/api/v1/", $url, "?{}"), &qs); + + Ok(self.get(self.route(&url))?) + } + } + + route!{$($rest)*} + }; + + (($method:ident ($($param:ident: $typ:ty,)*)) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { + doc_comment! { + concat!( + "Equivalent to `", stringify!($method), " /api/v1/", + $url, + "`\n# Errors\nIf `access_token` is not set.", + ), + fn $name(&self, $($param: $typ,)*) -> Result<$ret> { + + let form_data = json!({ + $( + stringify!($param): $param, + )* + }); + + let response = self.send( + self.client.$method(&self.route(concat!("/api/v1/", $url))) + .json(&form_data) + )?; + + let status = response.status().clone(); + + if status.is_client_error() { + return Err(Error::Client(status)); + } else if status.is_server_error() { + return Err(Error::Server(status)); + } + + deserialise(response) + } + } + + route!{$($rest)*} + }; + + (($method:ident) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { + doc_comment! { + concat!( + "Equivalent to `", stringify!($method), " /api/v1/", + $url, + "`\n# Errors\nIf `access_token` is not set.", + "\n", + "```no_run", + "# extern crate elefren;\n", + "# use elefren::prelude::*;\n", + "# fn main() -> Result<(), Box<::std::error::Error>> {\n", + "# let data = Data {\n", + "# base: \"https://example.com\".into(),\n", + "# client_id: \"taosuah\".into(),\n", + "# client_secret: \"htnjdiuae\".into(),\n", + "# redirect: \"https://example.com\".into(),\n", + "# token: \"tsaohueaheis\".into(),\n", + "# };\n", + "let client = Mastodon::from(data);\n", + "client.", stringify!($name), "();\n", + "# Ok(())\n", + "# }\n", + "```" + ), + fn $name(&self) -> Result<$ret> { + self.$method(self.route(concat!("/api/v1/", $url))) + } + } + + route!{$($rest)*} + }; + + () => {} +} + +macro_rules! route_id { + + ($(($method:ident) $name:ident: $url:expr => $ret:ty,)*) => { + $( + doc_comment! { + concat!( + "Equivalent to `", stringify!($method), " /api/v1/", + $url, + "`\n# Errors\nIf `access_token` is not set.", + "\n", + "```no_run", + "# extern crate elefren;\n", + "# use elefren::prelude::*;\n", + "# fn main() -> Result<(), Box<::std::error::Error>> {\n", + "# let data = Data {\n", + "# base: \"https://example.com\".into(),\n", + "# client_id: \"taosuah\".into(),\n", + "# client_secret: \"htnjdiuae\".into(),\n", + "# redirect: \"https://example.com\".into(),\n", + "# token: \"tsaohueaheis\".into(),\n", + "# };\n", + "let client = Mastodon::from(data);\n", + "client.", stringify!($name), "(\"42\");\n", + "# Ok(())\n", + "# }\n", + "```" + ), + fn $name(&self, id: &str) -> Result<$ret> { + self.$method(self.route(&format!(concat!("/api/v1/", $url), id))) + } + } + )* + } + +} +macro_rules! paged_routes_with_id { + + (($method:ident) $name:ident: $url:expr => $ret:ty, $($rest:tt)*) => { + doc_comment! { + concat!( + "Equivalent to `", stringify!($method), " /api/v1/", + $url, + "`\n# Errors\nIf `access_token` is not set.", + "\n", + "```no_run", + "# extern crate elefren;\n", + "# use elefren::prelude::*;\n", + "# fn main() -> Result<(), Box<::std::error::Error>> {\n", + "# let data = Data {\n", + "# base: \"https://example.com\".into(),\n", + "# client_id: \"taosuah\".into(),\n", + "# client_secret: \"htnjdiuae\".into(),\n", + "# redirect: \"https://example.com\".into(),\n", + "# token: \"tsaohueaheis\".into(),\n", + "# };\n", + "let client = Mastodon::from(data);\n", + "client.", stringify!($name), "(\"some-id\");\n", + "# Ok(())\n", + "# }\n", + "```" + ), + fn $name(&self, id: &str) -> Result> { + let url = self.route(&format!(concat!("/api/v1/", $url), id)); + let response = self.send( + self.client.$method(&url) + )?; + + Page::new(self, response) + } + } + + paged_routes_with_id!{$($rest)*} + }; + + () => {} +} diff --git a/src/mastodon_client.rs b/src/mastodon_client.rs new file mode 100644 index 0000000..6a63c7c --- /dev/null +++ b/src/mastodon_client.rs @@ -0,0 +1,388 @@ +use std::borrow::Cow; + +use entities::prelude::*; +use errors::Result; +use http_send::{HttpSend, HttpSender}; +use page::Page; +use requests::{ + AddFilterRequest, + AddPushRequest, + StatusesRequest, + UpdateCredsRequest, + UpdatePushRequest, +}; +use status_builder::NewStatus; + +/// Represents the set of methods that a Mastodon Client can do, so that +/// implementations might be swapped out for testing +#[allow(unused)] +pub trait MastodonClient { + /// Type that wraps streaming API streams + type Stream: Iterator; + + /// GET /api/v1/favourites + fn favourites(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/blocks + fn blocks(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/domain_blocks + fn domain_blocks(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/follow_requests + fn follow_requests(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/timelines/home + fn get_home_timeline(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/custom_emojis + fn get_emojis(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/mutes + fn mutes(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/notifications + fn notifications(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/reports + fn reports(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/:id/followers + fn followers(&self, id: &str) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/:id/following + fn following(&self, id: &str) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/statuses/:id/reblogged_by + fn reblogged_by(&self, id: &str) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/statuses/:id/favourited_by + fn favourited_by(&self, id: &str) -> Result> { + unimplemented!("This method was not implemented"); + } + /// DELETE /api/v1/domain_blocks + fn unblock_domain(&self, domain: String) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/instance + fn instance(&self) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/verify_credentials + fn verify_credentials(&self) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/reports + fn report(&self, account_id: &str, status_ids: Vec<&str>, comment: String) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/domain_blocks + fn block_domain(&self, domain: String) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/accounts/follow_requests/authorize + fn authorize_follow_request(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/accounts/follow_requests/reject + fn reject_follow_request(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/search + fn search<'a>(&self, q: &'a str, resolve: bool) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v2/search + fn search_v2<'a>(&self, q: &'a str, resolve: bool) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/follows + fn follows(&self, uri: Cow<'static, str>) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/media + fn media(&self, file: Cow<'static, str>) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/notifications/clear + fn clear_notifications(&self) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/notifications/dismiss + fn dismiss_notification(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/:id + fn get_account(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/accounts/:id/follow + fn follow(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/accounts/:id/unfollow + fn unfollow(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/:id/block + fn block(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/:id/unblock + fn unblock(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/:id/mute + fn mute(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/:id/unmute + fn unmute(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/notifications/:id + fn get_notification(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/statuses/:id + fn get_status(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/statuses/:id/context + fn get_context(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/statuses/:id/card + fn get_card(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/statuses/:id/reblog + fn reblog(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/statuses/:id/unreblog + fn unreblog(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/statuses/:id/favourite + fn favourite(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/statuses/:id/unfavourite + fn unfavourite(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// DELETE /api/v1/statuses/:id + fn delete_status(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// PATCH /api/v1/accounts/update_credentials + fn update_credentials(&self, builder: &mut UpdateCredsRequest) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/statuses + fn new_status(&self, status: NewStatus) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/timelines/public + fn get_public_timeline(&self, local: bool) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/timelines/tag/:hashtag + fn get_tagged_timeline(&self, hashtag: String, local: bool) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/:id/statuses + fn statuses<'a, 'b: 'a, S>(&'b self, id: &'b str, request: S) -> Result> + where + S: Into>>, + { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/relationships + fn relationships(&self, ids: &[&str]) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/accounts/search?q=:query&limit=:limit&following=:following + fn search_accounts( + &self, + query: &str, + limit: Option, + following: bool, + ) -> Result> { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/push/subscription + fn add_push_subscription(&self, request: &AddPushRequest) -> Result { + unimplemented!("This method was not implemented"); + } + /// PUT /api/v1/push/subscription + fn update_push_data(&self, request: &UpdatePushRequest) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/push/subscription + fn get_push_subscription(&self) -> Result { + unimplemented!("This method was not implemented"); + } + /// DELETE /api/v1/push/subscription + fn delete_push_subscription(&self) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/filters + fn get_filters(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/filters + fn add_filter(&self, request: &mut AddFilterRequest) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/filters/:id + fn get_filter(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// PUT /api/v1/filters/:id + fn update_filter(&self, id: &str, request: &mut AddFilterRequest) -> Result { + unimplemented!("This method was not implemented"); + } + /// DELETE /api/v1/filters/:id + fn delete_filter(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/suggestions + fn get_follow_suggestions(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// DELETE /api/v1/suggestions/:account_id + fn delete_from_suggestions(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/endorsements + fn get_endorsements(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/accounts/:id/pin + fn endorse_user(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// POST /api/v1/accounts/:id/unpin + fn unendorse_user(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// Shortcut for: `let me = client.verify_credentials(); client.followers()` + /// + /// ```no_run + /// # extern crate elefren; + /// # use std::error::Error; + /// # use elefren::prelude::*; + /// # fn main() -> Result<(), Box> { + /// # let data = Data { + /// # base: "".into(), + /// # client_id: "".into(), + /// # client_secret: "".into(), + /// # redirect: "".into(), + /// # token: "".into(), + /// # }; + /// # let client = Mastodon::from(data); + /// let follows_me = client.follows_me()?; + /// # Ok(()) + /// # } + fn follows_me(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + /// Shortcut for + /// `let me = client.verify_credentials(); client.following(&me.id)` + /// + /// ```no_run + /// # extern crate elefren; + /// # use std::error::Error; + /// # use elefren::prelude::*; + /// # fn main() -> Result<(), Box> { + /// # let data = Data { + /// # base: "".into(), + /// # client_id: "".into(), + /// # client_secret: "".into(), + /// # redirect: "".into(), + /// # token: "".into(), + /// # }; + /// # let client = Mastodon::from(data); + /// let follows_me = client.followed_by_me()?; + /// # Ok(()) + /// # } + fn followed_by_me(&self) -> Result> { + unimplemented!("This method was not implemented"); + } + + /// Returns events that are relevant to the authorized user, i.e. home + /// timeline and notifications + fn streaming_user(&self) -> Result { + unimplemented!("This method was not implemented"); + } + + /// Returns all public statuses + fn streaming_public(&self) -> Result { + unimplemented!("This method was not implemented"); + } + + /// Returns all local statuses + fn streaming_local(&self) -> Result { + unimplemented!("This method was not implemented"); + } + + /// Returns all public statuses for a particular hashtag + fn streaming_public_hashtag(&self, hashtag: &str) -> Result { + unimplemented!("This method was not implemented"); + } + + /// Returns all local statuses for a particular hashtag + fn streaming_local_hashtag(&self, hashtag: &str) -> Result { + unimplemented!("This method was not implemented"); + } + + /// Returns statuses for a list + fn streaming_list(&self, list_id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + + /// Returns all direct messages + fn streaming_direct(&self) -> Result { + unimplemented!("This method was not implemented"); + } +} + +/// Trait that represents clients that can make unauthenticated calls to a +/// mastodon instance +#[allow(unused)] +pub trait MastodonUnauthenticated { + /// GET /api/v1/statuses/:id + fn get_status(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/statuses/:id/context + fn get_context(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/statuses/:id/card + fn get_card(&self, id: &str) -> Result { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/statuses/:id/reblogged_by + fn reblogged_by(&self, id: &str) -> Result> { + unimplemented!("This method was not implemented"); + } + /// GET /api/v1/statuses/:id/favourited_by + fn favourited_by(&self, id: &str) -> Result> { + unimplemented!("This method was not implemented"); + } +} diff --git a/src/page.rs b/src/page.rs new file mode 100644 index 0000000..3a47ce6 --- /dev/null +++ b/src/page.rs @@ -0,0 +1,226 @@ +use super::{deserialise, Mastodon, Result}; +use entities::itemsiter::ItemsIter; +use hyper_old_types::header::{parsing, Link, RelationType}; +use reqwest::{header::LINK, Response}; +use serde::Deserialize; +use url::Url; + +use http_send::HttpSend; + +macro_rules! pages { + ($($direction:ident: $fun:ident),*) => { + + $( + doc_comment!(concat!( + "Method to retrieve the ", stringify!($direction), " page of results"), + pub fn $fun(&mut self) -> Result>> { + let url = match self.$direction.take() { + Some(s) => s, + None => return Ok(None), + }; + + let response = self.mastodon.send( + self.mastodon.client.get(url) + )?; + + let (prev, next) = get_links(&response)?; + self.next = next; + self.prev = prev; + + deserialise(response) + }); + )* + } +} + +/// Owned version of the `Page` struct in this module. Allows this to be more +/// easily stored for later use +/// +/// # Example +/// +/// ```no_run +/// # extern crate elefren; +/// # use elefren::Mastodon; +/// # use elefren::page::OwnedPage; +/// # use elefren::http_send::HttpSender; +/// # use elefren::entities::status::Status; +/// # use std::cell::RefCell; +/// # use elefren::prelude::*; +/// # fn main() -> Result<(), elefren::Error> { +/// # let data = Data { +/// # base: "".into(), +/// # client_id: "".into(), +/// # client_secret: "".into(), +/// # redirect: "".into(), +/// # token: "".into(), +/// # }; +/// struct HomeTimeline { +/// client: Mastodon, +/// page: RefCell>>, +/// } +/// let client = Mastodon::from(data); +/// let home = client.get_home_timeline()?.to_owned(); +/// let tl = HomeTimeline { +/// client, +/// page: RefCell::new(Some(home)), +/// }; +/// # Ok(()) +/// # } +/// ``` +#[derive(Debug, Clone)] +pub struct OwnedPage Deserialize<'de>, H: HttpSend> { + mastodon: Mastodon, + next: Option, + prev: Option, + /// Initial set of items + pub initial_items: Vec, +} + +impl Deserialize<'de>, H: HttpSend> OwnedPage { + pages! { + next: next_page, + prev: prev_page + } +} + +impl<'a, T: for<'de> Deserialize<'de>, H: HttpSend> From> for OwnedPage { + fn from(page: Page<'a, T, H>) -> OwnedPage { + OwnedPage { + mastodon: page.mastodon.clone(), + next: page.next, + prev: page.prev, + initial_items: page.initial_items, + } + } +} + +/// Represents a single page of API results +#[derive(Debug, Clone)] +pub struct Page<'a, T: for<'de> Deserialize<'de>, H: 'a + HttpSend> { + mastodon: &'a Mastodon, + next: Option, + prev: Option, + /// Initial set of items + pub initial_items: Vec, +} + +impl<'a, T: for<'de> Deserialize<'de>, H: HttpSend> Page<'a, T, H> { + pages! { + next: next_page, + prev: prev_page + } + + pub(crate) fn new(mastodon: &'a Mastodon, response: Response) -> Result { + let (prev, next) = get_links(&response)?; + Ok(Page { + initial_items: deserialise(response)?, + next, + prev, + mastodon, + }) + } +} + +impl<'a, T: Clone + for<'de> Deserialize<'de>, H: HttpSend> Page<'a, T, H> { + /// Returns an owned version of this struct that doesn't borrow the client + /// that created it + /// + /// # Example + /// + /// ```no_run + /// # extern crate elefren; + /// # use elefren::Mastodon; + /// # use elefren::page::OwnedPage; + /// # use elefren::http_send::HttpSender; + /// # use elefren::entities::status::Status; + /// # use std::cell::RefCell; + /// # use elefren::prelude::*; + /// # fn main() -> Result<(), elefren::Error> { + /// # let data = Data { + /// # base: "".into(), + /// # client_id: "".into(), + /// # client_secret: "".into(), + /// # redirect: "".into(), + /// # token: "".into(), + /// # }; + /// struct HomeTimeline { + /// client: Mastodon, + /// page: RefCell>>, + /// } + /// let client = Mastodon::from(data); + /// let home = client.get_home_timeline()?.to_owned(); + /// let tl = HomeTimeline { + /// client, + /// page: RefCell::new(Some(home)), + /// }; + /// # Ok(()) + /// # } + /// ``` + pub fn to_owned(self) -> OwnedPage { + OwnedPage::from(self) + } + + /// Returns an iterator that provides a stream of `T`s + /// + /// This abstracts away the process of iterating over each item in a page, + /// then making an http call, then iterating over each item in the new + /// page, etc. The iterator provides a stream of `T`s, calling + /// `self.next_page()` + /// when necessary to get + /// more of them, until + /// there are no more items. + /// + /// # Example + /// + /// ```no_run + /// # extern crate elefren; + /// # use std::error::Error; + /// use elefren::prelude::*; + /// # fn main() -> Result<(), Box> { + /// # let data = Data { + /// # base: "".into(), + /// # client_id: "".into(), + /// # client_secret: "".into(), + /// # redirect: "".into(), + /// # token: "".into(), + /// # }; + /// let mastodon = Mastodon::from(data); + /// let req = StatusesRequest::new(); + /// let resp = mastodon.statuses("some-id", req)?; + /// for status in resp.items_iter() { + /// // do something with status + /// } + /// # Ok(()) + /// # } + /// ``` + pub fn items_iter(self) -> impl Iterator + 'a + where + T: 'a, + { + ItemsIter::new(self) + } +} + +fn get_links(response: &Response) -> Result<(Option, Option)> { + let mut prev = None; + let mut next = None; + + if let Some(link_header) = response.headers().get(LINK) { + let link_header = link_header.to_str()?; + let link_header = link_header.as_bytes(); + let link_header: Link = parsing::from_raw_str(&link_header)?; + for value in link_header.values() { + if let Some(relations) = value.rel() { + if relations.contains(&RelationType::Next) { + next = Some(Url::parse(value.link())?); + } + + if relations.contains(&RelationType::Prev) { + prev = Some(Url::parse(value.link())?); + } + } + } + } + + Ok((prev, next)) +} diff --git a/src/registration.rs b/src/registration.rs new file mode 100644 index 0000000..7aba7c5 --- /dev/null +++ b/src/registration.rs @@ -0,0 +1,430 @@ +use std::borrow::Cow; + +use reqwest::{Client, RequestBuilder, Response}; +use try_from::TryInto; +use url::percent_encoding::{utf8_percent_encode, DEFAULT_ENCODE_SET}; + +use apps::{App, AppBuilder}; +use http_send::{HttpSend, HttpSender}; +use scopes::Scopes; +use Data; +use Error; +use Mastodon; +use MastodonBuilder; +use Result; + +const DEFAULT_REDIRECT_URI: &'static str = "urn:ietf:wg:oauth:2.0:oob"; + +/// Handles registering your mastodon app to your instance. It is recommended +/// you cache your data struct to avoid registering on every run. +#[derive(Debug, Clone)] +pub struct Registration<'a, H: HttpSend = HttpSender> { + base: String, + client: Client, + app_builder: AppBuilder<'a>, + force_login: bool, + http_sender: H, +} + +#[derive(Deserialize)] +struct OAuth { + client_id: String, + client_secret: String, + #[serde(default = "default_redirect_uri")] + redirect_uri: String, +} + +fn default_redirect_uri() -> String { + DEFAULT_REDIRECT_URI.to_string() +} + +#[derive(Deserialize)] +struct AccessToken { + access_token: String, +} + +impl<'a> Registration<'a, HttpSender> { + /// Construct a new registration process to the instance of the `base` url. + /// ``` + /// use elefren::prelude::*; + /// + /// let registration = Registration::new("https://mastodon.social"); + /// ``` + pub fn new>(base: I) -> Self { + Registration { + base: base.into(), + client: Client::new(), + app_builder: AppBuilder::new(), + force_login: false, + http_sender: HttpSender, + } + } +} + +impl<'a, H: HttpSend> Registration<'a, H> { + #[allow(dead_code)] + pub(crate) fn with_sender>(base: I, http_sender: H) -> Self { + Registration { + base: base.into(), + client: Client::new(), + app_builder: AppBuilder::new(), + force_login: false, + http_sender, + } + } + + /// Sets the name of this app + /// + /// This is required, and if this isn't set then the AppBuilder::build + /// method will fail + pub fn client_name>>(&mut self, name: I) -> &mut Self { + self.app_builder.client_name(name.into()); + self + } + + /// Sets the redirect uris that this app uses + pub fn redirect_uris>>(&mut self, uris: I) -> &mut Self { + self.app_builder.redirect_uris(uris); + self + } + + /// Sets the scopes that this app requires + /// + /// The default for an app is Scopes::Read + pub fn scopes(&mut self, scopes: Scopes) -> &mut Self { + self.app_builder.scopes(scopes); + self + } + + /// Sets the optional "website" to register the app with + pub fn website>>(&mut self, website: I) -> &mut Self { + self.app_builder.website(website); + self + } + + /// Forces the user to re-login (useful if you need to re-auth as a + /// different user on the same instance + pub fn force_login(&mut self, force_login: bool) -> &mut Self { + self.force_login = force_login; + self + } + + fn send(&self, req: RequestBuilder) -> Result { + Ok(self.http_sender.send(&self.client, req)?) + } + + /// Register the given application + /// + /// ```no_run + /// # extern crate elefren; + /// # fn main () -> elefren::Result<()> { + /// use elefren::{apps::App, prelude::*}; + /// + /// let mut app = App::builder(); + /// app.client_name("elefren_test"); + /// + /// let registration = Registration::new("https://mastodon.social").register(app)?; + /// let url = registration.authorize_url()?; + /// // Here you now need to open the url in the browser + /// // And handle a the redirect url coming back with the code. + /// let code = String::from("RETURNED_FROM_BROWSER"); + /// let mastodon = registration.complete(&code)?; + /// + /// println!("{:?}", mastodon.get_home_timeline()?.initial_items); + /// # Ok(()) + /// # } + /// ``` + pub fn register>(&mut self, app: I) -> Result> + where + Error: From<>::Err>, + { + let app = app.try_into()?; + let oauth = self.send_app(&app)?; + + Ok(Registered { + base: self.base.clone(), + client: self.client.clone(), + client_id: oauth.client_id, + client_secret: oauth.client_secret, + redirect: oauth.redirect_uri, + scopes: app.scopes().clone(), + force_login: self.force_login, + http_sender: self.http_sender.clone(), + }) + } + + /// Register the application with the server from the `base` url. + /// + /// ```no_run + /// # extern crate elefren; + /// # fn main () -> elefren::Result<()> { + /// use elefren::prelude::*; + /// + /// let registration = Registration::new("https://mastodon.social") + /// .client_name("elefren_test") + /// .build()?; + /// let url = registration.authorize_url()?; + /// // Here you now need to open the url in the browser + /// // And handle a the redirect url coming back with the code. + /// let code = String::from("RETURNED_FROM_BROWSER"); + /// let mastodon = registration.complete(&code)?; + /// + /// println!("{:?}", mastodon.get_home_timeline()?.initial_items); + /// # Ok(()) + /// # } + /// ``` + pub fn build(&mut self) -> Result> { + let app: App = self.app_builder.clone().build()?; + let oauth = self.send_app(&app)?; + + Ok(Registered { + base: self.base.clone(), + client: self.client.clone(), + client_id: oauth.client_id, + client_secret: oauth.client_secret, + redirect: oauth.redirect_uri, + scopes: app.scopes().clone(), + force_login: self.force_login, + http_sender: self.http_sender.clone(), + }) + } + + fn send_app(&self, app: &App) -> Result { + let url = format!("{}/api/v1/apps", self.base); + Ok(self.send(self.client.post(&url).json(&app))?.json()?) + } +} + +impl Registered { + /// Skip having to retrieve the client id and secret from the server by + /// creating a `Registered` struct directly + /// + /// # Example + /// + /// ```no_run + /// # extern crate elefren; + /// # fn main() -> elefren::Result<()> { + /// use elefren::{prelude::*, registration::Registered}; + /// + /// let registration = Registered::from_parts( + /// "https://example.com", + /// "the-client-id", + /// "the-client-secret", + /// "https://example.com/redirect", + /// Scopes::read_all(), + /// false, + /// ); + /// let url = registration.authorize_url()?; + /// // Here you now need to open the url in the browser + /// // And handle a the redirect url coming back with the code. + /// let code = String::from("RETURNED_FROM_BROWSER"); + /// let mastodon = registration.complete(&code)?; + /// + /// println!("{:?}", mastodon.get_home_timeline()?.initial_items); + /// # Ok(()) + /// # } + /// ``` + pub fn from_parts( + base: &str, + client_id: &str, + client_secret: &str, + redirect: &str, + scopes: Scopes, + force_login: bool, + ) -> Registered { + Registered { + base: base.to_string(), + client: Client::new(), + client_id: client_id.to_string(), + client_secret: client_secret.to_string(), + redirect: redirect.to_string(), + scopes, + force_login, + http_sender: HttpSender, + } + } +} + +impl Registered { + fn send(&self, req: RequestBuilder) -> Result { + Ok(self.http_sender.send(&self.client, req)?) + } + + /// Returns the parts of the `Registered` struct that can be used to + /// recreate another `Registered` struct + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::{prelude::*, registration::Registered}; + /// # fn main() -> Result<(), Box> { + /// + /// let origbase = "https://example.social"; + /// let origclient_id = "some-client_id"; + /// let origclient_secret = "some-client-secret"; + /// let origredirect = "https://example.social/redirect"; + /// let origscopes = Scopes::all(); + /// let origforce_login = false; + /// + /// let registered = Registered::from_parts( + /// origbase, + /// origclient_id, + /// origclient_secret, + /// origredirect, + /// origscopes.clone(), + /// origforce_login, + /// ); + /// + /// let (base, client_id, client_secret, redirect, scopes, force_login) = registered.into_parts(); + /// + /// assert_eq!(origbase, &base); + /// assert_eq!(origclient_id, &client_id); + /// assert_eq!(origclient_secret, &client_secret); + /// assert_eq!(origredirect, &redirect); + /// assert_eq!(origscopes, scopes); + /// assert_eq!(origforce_login, force_login); + /// # Ok(()) + /// # } + /// ``` + pub fn into_parts(self) -> (String, String, String, String, Scopes, bool) { + ( + self.base, + self.client_id, + self.client_secret, + self.redirect, + self.scopes, + self.force_login, + ) + } + + /// Returns the full url needed for authorisation. This needs to be opened + /// in a browser. + pub fn authorize_url(&self) -> Result { + let scopes = format!("{}", self.scopes); + let scopes: String = utf8_percent_encode(&scopes, DEFAULT_ENCODE_SET).collect(); + let url = if self.force_login { + format!( + "{}/oauth/authorize?client_id={}&redirect_uri={}&scope={}&force_login=true&\ + response_type=code", + self.base, self.client_id, self.redirect, scopes, + ) + } else { + format!( + "{}/oauth/authorize?client_id={}&redirect_uri={}&scope={}&response_type=code", + self.base, self.client_id, self.redirect, scopes, + ) + }; + + Ok(url) + } + + /// Create an access token from the client id, client secret, and code + /// provided by the authorisation url. + pub fn complete(&self, code: &str) -> Result> { + let url = format!( + "{}/oauth/token?client_id={}&client_secret={}&code={}&grant_type=authorization_code&\ + redirect_uri={}", + self.base, self.client_id, self.client_secret, code, self.redirect + ); + + let token: AccessToken = self.send(self.client.post(&url))?.json()?; + + let data = Data { + base: self.base.clone().into(), + client_id: self.client_id.clone().into(), + client_secret: self.client_secret.clone().into(), + redirect: self.redirect.clone().into(), + token: token.access_token.into(), + }; + + let mut builder = MastodonBuilder::new(self.http_sender.clone()); + builder.client(self.client.clone()).data(data); + Ok(builder.build()?) + } +} + +/// Represents the state of the auth flow when the app has been registered but +/// the user is not authenticated +#[derive(Debug, Clone)] +pub struct Registered { + base: String, + client: Client, + client_id: String, + client_secret: String, + redirect: String, + scopes: Scopes, + force_login: bool, + http_sender: H, +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn test_registration_new() { + let r = Registration::new("https://example.com"); + assert_eq!(r.base, "https://example.com".to_string()); + assert_eq!(r.app_builder, AppBuilder::new()); + assert_eq!(r.http_sender, HttpSender); + } + + #[test] + fn test_registration_with_sender() { + let r = Registration::with_sender("https://example.com", HttpSender); + assert_eq!(r.base, "https://example.com".to_string()); + assert_eq!(r.app_builder, AppBuilder::new()); + assert_eq!(r.http_sender, HttpSender); + } + + #[test] + fn test_set_client_name() { + let mut r = Registration::new("https://example.com"); + r.client_name("foo-test"); + + assert_eq!(r.base, "https://example.com".to_string()); + assert_eq!( + &mut r.app_builder, + AppBuilder::new().client_name("foo-test") + ); + } + + #[test] + fn test_set_redirect_uris() { + let mut r = Registration::new("https://example.com"); + r.redirect_uris("https://foo.com"); + + assert_eq!(r.base, "https://example.com".to_string()); + assert_eq!( + &mut r.app_builder, + AppBuilder::new().redirect_uris("https://foo.com") + ); + } + + #[test] + fn test_set_scopes() { + let mut r = Registration::new("https://example.com"); + r.scopes(Scopes::all()); + + assert_eq!(r.base, "https://example.com".to_string()); + assert_eq!(&mut r.app_builder, AppBuilder::new().scopes(Scopes::all())); + } + + #[test] + fn test_set_website() { + let mut r = Registration::new("https://example.com"); + r.website("https://website.example.com"); + + assert_eq!(r.base, "https://example.com".to_string()); + assert_eq!( + &mut r.app_builder, + AppBuilder::new().website("https://website.example.com") + ); + } + + #[test] + fn test_default_redirect_uri() { + assert_eq!(&default_redirect_uri()[..], DEFAULT_REDIRECT_URI); + } +} diff --git a/src/requests/filter.rs b/src/requests/filter.rs new file mode 100644 index 0000000..6f3c700 --- /dev/null +++ b/src/requests/filter.rs @@ -0,0 +1,154 @@ +use entities::filter::FilterContext; +use std::time::Duration; + +/// Form used to create a filter +/// +/// # Example +/// +/// ``` +/// # extern crate elefren; +/// # use std::error::Error; +/// use elefren::{entities::filter::FilterContext, requests::AddFilterRequest}; +/// # fn main() -> Result<(), Box> { +/// let request = AddFilterRequest::new("foo", FilterContext::Home); +/// # Ok(()) +/// # } +/// ``` +#[derive(Debug, Clone, PartialEq, Serialize)] +pub struct AddFilterRequest { + phrase: String, + context: FilterContext, + irreversible: Option, + whole_word: Option, + #[serde(serialize_with = "serialize_duration::ser")] + expires_in: Option, +} + +impl AddFilterRequest { + /// Create a new AddFilterRequest + pub fn new(phrase: &str, context: FilterContext) -> AddFilterRequest { + AddFilterRequest { + phrase: phrase.to_string(), + context, + irreversible: None, + whole_word: None, + expires_in: None, + } + } + + /// Set `irreversible` to `true` + pub fn irreversible(&mut self) -> &mut Self { + self.irreversible = Some(true); + self + } + + /// Set `whole_word` to `true` + pub fn whole_word(&mut self) -> &mut Self { + self.whole_word = Some(true); + self + } + + /// Set `expires_in` to a duration + pub fn expires_in(&mut self, d: Duration) -> &mut Self { + self.expires_in = Some(d); + self + } +} + +mod serialize_duration { + use serde::ser::Serializer; + use std::time::Duration; + + pub(crate) fn ser(duration: &Option, s: S) -> Result + where + S: Serializer, + { + if let Some(d) = duration { + let sec = d.as_secs(); + s.serialize_u64(sec) + } else { + s.serialize_none() + } + } +} + +#[cfg(test)] +mod tests { + use super::*; + use serde_json; + use std::time::Duration; + + #[test] + fn test_new() { + let request = AddFilterRequest::new("foo", FilterContext::Home); + assert_eq!( + request, + AddFilterRequest { + phrase: "foo".to_string(), + context: FilterContext::Home, + irreversible: None, + whole_word: None, + expires_in: None, + } + ) + } + + #[test] + fn test_irreversible() { + let mut request = AddFilterRequest::new("foo", FilterContext::Home); + request.irreversible(); + assert_eq!( + request, + AddFilterRequest { + phrase: "foo".to_string(), + context: FilterContext::Home, + irreversible: Some(true), + whole_word: None, + expires_in: None, + } + ) + } + + #[test] + fn test_whole_word() { + let mut request = AddFilterRequest::new("foo", FilterContext::Home); + request.whole_word(); + assert_eq!( + request, + AddFilterRequest { + phrase: "foo".to_string(), + context: FilterContext::Home, + irreversible: None, + whole_word: Some(true), + expires_in: None, + } + ) + } + + #[test] + fn test_expires_in() { + let mut request = AddFilterRequest::new("foo", FilterContext::Home); + request.expires_in(Duration::from_secs(300)); + assert_eq!( + request, + AddFilterRequest { + phrase: "foo".to_string(), + context: FilterContext::Home, + irreversible: None, + whole_word: None, + expires_in: Some(Duration::from_secs(300)), + } + ) + } + + #[test] + fn test_serialize_request() { + let mut request = AddFilterRequest::new("foo", FilterContext::Home); + request.expires_in(Duration::from_secs(300)); + let ser = serde_json::to_string(&request).expect("Couldn't serialize"); + assert_eq!( + ser, + r#"{"phrase":"foo","context":"home","irreversible":null,"whole_word":null,"expires_in":300}"# + ) + } +} diff --git a/src/requests/mod.rs b/src/requests/mod.rs new file mode 100644 index 0000000..375ebd1 --- /dev/null +++ b/src/requests/mod.rs @@ -0,0 +1,13 @@ +/// Data structure for the MastodonClient::add_filter method +pub use self::filter::AddFilterRequest; +/// Data structure for the MastodonClient::add_push_subscription method +pub use self::push::{AddPushRequest, Keys, UpdatePushRequest}; +/// Data structure for the MastodonClient::statuses method +pub use self::statuses::StatusesRequest; +/// Data structure for the MastodonClient::update_credentials method +pub use self::update_credentials::UpdateCredsRequest; + +mod filter; +mod push; +mod statuses; +mod update_credentials; diff --git a/src/requests/push.rs b/src/requests/push.rs new file mode 100644 index 0000000..c609b07 --- /dev/null +++ b/src/requests/push.rs @@ -0,0 +1,633 @@ +use entities::push::{add_subscription, update_data}; +use errors::Result; + +/// Container for the key & auth strings for an AddPushRequest +/// +/// # Example +/// +/// ``` +/// # extern crate elefren; +/// use elefren::requests::Keys; +/// +/// let keys = Keys::new("anetohias===", "oeatssah="); +/// ``` +#[derive(Debug, Default, Clone, PartialEq)] +pub struct Keys { + pub(crate) p256dh: String, + pub(crate) auth: String, +} + +impl Keys { + /// Create the `Keys` container + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::requests::Keys; + /// + /// let keys = Keys::new("anetohias===", "oeatssah="); + /// ``` + pub fn new(p256dh: &str, auth: &str) -> Keys { + Keys { + p256dh: p256dh.to_string(), + auth: auth.to_string(), + } + } +} + +/// Builder to pass to the Mastodon::add_push_subscription method +/// +/// # Example +/// +/// ```no_run +/// # extern crate elefren; +/// # use elefren::{MastodonClient, Mastodon, Data}; +/// # fn main() -> Result<(), elefren::Error> { +/// # let data = Data { +/// # base: "".into(), +/// # client_id: "".into(), +/// # client_secret: "".into(), +/// # redirect: "".into(), +/// # token: "".into(), +/// # }; +/// use elefren::requests::{AddPushRequest, Keys}; +/// +/// let client = Mastodon::from(data); +/// +/// let keys = Keys::new("stahesuahoei293ise===", "tasecoa,nmeozka=="); +/// let mut request = AddPushRequest::new("http://example.com/push/endpoint", &keys); +/// request.follow().reblog(); +/// +/// client.add_push_subscription(&request)?; +/// # Ok(()) +/// # } +/// ``` +#[derive(Debug, Default, Clone, PartialEq)] +pub struct AddPushRequest { + endpoint: String, + + p256dh: String, + auth: String, + + follow: Option, + favourite: Option, + reblog: Option, + mention: Option, +} + +impl AddPushRequest { + /// Construct a new AddPushRequest + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::requests::{AddPushRequest, Keys}; + /// let keys = Keys::new("abcdef===", "foobar=="); + /// let push_endpoint = "https://example.com/push/endpoint"; + /// let request = AddPushRequest::new(push_endpoint, &keys); + /// ``` + pub fn new(endpoint: &str, keys: &Keys) -> AddPushRequest { + AddPushRequest { + endpoint: endpoint.to_string(), + p256dh: keys.p256dh.clone(), + auth: keys.auth.clone(), + ..Default::default() + } + } + + /// A flag that indicates if you want follow notifications pushed + /// + /// # Example + /// ``` + /// # extern crate elefren; + /// use elefren::requests::{AddPushRequest, Keys}; + /// let keys = Keys::new("abcdef===", "foobar=="); + /// let push_endpoint = "https://example.com/push/endpoint"; + /// let mut request = AddPushRequest::new(push_endpoint, &keys); + /// request.follow(); + /// ``` + pub fn follow(&mut self) -> &mut Self { + self.follow = Some(true); + self + } + + /// A flag that indicates if you want favourite notifications pushed + /// + /// # Example + /// ``` + /// # extern crate elefren; + /// use elefren::requests::{AddPushRequest, Keys}; + /// let keys = Keys::new("abcdef===", "foobar=="); + /// let push_endpoint = "https://example.com/push/endpoint"; + /// let mut request = AddPushRequest::new(push_endpoint, &keys); + /// request.favourite(); + /// ``` + pub fn favourite(&mut self) -> &mut Self { + self.favourite = Some(true); + self + } + + /// A flag that indicates if you want reblog notifications pushed + /// + /// # Example + /// ``` + /// # extern crate elefren; + /// use elefren::requests::{AddPushRequest, Keys}; + /// let keys = Keys::new("abcdef===", "foobar=="); + /// let push_endpoint = "https://example.com/push/endpoint"; + /// let mut request = AddPushRequest::new(push_endpoint, &keys); + /// request.reblog(); + /// ``` + pub fn reblog(&mut self) -> &mut Self { + self.reblog = Some(true); + self + } + + /// A flag that indicates if you want mention notifications pushed + /// + /// # Example + /// ``` + /// # extern crate elefren; + /// use elefren::requests::{AddPushRequest, Keys}; + /// let keys = Keys::new("abcdef===", "foobar=="); + /// let push_endpoint = "https://example.com/push/endpoint"; + /// let mut request = AddPushRequest::new(push_endpoint, &keys); + /// request.mention(); + /// ``` + pub fn mention(&mut self) -> &mut Self { + self.mention = Some(true); + self + } + + fn flags_present(&self) -> bool { + self.follow.is_some() + || self.favourite.is_some() + || self.reblog.is_some() + || self.mention.is_some() + } + + pub(crate) fn build(&self) -> Result { + use entities::push::{ + add_subscription::{Data, Form, Keys, Subscription}, + Alerts, + }; + let mut form = Form { + subscription: Subscription { + endpoint: self.endpoint.clone(), + keys: Keys { + p256dh: self.p256dh.clone(), + auth: self.auth.clone(), + }, + }, + data: None, + }; + if self.flags_present() { + let mut alerts = Alerts::default(); + + if let Some(follow) = self.follow { + alerts.follow = Some(follow); + } + + if let Some(favourite) = self.favourite { + alerts.favourite = Some(favourite); + } + + if let Some(reblog) = self.reblog { + alerts.reblog = Some(reblog); + } + + if let Some(mention) = self.mention { + alerts.mention = Some(mention); + } + + form.data = Some(Data { + alerts: Some(alerts), + }); + } + Ok(form) + } +} + +/// Builder to pass to the Mastodon::update_push_data method +/// +/// # Example +/// +/// ```no_run +/// # extern crate elefren; +/// # use elefren::{MastodonClient, Mastodon, Data}; +/// # fn main() -> Result<(), elefren::Error> { +/// # let data = Data { +/// # base: "".into(), +/// # client_id: "".into(), +/// # client_secret: "".into(), +/// # redirect: "".into(), +/// # token: "".into(), +/// # }; +/// use elefren::requests::UpdatePushRequest; +/// +/// let client = Mastodon::from(data); +/// +/// let mut request = UpdatePushRequest::new("foobar"); +/// request.follow(true).reblog(true); +/// +/// client.update_push_data(&request)?; +/// # Ok(()) +/// # } +/// ``` +#[derive(Debug, Default, Clone, PartialEq, Serialize)] +pub struct UpdatePushRequest { + id: String, + follow: Option, + favourite: Option, + reblog: Option, + mention: Option, +} + +impl UpdatePushRequest { + /// Construct a new UpdatePushRequest + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::requests::UpdatePushRequest; + /// let request = UpdatePushRequest::new("some-id"); + /// ``` + pub fn new(id: &str) -> UpdatePushRequest { + UpdatePushRequest { + id: id.to_string(), + ..Default::default() + } + } + + /// A flag that indicates if you want follow notifications pushed + /// + /// # Example + /// ``` + /// # extern crate elefren; + /// use elefren::requests::UpdatePushRequest; + /// let mut request = UpdatePushRequest::new("foobar"); + /// request.follow(true); + /// ``` + pub fn follow(&mut self, follow: bool) -> &mut Self { + self.follow = Some(follow); + self + } + + /// A flag that indicates if you want favourite notifications pushed + /// + /// # Example + /// ``` + /// # extern crate elefren; + /// use elefren::requests::UpdatePushRequest; + /// let mut request = UpdatePushRequest::new("foobar"); + /// request.favourite(true); + /// ``` + pub fn favourite(&mut self, favourite: bool) -> &mut Self { + self.favourite = Some(favourite); + self + } + + /// A flag that indicates if you want reblog notifications pushed + /// + /// # Example + /// ``` + /// # extern crate elefren; + /// use elefren::requests::UpdatePushRequest; + /// let mut request = UpdatePushRequest::new("foobar"); + /// request.reblog(true); + /// ``` + pub fn reblog(&mut self, reblog: bool) -> &mut Self { + self.reblog = Some(reblog); + self + } + + /// A flag that indicates if you want mention notifications pushed + /// + /// # Example + /// ``` + /// # extern crate elefren; + /// use elefren::requests::UpdatePushRequest; + /// let mut request = UpdatePushRequest::new("foobar"); + /// request.mention(true); + /// ``` + pub fn mention(&mut self, mention: bool) -> &mut Self { + self.mention = Some(mention); + self + } + + fn flags_present(&self) -> bool { + self.follow.is_some() + || self.favourite.is_some() + || self.reblog.is_some() + || self.mention.is_some() + } + + pub(crate) fn build(&self) -> update_data::Form { + use entities::push::{ + update_data::{Data, Form}, + Alerts, + }; + + let mut form = Form { + id: self.id.clone(), + ..Default::default() + }; + + if self.flags_present() { + let mut alerts = Alerts::default(); + if let Some(follow) = self.follow { + alerts.follow = Some(follow); + } + if let Some(favourite) = self.favourite { + alerts.favourite = Some(favourite); + } + if let Some(reblog) = self.reblog { + alerts.reblog = Some(reblog); + } + if let Some(mention) = self.mention { + alerts.mention = Some(mention); + } + form.data = Data { + alerts: Some(alerts), + }; + } + form + } +} + +#[cfg(test)] +mod tests { + use super::*; + use entities::push::{add_subscription, update_data, Alerts}; + + #[test] + fn test_keys_new() { + let keys = Keys::new("anetohias===", "oeatssah="); + assert_eq!( + keys, + Keys { + p256dh: "anetohias===".to_string(), + auth: "oeatssah=".to_string() + } + ); + } + + #[test] + fn test_add_push_request_new() { + let endpoint = "https://example.com/push/endpoint"; + let keys = Keys::new("anetohias===", "oeatssah="); + let req = AddPushRequest::new(endpoint, &keys); + assert_eq!( + req, + AddPushRequest { + endpoint: "https://example.com/push/endpoint".to_string(), + p256dh: "anetohias===".to_string(), + auth: "oeatssah=".to_string(), + follow: None, + favourite: None, + reblog: None, + mention: None, + } + ); + } + #[test] + fn test_add_push_request_follow() { + let endpoint = "https://example.com/push/endpoint"; + let keys = Keys::new("anetohias===", "oeatssah="); + let mut req = AddPushRequest::new(endpoint, &keys); + req.follow(); + assert_eq!( + req, + AddPushRequest { + endpoint: "https://example.com/push/endpoint".to_string(), + p256dh: "anetohias===".to_string(), + auth: "oeatssah=".to_string(), + follow: Some(true), + favourite: None, + reblog: None, + mention: None, + } + ); + } + + #[test] + fn test_add_push_request_favourite() { + let endpoint = "https://example.com/push/endpoint"; + let keys = Keys::new("anetohias===", "oeatssah="); + let mut req = AddPushRequest::new(endpoint, &keys); + req.favourite(); + assert_eq!( + req, + AddPushRequest { + endpoint: "https://example.com/push/endpoint".to_string(), + p256dh: "anetohias===".to_string(), + auth: "oeatssah=".to_string(), + follow: None, + favourite: Some(true), + reblog: None, + mention: None, + } + ); + } + #[test] + fn test_add_push_request_reblog() { + let endpoint = "https://example.com/push/endpoint"; + let keys = Keys::new("anetohias===", "oeatssah="); + let mut req = AddPushRequest::new(endpoint, &keys); + req.reblog(); + assert_eq!( + req, + AddPushRequest { + endpoint: "https://example.com/push/endpoint".to_string(), + p256dh: "anetohias===".to_string(), + auth: "oeatssah=".to_string(), + follow: None, + favourite: None, + reblog: Some(true), + mention: None, + } + ); + } + #[test] + fn test_add_push_request_mention() { + let endpoint = "https://example.com/push/endpoint"; + let keys = Keys::new("anetohias===", "oeatssah="); + let mut req = AddPushRequest::new(endpoint, &keys); + req.mention(); + assert_eq!( + req, + AddPushRequest { + endpoint: "https://example.com/push/endpoint".to_string(), + p256dh: "anetohias===".to_string(), + auth: "oeatssah=".to_string(), + follow: None, + favourite: None, + reblog: None, + mention: Some(true), + } + ); + } + #[test] + fn test_add_push_request_build_no_flags() { + let endpoint = "https://example.com/push/endpoint"; + let keys = Keys::new("anetohias===", "oeatssah="); + let req = AddPushRequest::new(endpoint, &keys); + let form = req.build().expect("Couldn't build form"); + assert_eq!( + form, + add_subscription::Form { + subscription: add_subscription::Subscription { + endpoint: "https://example.com/push/endpoint".to_string(), + keys: add_subscription::Keys { + p256dh: "anetohias===".to_string(), + auth: "oeatssah=".to_string(), + }, + }, + data: None, + } + ); + } + + #[test] + fn test_add_push_request_build() { + let endpoint = "https://example.com/push/endpoint"; + let keys = Keys::new("anetohias===", "oeatssah="); + let mut req = AddPushRequest::new(endpoint, &keys); + req.follow().reblog(); + let form = req.build().expect("Couldn't build form"); + assert_eq!( + form, + add_subscription::Form { + subscription: add_subscription::Subscription { + endpoint: "https://example.com/push/endpoint".to_string(), + keys: add_subscription::Keys { + p256dh: "anetohias===".to_string(), + auth: "oeatssah=".to_string(), + }, + }, + data: Some(add_subscription::Data { + alerts: Some(Alerts { + follow: Some(true), + favourite: None, + reblog: Some(true), + mention: None, + }), + }), + } + ); + } + + #[test] + fn test_update_push_request_new() { + let req = UpdatePushRequest::new("some-id"); + assert_eq!( + req, + UpdatePushRequest { + id: "some-id".to_string(), + follow: None, + favourite: None, + reblog: None, + mention: None, + } + ); + } + + #[test] + fn test_update_push_request_follow() { + let mut req = UpdatePushRequest::new("some-id"); + req.follow(true); + assert_eq!( + req, + UpdatePushRequest { + id: "some-id".to_string(), + follow: Some(true), + favourite: None, + reblog: None, + mention: None, + } + ); + } + #[test] + fn test_update_push_request_favourite() { + let mut req = UpdatePushRequest::new("some-id"); + req.favourite(true); + assert_eq!( + req, + UpdatePushRequest { + id: "some-id".to_string(), + follow: None, + favourite: Some(true), + reblog: None, + mention: None, + } + ); + } + #[test] + fn test_update_push_request_reblog() { + let mut req = UpdatePushRequest::new("some-id"); + req.reblog(true); + assert_eq!( + req, + UpdatePushRequest { + id: "some-id".to_string(), + follow: None, + favourite: None, + reblog: Some(true), + mention: None, + } + ); + } + #[test] + fn test_update_push_request_mention() { + let mut req = UpdatePushRequest::new("some-id"); + req.mention(true); + assert_eq!( + req, + UpdatePushRequest { + id: "some-id".to_string(), + follow: None, + favourite: None, + reblog: None, + mention: Some(true), + } + ); + } + #[test] + fn test_update_push_request_build_no_flags() { + let req = UpdatePushRequest::new("some-id"); + let form = req.build(); + assert_eq!( + form, + update_data::Form { + id: "some-id".to_string(), + data: update_data::Data { + alerts: None, + }, + } + ); + } + + #[test] + fn test_update_push_request_build() { + let mut req = UpdatePushRequest::new("some-id"); + req.favourite(false); + let form = req.build(); + assert_eq!( + form, + update_data::Form { + id: "some-id".to_string(), + data: update_data::Data { + alerts: Some(Alerts { + follow: None, + favourite: Some(false), + reblog: None, + mention: None, + }), + }, + } + ); + } +} diff --git a/src/requests/statuses.rs b/src/requests/statuses.rs new file mode 100644 index 0000000..6e3da08 --- /dev/null +++ b/src/requests/statuses.rs @@ -0,0 +1,614 @@ +use errors::Error; +use serde_qs; +use std::{borrow::Cow, convert::Into}; + +mod bool_qs_serialize { + use serde::Serializer; + + pub fn is_false(b: &bool) -> bool { + !*b + } + + pub fn serialize(b: &bool, s: S) -> Result { + if *b { + s.serialize_i64(1) + } else { + s.serialize_i64(0) + } + } +} + +/// Builder for making a client.statuses() call +/// +/// # Example +/// +/// ``` +/// # extern crate elefren; +/// # use elefren::StatusesRequest; +/// let mut request = StatusesRequest::new(); +/// request.only_media().pinned().since_id("foo"); +/// # assert_eq!(&request.to_querystring().expect("Couldn't serialize qs")[..], "?only_media=1&pinned=1&since_id=foo"); +/// ``` +#[derive(Clone, Debug, Default, PartialEq, Serialize)] +pub struct StatusesRequest<'a> { + #[serde(skip_serializing_if = "bool_qs_serialize::is_false")] + #[serde(serialize_with = "bool_qs_serialize::serialize")] + only_media: bool, + #[serde(skip_serializing_if = "bool_qs_serialize::is_false")] + #[serde(serialize_with = "bool_qs_serialize::serialize")] + exclude_replies: bool, + #[serde(skip_serializing_if = "bool_qs_serialize::is_false")] + #[serde(serialize_with = "bool_qs_serialize::serialize")] + pinned: bool, + #[serde(skip_serializing_if = "Option::is_none")] + max_id: Option>, + #[serde(skip_serializing_if = "Option::is_none")] + since_id: Option>, + #[serde(skip_serializing_if = "Option::is_none")] + limit: Option, + #[serde(skip_serializing_if = "Option::is_none")] + min_id: Option>, +} + +impl<'a> Into>> for &'a mut StatusesRequest<'a> { + fn into(self) -> Option> { + Some(StatusesRequest { + only_media: self.only_media, + exclude_replies: self.exclude_replies, + pinned: self.pinned, + max_id: self.max_id.clone(), + since_id: self.since_id.clone(), + limit: self.limit.clone(), + min_id: self.min_id.clone(), + }) + } +} + +impl<'a> StatusesRequest<'a> { + /// Construct a new `StatusesRequest` object + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// # use elefren::StatusesRequest; + /// let request = StatusesRequest::new(); + /// ``` + pub fn new() -> Self { + Self::default() + } + + /// Set the `?only_media=1` flag for the .statuses() request + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// # use elefren::StatusesRequest; + /// let mut request = StatusesRequest::new(); + /// assert_eq!(&request.only_media().to_querystring().expect("Couldn't serialize qs"), "?only_media=1"); + pub fn only_media(&mut self) -> &mut Self { + self.only_media = true; + self + } + + /// Set the `?exclude_replies=1` flag for the .statuses() request + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// # use elefren::StatusesRequest; + /// let mut request = StatusesRequest::new(); + /// assert_eq!( + /// &request + /// .exclude_replies() + /// .to_querystring() + /// .expect("Couldn't serialize qs"), + /// "?exclude_replies=1" + /// ); + /// ``` + pub fn exclude_replies(&mut self) -> &mut Self { + self.exclude_replies = true; + self + } + + /// Set the `?pinned=1` flag for the .statuses() request + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// # use elefren::StatusesRequest; + /// let mut request = StatusesRequest::new(); + /// assert_eq!( + /// &request + /// .pinned() + /// .to_querystring() + /// .expect("Couldn't serialize qs"), + /// "?pinned=1" + /// ); + /// ``` + pub fn pinned(&mut self) -> &mut Self { + self.pinned = true; + self + } + + /// Set the `?max_id=:max_id` flag for the .statuses() request + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// # use elefren::StatusesRequest; + /// let mut request = StatusesRequest::new(); + /// assert_eq!( + /// &request + /// .max_id("foo") + /// .to_querystring() + /// .expect("Couldn't serialize qs"), + /// "?max_id=foo" + /// ); + /// ``` + pub fn max_id>>(&mut self, max_id: S) -> &mut Self { + self.max_id = Some(max_id.into()); + self + } + + /// Set the `?since_id=:since_id` flag for the .statuses() request + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// # use elefren::StatusesRequest; + /// let mut request = StatusesRequest::new(); + /// assert_eq!( + /// &request + /// .since_id("foo") + /// .to_querystring() + /// .expect("Couldn't serialize qs"), + /// "?since_id=foo" + /// ); + /// ``` + pub fn since_id>>(&mut self, since_id: S) -> &mut Self { + self.since_id = Some(since_id.into()); + self + } + + /// Set the `?limit=:limit` flag for the .statuses() request + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// # use elefren::StatusesRequest; + /// let mut request = StatusesRequest::new(); + /// assert_eq!( + /// &request + /// .limit(10) + /// .to_querystring() + /// .expect("Couldn't serialize qs"), + /// "?limit=10" + /// ); + /// ``` + pub fn limit(&mut self, limit: usize) -> &mut Self { + self.limit = Some(limit); + self + } + + /// Set the `?min_id=:min_id` flag for the .statuses() request + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// # use elefren::StatusesRequest; + /// let mut request = StatusesRequest::new(); + /// assert_eq!( + /// &request + /// .min_id("foobar") + /// .to_querystring() + /// .expect("Couldn't serialize qs"), + /// "?min_id=foobar" + /// ); + /// ``` + pub fn min_id>>(&mut self, min_id: S) -> &mut Self { + self.min_id = Some(min_id.into()); + self + } + + /// Turns this builder into a querystring + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// # use elefren::StatusesRequest; + /// let mut request = StatusesRequest::new(); + /// assert_eq!( + /// &request + /// .limit(10) + /// .pinned() + /// .to_querystring() + /// .expect("Couldn't serialize qs"), + /// "?pinned=1&limit=10" + /// ); + /// ``` + pub fn to_querystring(&self) -> Result { + Ok(format!("?{}", serde_qs::to_string(&self)?)) + } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn test_new() { + let request = StatusesRequest::new(); + assert_eq!( + request, + StatusesRequest { + only_media: false, + exclude_replies: false, + pinned: false, + max_id: None, + since_id: None, + limit: None, + min_id: None, + } + ); + } + + #[test] + fn test_only_media() { + let mut request = StatusesRequest::new(); + request.only_media(); + assert_eq!( + request, + StatusesRequest { + only_media: true, + exclude_replies: false, + pinned: false, + max_id: None, + since_id: None, + limit: None, + min_id: None, + } + ); + } + + #[test] + fn test_exclude_replies() { + let mut request = StatusesRequest::new(); + request.exclude_replies(); + assert_eq!( + request, + StatusesRequest { + only_media: false, + exclude_replies: true, + pinned: false, + max_id: None, + since_id: None, + limit: None, + min_id: None, + } + ); + } + #[test] + fn test_pinned() { + let mut request = StatusesRequest::new(); + request.pinned(); + assert_eq!( + request, + StatusesRequest { + only_media: false, + exclude_replies: false, + pinned: true, + max_id: None, + since_id: None, + limit: None, + min_id: None, + } + ); + } + #[test] + fn test_max_id() { + let mut request = StatusesRequest::new(); + request.max_id("foo"); + assert_eq!( + request, + StatusesRequest { + only_media: false, + exclude_replies: false, + pinned: false, + max_id: Some("foo".into()), + since_id: None, + limit: None, + min_id: None, + } + ); + } + #[test] + fn test_since_id() { + let mut request = StatusesRequest::new(); + request.since_id("foo"); + assert_eq!( + request, + StatusesRequest { + only_media: false, + exclude_replies: false, + pinned: false, + max_id: None, + since_id: Some("foo".into()), + limit: None, + min_id: None, + } + ); + } + #[test] + fn test_limit() { + let mut request = StatusesRequest::new(); + request.limit(42); + assert_eq!( + request, + StatusesRequest { + only_media: false, + exclude_replies: false, + pinned: false, + max_id: None, + since_id: None, + limit: Some(42), + min_id: None, + } + ); + } + #[test] + fn test_min_id() { + let mut request = StatusesRequest::new(); + request.min_id("foo"); + assert_eq!( + request, + StatusesRequest { + only_media: false, + exclude_replies: false, + pinned: false, + max_id: None, + since_id: None, + limit: None, + min_id: Some("foo".into()), + } + ); + } + #[test] + fn test_to_querystring() { + macro_rules! qs_test { + (|$r:ident| $b:block, $expected:expr) => { + { + let mut $r = StatusesRequest::new(); + $b + let qs = $r.to_querystring().expect("Failed to serialize querystring"); + assert_eq!(&qs, $expected); + } + } + } + + qs_test!( + |request| { + request.only_media(); + }, + "?only_media=1" + ); + qs_test!( + |request| { + request.exclude_replies(); + }, + "?exclude_replies=1" + ); + qs_test!( + |request| { + request.pinned(); + }, + "?pinned=1" + ); + qs_test!( + |request| { + request.max_id("foo"); + }, + "?max_id=foo" + ); + qs_test!( + |request| { + request.since_id("foo"); + }, + "?since_id=foo" + ); + qs_test!( + |request| { + request.limit(42); + }, + "?limit=42" + ); + qs_test!( + |request| { + request.only_media().exclude_replies(); + }, + "?only_media=1&exclude_replies=1" + ); + qs_test!( + |request| { + request.only_media().pinned(); + }, + "?only_media=1&pinned=1" + ); + qs_test!( + |request| { + request.only_media().max_id("foo"); + }, + "?only_media=1&max_id=foo" + ); + qs_test!( + |request| { + request.only_media().since_id("foo"); + }, + "?only_media=1&since_id=foo" + ); + qs_test!( + |request| { + request.only_media().limit(42); + }, + "?only_media=1&limit=42" + ); + qs_test!( + |request| { + request.exclude_replies().only_media(); + }, + "?only_media=1&exclude_replies=1" + ); + qs_test!( + |request| { + request.exclude_replies().pinned(); + }, + "?exclude_replies=1&pinned=1" + ); + qs_test!( + |request| { + request.exclude_replies().max_id("foo"); + }, + "?exclude_replies=1&max_id=foo" + ); + qs_test!( + |request| { + request.exclude_replies().since_id("foo"); + }, + "?exclude_replies=1&since_id=foo" + ); + qs_test!( + |request| { + request.exclude_replies().limit(42); + }, + "?exclude_replies=1&limit=42" + ); + qs_test!( + |request| { + request.pinned().only_media(); + }, + "?only_media=1&pinned=1" + ); + qs_test!( + |request| { + request.pinned().exclude_replies(); + }, + "?exclude_replies=1&pinned=1" + ); + qs_test!( + |request| { + request.pinned().max_id("foo"); + }, + "?pinned=1&max_id=foo" + ); + qs_test!( + |request| { + request.pinned().since_id("foo"); + }, + "?pinned=1&since_id=foo" + ); + qs_test!( + |request| { + request.pinned().limit(42); + }, + "?pinned=1&limit=42" + ); + qs_test!( + |request| { + request.max_id("foo").only_media(); + }, + "?only_media=1&max_id=foo" + ); + qs_test!( + |request| { + request.max_id("foo").exclude_replies(); + }, + "?exclude_replies=1&max_id=foo" + ); + qs_test!( + |request| { + request.max_id("foo").pinned(); + }, + "?pinned=1&max_id=foo" + ); + qs_test!( + |request| { + request.max_id("foo").since_id("foo"); + }, + "?max_id=foo&since_id=foo" + ); + qs_test!( + |request| { + request.max_id("foo").limit(42); + }, + "?max_id=foo&limit=42" + ); + qs_test!( + |request| { + request.since_id("foo").only_media(); + }, + "?only_media=1&since_id=foo" + ); + qs_test!( + |request| { + request.since_id("foo").exclude_replies(); + }, + "?exclude_replies=1&since_id=foo" + ); + qs_test!( + |request| { + request.since_id("foo").pinned(); + }, + "?pinned=1&since_id=foo" + ); + qs_test!( + |request| { + request.since_id("foo").max_id("foo"); + }, + "?max_id=foo&since_id=foo" + ); + qs_test!( + |request| { + request.since_id("foo").limit(42); + }, + "?since_id=foo&limit=42" + ); + qs_test!( + |request| { + request.limit(42).only_media(); + }, + "?only_media=1&limit=42" + ); + qs_test!( + |request| { + request.limit(42).exclude_replies(); + }, + "?exclude_replies=1&limit=42" + ); + qs_test!( + |request| { + request.limit(42).pinned(); + }, + "?pinned=1&limit=42" + ); + qs_test!( + |request| { + request.limit(42).max_id("foo"); + }, + "?max_id=foo&limit=42" + ); + qs_test!( + |request| { + request.limit(42).since_id("foo"); + }, + "?since_id=foo&limit=42" + ); + } +} diff --git a/src/requests/update_credentials.rs b/src/requests/update_credentials.rs new file mode 100644 index 0000000..7784cc3 --- /dev/null +++ b/src/requests/update_credentials.rs @@ -0,0 +1,327 @@ +use std::{ + fmt::Display, + path::{Path, PathBuf}, +}; + +use entities::account::{Credentials, MetadataField, UpdateSource}; +use errors::Result; +use status_builder; + +/// Builder to pass to the Mastodon::update_credentials method +/// +/// # Example +/// +/// ```no_run +/// # extern crate elefren; +/// # use elefren::Data; +/// # fn main() -> Result<(), elefren::Error> { +/// # let data = Data { +/// # base: "".into(), +/// # client_id: "".into(), +/// # client_secret: "".into(), +/// # redirect: "".into(), +/// # token: "".into(), +/// # }; +/// use elefren::{prelude::*, status_builder::Visibility, UpdateCredsRequest}; +/// +/// let client = Mastodon::from(data); +/// let mut builder = UpdateCredsRequest::new(); +/// +/// builder.privacy(Visibility::Unlisted); +/// +/// let result = client.update_credentials(&mut builder)?; +/// # Ok(()) +/// # } +/// ``` +#[derive(Debug, Default, Clone, PartialEq)] +pub struct UpdateCredsRequest { + display_name: Option, + note: Option, + avatar: Option, + header: Option, + field_attributes: Vec, + + // UpdateSource fields + privacy: Option, + sensitive: Option, +} + +impl UpdateCredsRequest { + /// Create a new UpdateCredsRequest + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::UpdateCredsRequest; + /// + /// let mut builder = UpdateCredsRequest::new(); + /// ``` + pub fn new() -> UpdateCredsRequest { + Default::default() + } + + /// Set the new display_name value + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::UpdateCredsRequest; + /// + /// let mut builder = UpdateCredsRequest::new(); + /// + /// builder.display_name("my new display name"); + /// ``` + pub fn display_name(&mut self, name: D) -> &mut Self { + self.display_name = Some(name.to_string()); + self + } + + /// Set the new note value + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::UpdateCredsRequest; + /// + /// let mut builder = UpdateCredsRequest::new(); + /// + /// builder.note("my new note"); + /// ``` + pub fn note(&mut self, note: D) -> &mut Self { + self.note = Some(note.to_string()); + self + } + + /// Set the new avatar value + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::UpdateCredsRequest; + /// + /// let mut builder = UpdateCredsRequest::new(); + /// + /// builder.avatar("/path/to/my/new/avatar"); + /// ``` + pub fn avatar>(&mut self, path: P) -> &mut Self { + let path = path.as_ref(); + let path = path.to_path_buf(); + self.avatar = Some(path); + self + } + + /// Set the new header value + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::UpdateCredsRequest; + /// + /// let mut builder = UpdateCredsRequest::new(); + /// + /// builder.header("/path/to/my/new/header"); + /// ``` + pub fn header>(&mut self, path: P) -> &mut Self { + let path = path.as_ref(); + let path = path.to_path_buf(); + self.header = Some(path); + self + } + + /// Set the new privacy value + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::{status_builder::Visibility, UpdateCredsRequest}; + /// + /// let mut builder = UpdateCredsRequest::new(); + /// + /// builder.privacy(Visibility::Public); + /// ``` + pub fn privacy(&mut self, privacy: status_builder::Visibility) -> &mut Self { + self.privacy = Some(privacy); + self + } + + /// Set the new sensitive value + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::UpdateCredsRequest; + /// + /// let mut builder = UpdateCredsRequest::new(); + /// + /// builder.sensitive(true); + /// ``` + pub fn sensitive(&mut self, sensitive: bool) -> &mut Self { + self.sensitive = Some(sensitive); + self + } + + /// Add a metadata field + /// + /// # Example + /// + /// ``` + /// # extern crate elefren; + /// use elefren::UpdateCredsRequest; + /// + /// let mut builder = UpdateCredsRequest::new(); + /// + /// builder.field_attribute("some key", "some value"); + /// ``` + pub fn field_attribute(&mut self, name: &str, value: &str) -> &mut Self { + self.field_attributes.push(MetadataField::new(name, value)); + self + } + + pub(crate) fn build(&mut self) -> Result { + Ok(Credentials { + display_name: self.display_name.clone(), + note: self.note.clone(), + avatar: self.avatar.clone(), + header: self.avatar.clone(), + source: Some(UpdateSource { + privacy: self.privacy.clone(), + sensitive: self.sensitive.clone(), + }), + fields_attributes: self.field_attributes.clone(), + }) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use entities::account::{Credentials, MetadataField, UpdateSource}; + use status_builder::Visibility; + + #[test] + fn test_update_creds_request_new() { + let builder = UpdateCredsRequest::new(); + assert_eq!( + builder, + UpdateCredsRequest { + ..Default::default() + } + ); + } + + #[test] + fn test_update_creds_request_display_name() { + let mut builder = UpdateCredsRequest::new(); + builder.display_name("foo"); + assert_eq!( + builder, + UpdateCredsRequest { + display_name: Some("foo".into()), + ..Default::default() + } + ); + } + + #[test] + fn test_update_creds_request_note() { + let mut builder = UpdateCredsRequest::new(); + builder.note("foo"); + assert_eq!( + builder, + UpdateCredsRequest { + note: Some("foo".into()), + ..Default::default() + } + ); + } + + #[test] + fn test_update_creds_request_avatar() { + let mut builder = UpdateCredsRequest::new(); + builder.avatar("/path/to/avatar.png"); + assert_eq!( + builder, + UpdateCredsRequest { + avatar: Some(Path::new("/path/to/avatar.png").to_path_buf()), + ..Default::default() + } + ); + } + + #[test] + fn test_update_creds_request_header() { + let mut builder = UpdateCredsRequest::new(); + builder.header("/path/to/header.png"); + assert_eq!( + builder, + UpdateCredsRequest { + header: Some(Path::new("/path/to/header.png").to_path_buf()), + ..Default::default() + } + ); + } + + #[test] + fn test_update_creds_request_privacy() { + let mut builder = UpdateCredsRequest::new(); + builder.privacy(Visibility::Public); + assert_eq!( + builder, + UpdateCredsRequest { + privacy: Some(Visibility::Public), + ..Default::default() + } + ); + } + + #[test] + fn test_update_creds_request_sensitive() { + let mut builder = UpdateCredsRequest::new(); + builder.sensitive(true); + assert_eq!( + builder, + UpdateCredsRequest { + sensitive: Some(true), + ..Default::default() + } + ); + } + + #[test] + fn test_update_creds_request_field_attribute() { + let mut builder = UpdateCredsRequest::new(); + builder.field_attribute("foo", "bar"); + assert_eq!( + builder, + UpdateCredsRequest { + field_attributes: vec![MetadataField::new("foo", "bar")], + ..Default::default() + } + ); + } + + #[test] + fn test_update_creds_request_build() { + let mut builder = UpdateCredsRequest::new(); + builder.display_name("test").note("a note"); + let creds = builder.build().expect("Couldn't build Credentials"); + assert_eq!( + creds, + Credentials { + display_name: Some("test".into()), + note: Some("a note".into()), + source: Some(UpdateSource { + ..Default::default() + }), + ..Default::default() + } + ); + } +} diff --git a/src/scopes.rs b/src/scopes.rs new file mode 100644 index 0000000..2cab0d2 --- /dev/null +++ b/src/scopes.rs @@ -0,0 +1,818 @@ +use std::{ + cmp::{Ordering, PartialEq, PartialOrd}, + collections::HashSet, + fmt, + ops::BitOr, + str::FromStr, +}; + +use serde::ser::{Serialize, Serializer}; + +use errors::Error; +use serde::{Deserialize, Deserializer}; +use serde::de::{self, Visitor}; + +/// Represents a set of OAuth scopes +/// +/// # Example +/// +/// ```rust +/// use elefren::prelude::*; +/// +/// let read = Scopes::read_all(); +/// let write = Scopes::write_all(); +/// let follow = Scopes::follow(); +/// let all = read | write | follow; +/// ``` +#[derive(Clone)] +pub struct Scopes { + scopes: HashSet, +} + +impl FromStr for Scopes { + type Err = Error; + + fn from_str(s: &str) -> Result { + let mut set = HashSet::new(); + for scope in s.split_whitespace() { + let scope = Scope::from_str(&scope)?; + set.insert(scope); + } + Ok(Scopes { + scopes: set, + }) + } +} + +impl Serialize for Scopes { + fn serialize(&self, serializer: S) -> ::std::result::Result + where + S: Serializer, + { + let repr = format!("{}", self); + serializer.serialize_str(&repr) + } +} + +struct DeserializeScopesVisitor; + +impl<'de> Visitor<'de> for DeserializeScopesVisitor { + type Value = Scopes; + + fn expecting(&self, formatter: &mut fmt::Formatter) -> Result<(), fmt::Error> { + write!(formatter, "space separated scopes") + } + + fn visit_str(self, v: &str) -> Result + where E: de::Error + { + Scopes::from_str(v).map_err(de::Error::custom) + } +} + +impl<'de> Deserialize<'de> for Scopes { + fn deserialize(deserializer: D) -> Result>::Error> where + D: Deserializer<'de> { + deserializer.deserialize_str(DeserializeScopesVisitor) + } +} + +impl Scopes { + /// Represents all available oauth scopes: "read write follow push" + /// + /// ``` + /// # extern crate elefren; + /// # use std::error::Error; + /// use elefren::scopes::Scopes; + /// + /// # fn main() -> Result<(), Box> { + /// let scope = Scopes::all(); + /// assert_eq!(&format!("{}", scope), "read write follow push"); + /// # Ok(()) + /// # } + /// ``` + pub fn all() -> Scopes { + Scopes::read_all() | Scopes::write_all() | Scopes::follow() | Scopes::push() + } + + /// Represents the full "read" scope + /// + /// ``` + /// # extern crate elefren; + /// # use std::error::Error; + /// use elefren::scopes::Scopes; + /// + /// # fn main() -> Result<(), Box> { + /// let scope = Scopes::read_all(); + /// assert_eq!(&format!("{}", scope), "read"); + /// # Ok(()) + /// # } + /// ``` + pub fn read_all() -> Scopes { + Scopes::_read(None) + } + + /// Represents a specific "read:___" scope + /// + /// ``` + /// # extern crate elefren; + /// # use std::error::Error; + /// use elefren::scopes::{Read, Scopes}; + /// + /// # fn main() -> Result<(), Box> { + /// let scope = Scopes::read(Read::Accounts); + /// assert_eq!(&format!("{}", scope), "read:accounts"); + /// # Ok(()) + /// # } + /// ``` + pub fn read(subscope: Read) -> Scopes { + Scopes::_read(Some(subscope)) + } + + /// Represents the full "write" scope + /// + /// ``` + /// # extern crate elefren; + /// # use std::error::Error; + /// use elefren::scopes::Scopes; + /// + /// # fn main() -> Result<(), Box> { + /// let scope = Scopes::write_all(); + /// assert_eq!(&format!("{}", scope), "write"); + /// # Ok(()) + /// # } + /// ``` + pub fn write_all() -> Scopes { + Scopes::_write(None) + } + + /// Represents a specific "write:___" scope + /// + /// ``` + /// # extern crate elefren; + /// # use std::error::Error; + /// use elefren::scopes::{Scopes, Write}; + /// + /// # fn main() -> Result<(), Box> { + /// let scope = Scopes::write(Write::Accounts); + /// assert_eq!(&format!("{}", scope), "write:accounts"); + /// # Ok(()) + /// # } + /// ``` + pub fn write(subscope: Write) -> Scopes { + Scopes::_write(Some(subscope)) + } + + /// Represents the "follow" scope + /// + /// ``` + /// # extern crate elefren; + /// # use std::error::Error; + /// use elefren::scopes::Scopes; + /// + /// # fn main() -> Result<(), Box> { + /// let scope = Scopes::follow(); + /// assert_eq!(&format!("{}", scope), "follow"); + /// # Ok(()) + /// # } + /// ``` + pub fn follow() -> Scopes { + Scopes::new(Scope::Follow) + } + + /// Represents the full "push" scope + /// + /// ``` + /// # extern crate elefren; + /// # use std::error::Error; + /// use elefren::scopes::Scopes; + /// + /// # fn main() -> Result<(), Box> { + /// let scope = Scopes::push(); + /// assert_eq!(&format!("{}", scope), "push"); + /// # Ok(()) + /// # } + /// ``` + pub fn push() -> Scopes { + Scopes::new(Scope::Push) + } + + /// Combines 2 scopes together + /// + /// # Example + /// + /// ```rust + /// use elefren::prelude::*; + /// + /// let read = Scopes::read_all(); + /// let write = Scopes::write_all(); + /// let read_write = read.and(write); + /// ``` + pub fn and(self, other: Scopes) -> Scopes { + let newset: HashSet<_> = self + .scopes + .union(&other.scopes) + .into_iter() + .map(|s| *s) + .collect(); + Scopes { + scopes: newset, + } + } + + fn _write(subscope: Option) -> Scopes { + Scopes::new(Scope::Write(subscope)) + } + + fn _read(subscope: Option) -> Scopes { + Scopes::new(Scope::Read(subscope)) + } + + fn new(scope: Scope) -> Scopes { + let mut set = HashSet::new(); + set.insert(scope); + Scopes { + scopes: set, + } + } +} + +impl BitOr for Scopes { + type Output = Scopes; + + fn bitor(self, other: Scopes) -> Self::Output { + self.and(other) + } +} + +impl PartialEq for Scopes { + fn eq(&self, other: &Scopes) -> bool { + self.scopes + .symmetric_difference(&other.scopes) + .next() + .is_none() + } +} + +impl Default for Scopes { + fn default() -> Scopes { + Scopes::read_all() + } +} + +impl fmt::Debug for Scopes { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + write!(f, "[")?; + for scope in &self.scopes { + write!(f, "{:?}", &scope)?; + } + Ok(write!(f, "]")?) + } +} + +impl fmt::Display for Scopes { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + let mut start = true; + let scopes = { + let mut scopes = self.scopes.iter().collect::>(); + scopes.sort(); + scopes + }; + for scope in &scopes { + if !start { + write!(f, " ")?; + } else { + start = false; + } + write!(f, "{}", &scope)?; + } + Ok(()) + } +} + +/// Permission scope of the application. +/// [Details on what each permission provides][1] +/// [1]: https://github.com/tootsuite/documentation/blob/master/Using-the-API/OAuth-details.md) +#[derive(Debug, Clone, Copy, PartialEq, PartialOrd, Eq, Hash, Serialize)] +enum Scope { + /// Read only permissions. + #[serde(rename = "read")] + Read(Option), + /// Write only permissions. + #[serde(rename = "write")] + Write(Option), + /// Only permission to add and remove followers. + #[serde(rename = "follow")] + Follow, + /// Push permissions + #[serde(rename = "push")] + Push, +} + +impl FromStr for Scope { + type Err = Error; + + fn from_str(s: &str) -> Result { + Ok(match s { + "read" => Scope::Read(None), + "write" => Scope::Write(None), + "follow" => Scope::Follow, + "push" => Scope::Push, + read if read.starts_with("read:") => { + let r: Read = Read::from_str(&read[5..])?; + Scope::Read(Some(r)) + }, + write if write.starts_with("write:") => { + let w: Write = Write::from_str(&write[6..])?; + Scope::Write(Some(w)) + }, + _ => return Err(Error::Other("Unknown scope".to_string())), + }) + } +} + +impl Ord for Scope { + fn cmp(&self, other: &Scope) -> Ordering { + match (*self, *other) { + (Scope::Read(None), Scope::Read(None)) => Ordering::Equal, + (Scope::Read(None), Scope::Read(Some(..))) => Ordering::Less, + (Scope::Read(Some(..)), Scope::Read(None)) => Ordering::Greater, + (Scope::Read(Some(ref a)), Scope::Read(Some(ref b))) => a.cmp(b), + + (Scope::Write(None), Scope::Write(None)) => Ordering::Equal, + (Scope::Write(None), Scope::Write(Some(..))) => Ordering::Less, + (Scope::Write(Some(..)), Scope::Write(None)) => Ordering::Greater, + (Scope::Write(Some(ref a)), Scope::Write(Some(ref b))) => a.cmp(b), + + (Scope::Read(..), Scope::Write(..)) => Ordering::Less, + (Scope::Read(..), Scope::Follow) => Ordering::Less, + (Scope::Read(..), Scope::Push) => Ordering::Less, + + (Scope::Write(..), Scope::Read(..)) => Ordering::Greater, + (Scope::Write(..), Scope::Follow) => Ordering::Less, + (Scope::Write(..), Scope::Push) => Ordering::Less, + + (Scope::Follow, Scope::Read(..)) => Ordering::Greater, + (Scope::Follow, Scope::Write(..)) => Ordering::Greater, + (Scope::Follow, Scope::Follow) => Ordering::Equal, + (Scope::Follow, Scope::Push) => Ordering::Less, + + (Scope::Push, Scope::Push) => Ordering::Equal, + (Scope::Push, _) => Ordering::Greater, + } + } +} + +impl fmt::Display for Scope { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + use self::Scope::*; + let s = match *self { + Read(Some(ref r)) => return fmt::Display::fmt(r, f), + Read(None) => "read", + Write(Some(ref w)) => return fmt::Display::fmt(w, f), + Write(None) => "write", + Follow => "follow", + Push => "push", + }; + write!(f, "{}", s) + } +} + +impl Default for Scope { + fn default() -> Self { + Scope::Read(None) + } +} + +/// Represents the granular "read:___" oauth scopes +#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Serialize)] +pub enum Read { + /// Accounts + #[serde(rename = "accounts")] + Accounts, + /// Blocks + #[serde(rename = "blocks")] + Blocks, + /// Favourites + #[serde(rename = "favourites")] + Favourites, + /// Filters + #[serde(rename = "filters")] + Filters, + /// Follows + #[serde(rename = "follows")] + Follows, + /// Lists + #[serde(rename = "lists")] + Lists, + /// Mutes + #[serde(rename = "mutes")] + Mutes, + /// Notifications + #[serde(rename = "notifications")] + Notifications, + /// Reports + #[serde(rename = "reports")] + Reports, + /// Search + #[serde(rename = "search")] + Search, + /// Statuses + #[serde(rename = "statuses")] + Statuses, +} + +impl FromStr for Read { + type Err = Error; + + fn from_str(s: &str) -> Result { + Ok(match s { + "accounts" => Read::Accounts, + "blocks" => Read::Blocks, + "favourites" => Read::Favourites, + "filters" => Read::Filters, + "follows" => Read::Follows, + "lists" => Read::Lists, + "mutes" => Read::Mutes, + "notifications" => Read::Notifications, + "reports" => Read::Reports, + "search" => Read::Search, + "statuses" => Read::Statuses, + _ => return Err(Error::Other("Unknown 'read' subcategory".to_string())), + }) + } +} + +impl PartialOrd for Read { + fn partial_cmp(&self, other: &Read) -> Option { + Some(self.cmp(other)) + } +} + +impl Ord for Read { + fn cmp(&self, other: &Read) -> Ordering { + let a = format!("{:?}", self); + let b = format!("{:?}", other); + a.cmp(&b) + } +} + +impl fmt::Display for Read { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + write!( + f, + "read:{}", + match *self { + Read::Accounts => "accounts", + Read::Blocks => "blocks", + Read::Favourites => "favourites", + Read::Filters => "filters", + Read::Follows => "follows", + Read::Lists => "lists", + Read::Mutes => "mutes", + Read::Notifications => "notifications", + Read::Reports => "reports", + Read::Search => "search", + Read::Statuses => "statuses", + } + ) + } +} + +/// Represents the granular "write:___" oauth scopes +#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Serialize)] +pub enum Write { + /// Accounts + #[serde(rename = "accounts")] + Accounts, + /// Blocks + #[serde(rename = "blocks")] + Blocks, + /// Favourites + #[serde(rename = "favourites")] + Favourites, + /// Filters + #[serde(rename = "filters")] + Filters, + /// Follows + #[serde(rename = "follows")] + Follows, + /// Lists + #[serde(rename = "lists")] + Lists, + /// Media + #[serde(rename = "media")] + Media, + /// Mutes + #[serde(rename = "mutes")] + Mutes, + /// Notifications + #[serde(rename = "notifications")] + Notifications, + /// Reports + #[serde(rename = "reports")] + Reports, + /// Statuses + #[serde(rename = "statuses")] + Statuses, +} + +impl FromStr for Write { + type Err = Error; + + fn from_str(s: &str) -> Result { + Ok(match s { + "accounts" => Write::Accounts, + "blocks" => Write::Blocks, + "favourites" => Write::Favourites, + "filters" => Write::Filters, + "follows" => Write::Follows, + "lists" => Write::Lists, + "media" => Write::Media, + "mutes" => Write::Mutes, + "notifications" => Write::Notifications, + "reports" => Write::Reports, + "statuses" => Write::Statuses, + _ => return Err(Error::Other("Unknown 'write' subcategory".to_string())), + }) + } +} + +impl PartialOrd for Write { + fn partial_cmp(&self, other: &Write) -> Option { + Some(self.cmp(other)) + } +} + +impl Ord for Write { + fn cmp(&self, other: &Write) -> Ordering { + let a = format!("{:?}", self); + let b = format!("{:?}", other); + a.cmp(&b) + } +} + +impl fmt::Display for Write { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + write!( + f, + "write:{}", + match *self { + Write::Accounts => "accounts", + Write::Blocks => "blocks", + Write::Favourites => "favourites", + Write::Filters => "filters", + Write::Follows => "follows", + Write::Lists => "lists", + Write::Media => "media", + Write::Mutes => "mutes", + Write::Notifications => "notifications", + Write::Reports => "reports", + Write::Statuses => "statuses", + } + ) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use serde_json; + + #[test] + fn test_write_cmp() { + let tests = [ + (Write::Accounts, Write::Blocks), + (Write::Blocks, Write::Favourites), + (Write::Favourites, Write::Filters), + (Write::Filters, Write::Follows), + (Write::Follows, Write::Lists), + (Write::Lists, Write::Media), + (Write::Media, Write::Mutes), + (Write::Mutes, Write::Notifications), + (Write::Notifications, Write::Reports), + (Write::Reports, Write::Statuses), + ]; + + for (a, b) in &tests { + assert!(a < b); + assert!(b > a); + } + } + + #[test] + fn test_read_cmp() { + let tests = [ + (Read::Accounts, Read::Blocks), + (Read::Blocks, Read::Favourites), + (Read::Favourites, Read::Filters), + (Read::Filters, Read::Follows), + (Read::Follows, Read::Lists), + (Read::Lists, Read::Mutes), + (Read::Mutes, Read::Notifications), + (Read::Notifications, Read::Reports), + (Read::Reports, Read::Search), + (Read::Search, Read::Statuses), + ]; + for (a, b) in &tests { + assert!(a < b); + assert!(b > a); + } + } + + #[test] + fn test_scope_cmp() { + let tests = [ + (Scope::Read(None), Scope::Read(Some(Read::Accounts))), + (Scope::Read(None), Scope::Read(Some(Read::Blocks))), + (Scope::Read(None), Scope::Read(Some(Read::Favourites))), + (Scope::Read(None), Scope::Read(Some(Read::Filters))), + (Scope::Read(None), Scope::Read(Some(Read::Follows))), + (Scope::Read(None), Scope::Read(Some(Read::Lists))), + (Scope::Read(None), Scope::Read(Some(Read::Mutes))), + (Scope::Read(None), Scope::Read(Some(Read::Notifications))), + (Scope::Read(None), Scope::Read(Some(Read::Reports))), + (Scope::Read(None), Scope::Read(Some(Read::Search))), + (Scope::Read(None), Scope::Read(Some(Read::Statuses))), + (Scope::Read(Some(Read::Statuses)), Scope::Write(None)), + (Scope::Read(Some(Read::Mutes)), Scope::Follow), + (Scope::Read(None), Scope::Push), + (Scope::Write(None), Scope::Write(Some(Write::Accounts))), + (Scope::Write(None), Scope::Write(Some(Write::Blocks))), + (Scope::Write(None), Scope::Write(Some(Write::Favourites))), + (Scope::Write(None), Scope::Write(Some(Write::Filters))), + (Scope::Write(None), Scope::Write(Some(Write::Follows))), + (Scope::Write(None), Scope::Write(Some(Write::Lists))), + (Scope::Write(None), Scope::Write(Some(Write::Media))), + (Scope::Write(None), Scope::Write(Some(Write::Mutes))), + (Scope::Write(None), Scope::Write(Some(Write::Notifications))), + (Scope::Write(None), Scope::Write(Some(Write::Reports))), + (Scope::Write(None), Scope::Write(Some(Write::Statuses))), + (Scope::Write(Some(Write::Statuses)), Scope::Follow), + (Scope::Write(Some(Write::Follows)), Scope::Push), + ]; + + for (a, b) in &tests { + assert!(a < b); + } + } + + #[test] + fn test_scope_display() { + let values = [ + Scope::Read(None), + Scope::Read(Some(Read::Accounts)), + Scope::Read(Some(Read::Blocks)), + Scope::Read(Some(Read::Favourites)), + Scope::Read(Some(Read::Filters)), + Scope::Read(Some(Read::Follows)), + Scope::Read(Some(Read::Lists)), + Scope::Read(Some(Read::Mutes)), + Scope::Read(Some(Read::Notifications)), + Scope::Read(Some(Read::Reports)), + Scope::Read(Some(Read::Search)), + Scope::Read(Some(Read::Statuses)), + Scope::Write(None), + Scope::Write(Some(Write::Accounts)), + Scope::Write(Some(Write::Blocks)), + Scope::Write(Some(Write::Favourites)), + Scope::Write(Some(Write::Filters)), + Scope::Write(Some(Write::Follows)), + Scope::Write(Some(Write::Lists)), + Scope::Write(Some(Write::Media)), + Scope::Write(Some(Write::Mutes)), + Scope::Write(Some(Write::Notifications)), + Scope::Write(Some(Write::Reports)), + Scope::Write(Some(Write::Statuses)), + Scope::Follow, + Scope::Push, + ]; + + let expecteds = [ + "read".to_string(), + "read:accounts".to_string(), + "read:blocks".to_string(), + "read:favourites".to_string(), + "read:filters".to_string(), + "read:follows".to_string(), + "read:lists".to_string(), + "read:mutes".to_string(), + "read:notifications".to_string(), + "read:reports".to_string(), + "read:search".to_string(), + "read:statuses".to_string(), + "write".to_string(), + "write:accounts".to_string(), + "write:blocks".to_string(), + "write:favourites".to_string(), + "write:filters".to_string(), + "write:follows".to_string(), + "write:lists".to_string(), + "write:media".to_string(), + "write:mutes".to_string(), + "write:notifications".to_string(), + "write:reports".to_string(), + "write:statuses".to_string(), + "follow".to_string(), + "push".to_string(), + ]; + + let tests = values.iter().zip(expecteds.iter()); + + for (value, expected) in tests { + let result = value.to_string(); + assert_eq!(&result, expected); + } + } + + #[test] + fn test_scopes_default() { + let default: Scope = Default::default(); + assert_eq!(default, Scope::Read(None)); + } + + #[test] + fn test_scopes_display() { + let tests = [ + ( + Scopes::read(Read::Accounts) | Scopes::follow(), + "read:accounts follow", + ), + ( + Scopes::read(Read::Follows) | Scopes::read(Read::Accounts) | Scopes::write_all(), + "read:accounts read:follows write", + ), + ]; + + for (a, b) in &tests { + assert_eq!(&format!("{}", a), b); + } + } + + #[test] + fn test_scopes_serialize_deserialize() { + let tests = [ + ( + Scopes::read_all() | Scopes::write(Write::Notifications) | Scopes::follow(), + "read write:notifications follow", + ), + (Scopes::follow() | Scopes::push(), "follow push"), + ]; + + for (a, b) in &tests { + let ser = serde_json::to_string(&a).expect("Couldn't serialize Scopes"); + let expected = format!("\"{}\"", b); + assert_eq!(&ser, &expected); + + let des : Scopes = serde_json::from_str(&ser).expect("Couldn't deserialize Scopes"); + assert_eq!(&des, a); + } + } + + #[test] + fn test_scope_from_str() { + let tests = [ + ("read", Scope::Read(None)), + ("read:accounts", Scope::Read(Some(Read::Accounts))), + ("read:blocks", Scope::Read(Some(Read::Blocks))), + ("read:favourites", Scope::Read(Some(Read::Favourites))), + ("read:filters", Scope::Read(Some(Read::Filters))), + ("read:follows", Scope::Read(Some(Read::Follows))), + ("read:lists", Scope::Read(Some(Read::Lists))), + ("read:mutes", Scope::Read(Some(Read::Mutes))), + ("read:notifications", Scope::Read(Some(Read::Notifications))), + ("read:reports", Scope::Read(Some(Read::Reports))), + ("read:search", Scope::Read(Some(Read::Search))), + ("read:statuses", Scope::Read(Some(Read::Statuses))), + ("write", Scope::Write(None)), + ("write:accounts", Scope::Write(Some(Write::Accounts))), + ("write:blocks", Scope::Write(Some(Write::Blocks))), + ("write:favourites", Scope::Write(Some(Write::Favourites))), + ("write:filters", Scope::Write(Some(Write::Filters))), + ("write:follows", Scope::Write(Some(Write::Follows))), + ("write:lists", Scope::Write(Some(Write::Lists))), + ("write:media", Scope::Write(Some(Write::Media))), + ("write:mutes", Scope::Write(Some(Write::Mutes))), + ( + "write:notifications", + Scope::Write(Some(Write::Notifications)), + ), + ("write:reports", Scope::Write(Some(Write::Reports))), + ("write:statuses", Scope::Write(Some(Write::Statuses))), + ("follow", Scope::Follow), + ("push", Scope::Push), + ]; + for (source, expected) in &tests { + let result = Scope::from_str(source).expect(&format!("Couldn't parse '{}'", &source)); + assert_eq!(result, *expected); + } + } + + #[test] + fn test_scopes_str_round_trip() { + let original = "read write follow push"; + let scopes = Scopes::from_str(original).expect("Couldn't convert to Scopes"); + let result = format!("{}", scopes); + assert_eq!(original, result); + } +} diff --git a/src/status_builder.rs b/src/status_builder.rs new file mode 100644 index 0000000..15a24eb --- /dev/null +++ b/src/status_builder.rs @@ -0,0 +1,366 @@ +use isolang::Language; + +/// A builder pattern struct for constructing a status. +/// +/// # Example +/// +/// ``` +/// # extern crate elefren; +/// # use elefren::{Language, StatusBuilder}; +/// +/// # fn main() -> Result<(), elefren::Error> { +/// let status = StatusBuilder::new() +/// .status("a status") +/// .sensitive(true) +/// .spoiler_text("a CW") +/// .language(Language::Eng) +/// .build()?; +/// # Ok(()) +/// # } +/// ``` +#[derive(Debug, Default, Clone, PartialEq)] +pub struct StatusBuilder { + status: Option, + in_reply_to_id: Option, + media_ids: Option>, + sensitive: Option, + spoiler_text: Option, + content_type: Option, + visibility: Option, + language: Option, +} + +impl StatusBuilder { + /// Create a StatusBuilder object + /// + /// # Example + /// + /// ```rust,no_run + /// # use elefren::prelude::*; + /// # use elefren::status_builder::Visibility; + /// # fn main() -> Result<(), elefren::Error> { + /// # let data = Data { + /// # base: "".into(), + /// # client_id: "".into(), + /// # client_secret: "".into(), + /// # redirect: "".into(), + /// # token: "".into(), + /// # }; + /// # let client = Mastodon::from(data); + /// let status = StatusBuilder::new() + /// .status("a status") + /// .visibility(Visibility::Public) + /// .build()?; + /// client.new_status(status)?; + /// # Ok(()) + /// # } + /// ``` + pub fn new() -> StatusBuilder { + StatusBuilder::default() + } + + /// Set the text for the post + /// + /// # Example + /// + /// ```rust + /// # use elefren::prelude::*; + /// # fn main() -> Result<(), elefren::Error> { + /// let status = StatusBuilder::new().status("awoooooo").build()?; + /// # Ok(()) + /// # } + /// ``` + pub fn status>(&mut self, status: I) -> &mut Self { + self.status = Some(status.into()); + self + } + + /// Set the in_reply_to_id for the post + /// + /// # Example + /// + /// ```rust + /// # use elefren::prelude::*; + /// # fn main() -> Result<(), elefren::Error> { + /// let status = StatusBuilder::new() + /// .status("awooooo") + /// .in_reply_to("12345") + /// .build()?; + /// # Ok(()) + /// # } + /// ``` + pub fn in_reply_to>(&mut self, id: I) -> &mut Self { + self.in_reply_to_id = Some(id.into()); + self + } + + /// Set the media_ids for the post + /// + /// # Example + /// + /// ```rust + /// # use elefren::prelude::*; + /// # fn main() -> Result<(), elefren::Error> { + /// let status = StatusBuilder::new().media_ids(&["foo", "bar"]).build()?; + /// # Ok(()) + /// # } + /// ``` + pub fn media_ids>( + &mut self, + ids: I, + ) -> &mut Self { + self.media_ids = Some(ids.into_iter().map(|s| s.to_string()).collect::>()); + self + } + + /// Set the sensitive attribute for the post + /// + /// # Example + /// + /// ```rust + /// # use elefren::prelude::*; + /// # fn main() -> Result<(), elefren::Error> { + /// let status = StatusBuilder::new() + /// .media_ids(&["foo", "bar"]) + /// .sensitive(true) + /// .build()?; + /// # Ok(()) + /// # } + /// ``` + pub fn sensitive(&mut self, sensitive: bool) -> &mut Self { + self.sensitive = Some(sensitive); + self + } + + /// Set the spoiler text/CW for the post + /// + /// # Example + /// + /// ```rust + /// # use elefren::prelude::*; + /// # fn main() -> Result<(), elefren::Error> { + /// let status = StatusBuilder::new() + /// .status("awoooo!!") + /// .spoiler_text("awoo inside") + /// .build()?; + /// # Ok(()) + /// # } + /// ``` + pub fn spoiler_text>(&mut self, spoiler_text: I) -> &mut Self { + self.spoiler_text = Some(spoiler_text.into()); + self + } + + /// Set the content type of the post + /// + /// This is a Pleroma and Glitch-soc extension of the API. + /// + /// # Possible values + /// - `text/plain` + /// - `text/html` + /// - `text/markdown` + /// - `text/bbcode` (Pleroma only) + /// + /// The set of supported content types may vary by server. + /// + /// # Example + /// + /// ```rust + /// # use elefren::prelude::*; + /// # fn main() -> Result<(), elefren::Error> { + /// let status = StatusBuilder::new() + /// .status("thicc") + /// .content_type("text/html") + /// .build()?; + /// # Ok(()) + /// # } + /// ``` + pub fn content_type>(&mut self, content_type: I) -> &mut Self { + self.content_type = Some(content_type.into()); + self + } + + /// Set the visibility for the post + /// + /// # Example + /// + /// ```rust + /// # use elefren::prelude::*; + /// # use elefren::status_builder::Visibility; + /// # fn main() -> Result<(), elefren::Error> { + /// let status = StatusBuilder::new() + /// .status("awooooooo") + /// .visibility(Visibility::Public) + /// .build()?; + /// # Ok(()) + /// # } + /// ``` + pub fn visibility(&mut self, visibility: Visibility) -> &mut Self { + self.visibility = Some(visibility); + self + } + + /// Set the language for the post + /// + /// # Example + /// + /// ```rust + /// # use elefren::prelude::*; + /// # use elefren::Language; + /// # fn main() -> Result<(), elefren::Error> { + /// let status = StatusBuilder::new() + /// .status("awoo!!!!") + /// .language(Language::Eng) + /// .build()?; + /// # Ok(()) + /// # } + /// ``` + pub fn language(&mut self, language: Language) -> &mut Self { + self.language = Some(language); + self + } + + /// Constructs a NewStatus + /// + /// # Example + /// + /// ```rust + /// # use elefren::prelude::*; + /// # fn main() -> Result<(), elefren::Error> { + /// let status = StatusBuilder::new().status("awoo!").build()?; + /// # Ok(()) + /// # } + /// ``` + pub fn build(&self) -> Result { + if self.status.is_none() && self.media_ids.is_none() { + return Err(crate::Error::Other( + "status text or media ids are required in order to post a status".to_string(), + )); + } + Ok(NewStatus { + status: self.status.clone(), + in_reply_to_id: self.in_reply_to_id.clone(), + media_ids: self.media_ids.clone(), + sensitive: self.sensitive.clone(), + spoiler_text: self.spoiler_text.clone(), + visibility: self.visibility.clone(), + language: self.language.clone(), + content_type: self.content_type.clone(), + }) + } +} + +/// Represents a post that can be sent to the POST /api/v1/status endpoint +#[derive(Debug, Default, Clone, Serialize, PartialEq)] +pub struct NewStatus { + #[serde(skip_serializing_if = "Option::is_none")] + status: Option, + #[serde(skip_serializing_if = "Option::is_none")] + in_reply_to_id: Option, + #[serde(skip_serializing_if = "Option::is_none")] + media_ids: Option>, + #[serde(skip_serializing_if = "Option::is_none")] + sensitive: Option, + #[serde(skip_serializing_if = "Option::is_none")] + spoiler_text: Option, + #[serde(skip_serializing_if = "Option::is_none")] + visibility: Option, + #[serde(skip_serializing_if = "Option::is_none")] + language: Option, + #[serde(skip_serializing_if = "Option::is_none")] + content_type: Option, +} + +/// The visibility of a status. +#[derive(Clone, Copy, Debug, Deserialize, Serialize, PartialEq)] +#[serde(rename_all = "lowercase")] +pub enum Visibility { + /// A Direct message to a user + Direct, + /// Only available to followers + Private, + /// Not shown in public timelines + Unlisted, + /// Posted to public timelines + Public, +} + +impl Default for Visibility { + fn default() -> Self { + Visibility::Public + } +} + +#[cfg(test)] +mod tests { + use super::*; + use isolang::Language; + use serde_json; + + #[test] + fn test_new() { + let s = StatusBuilder::new() + .status("a status") + .build() + .expect("Couldn't build status"); + let expected = NewStatus { + status: Some("a status".to_string()), + in_reply_to_id: None, + media_ids: None, + sensitive: None, + spoiler_text: None, + visibility: None, + language: None, + content_type: None, + }; + assert_eq!(s, expected); + } + + #[test] + fn test_default_visibility() { + let v: Visibility = Default::default(); + assert_eq!(v, Visibility::Public); + } + + #[test] + fn test_serialize_visibility() { + assert_eq!( + serde_json::to_string(&Visibility::Direct).expect("couldn't serialize visibility"), + "\"direct\"".to_string() + ); + assert_eq!( + serde_json::to_string(&Visibility::Private).expect("couldn't serialize visibility"), + "\"private\"".to_string() + ); + assert_eq!( + serde_json::to_string(&Visibility::Unlisted).expect("couldn't serialize visibility"), + "\"unlisted\"".to_string() + ); + assert_eq!( + serde_json::to_string(&Visibility::Public).expect("couldn't serialize visibility"), + "\"public\"".to_string() + ); + } + + #[test] + fn test_serialize_status() { + let status = StatusBuilder::new() + .status("a status") + .build() + .expect("Couldn't build status"); + assert_eq!( + serde_json::to_string(&status).expect("Couldn't serialize status"), + "{\"status\":\"a status\"}".to_string() + ); + + let status = StatusBuilder::new() + .status("a status") + .language(Language::Eng) + .build() + .expect("Couldn't build status"); + assert_eq!( + serde_json::to_string(&status).expect("Couldn't serialize status"), + "{\"status\":\"a status\",\"language\":\"eng\"}" + ); + } +} diff --git a/tests/skeptic.rs b/tests/skeptic.rs new file mode 100644 index 0000000..ff46c9c --- /dev/null +++ b/tests/skeptic.rs @@ -0,0 +1 @@ +include!(concat!(env!("OUT_DIR"), "/skeptic-tests.rs"));