// zotapi - Rust wrapper for Sot API as implemented by Hubzilla // Copyright (C) 2018 Harald Eilertsen // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . extern crate zotapi; extern crate mockito; use mockito::{mock, Matcher, Mock}; fn mock_with_authorization(method: &str, url: &str) -> Mock { mock(method, url) .match_header("Authorization", Matcher::Exact(format!("Basic {}", base64::encode("testuser:test1234")))) } fn default_mock(method: &str, url: &str) -> Mock { mock_with_authorization(method, url) .with_status(200) .with_header("content-type", "application/json") .with_body("{}") .create() } fn client() -> zotapi::Client { zotapi::client(&mockito::server_url(), "testuser", "test1234") } #[test] fn get_channel_stream() { let m = default_mock("GET", "/api/z/1.0/channel/stream"); let data = client().channel_stream(); m.assert(); assert_eq!(data.unwrap(), "{}"); } #[test] fn get_network_stream() { let m = default_mock("GET", "/api/z/1.0/network/stream"); let data = client().network_stream(); m.assert(); assert_eq!(data.unwrap(), "{}"); } #[test] fn return_error_if_invalid_auth_provided() { let m = mock("GET", "/api/z/1.0/channel/stream") .with_status(401) .with_header("content-type", "text") .with_body("This api requires login") .create(); let data = client().channel_stream(); m.assert(); assert!(data.is_err()); assert_eq!(format!("{:?}", data), "Err(Unauthorized)"); } #[test] fn create_new_post() { let m = mock_with_authorization("POST", "/api/z/1.0/item/update") .match_body("body=This+is+a+test") .with_status(200) .with_header("content-type", "application/json") .with_body("{}") .create(); let _res = client().item().body("This is a test").create(); m.assert(); } #[test] fn create_new_post_with_title() { let m = mock_with_authorization("POST", "/api/z/1.0/item/update") .match_body( Matcher::AllOf(vec![ Matcher::UrlEncoded("title".to_string(), "A+title".to_string()), Matcher::UrlEncoded("body".to_string(), "This+is+a+test".to_string()), ]) ) .with_status(200) .with_header("content-type", "application/json") .with_body("{}") .create(); let _res = client().item().title("A title").body("This is a test").create(); m.assert(); } #[test] fn upload_item_with_media_file() { let m = mock_with_authorization("POST", "/api/z/1.0/item/update") .match_header("content-type", Matcher::Regex("^multipart/form-data; boundary=.+$".into())) .match_body(Matcher::Regex( "--.+\r\n".to_owned() + "Content-Disposition: form-data; name=\"body\"\r\n" + "\r\nThis is a test\r\n" + "--.+\r\n" + "Content-Disposition: form-data; name=\"title\"\r\n" + "\r\nA title\r\n" + "--.+\r\n" + "Content-Disposition: form-data; name=\"media\"; filename=\"testfile.txt\"\r\n" + "Content-Type: text/plain\r\n" + "\r\ntestfile contents\n" + "\r\n--.+--\r\n")) .with_status(200) .with_header("content-type", "application/json") .with_body("{}") .create(); let _res = client().item() .title("A title") .body("This is a test") .file("tests/fixtures/testfile.txt") .create(); m.assert(); } #[test] fn upload_item_with_two_files() { let m = mock_with_authorization("POST", "/api/z/1.0/item/update") .match_header("content-type", Matcher::Regex("multipart/form-data; boundary=.+".into())) .match_body(Matcher::Regex( "--.+\r\n".to_owned() + "Content-Disposition: form-data; name=\"body\"\r\n" + "\r\nThis is a test\r\n" + "--.+\r\n" + "Content-Disposition: form-data; name=\"title\"\r\n" + "\r\nA title\r\n" + "--.+\r\n" + "Content-Disposition: form-data; name=\"media\"; filename=\"testfile.txt\"\r\n" + "Content-Type: text/plain\r\n" + "\r\ntestfile contents\n" + "\r\n--.+\r\n" + "Content-Disposition: form-data; name=\"media\"; filename=\"testfile.txt\"\r\n" + "Content-Type: text/plain\r\n" + "\r\ntestfile contents\n" + "\r\n--.+--\r\n")) .with_status(200) .with_header("content-type", "application/json") .with_body("{}") .create(); let _res = client().item() .title("A title") .body("This is a test") .file("tests/fixtures/testfile.txt") .file("tests/fixtures/testfile.txt") .create(); m.assert(); } #[test] fn fetch_xchan_by_address() { let m = default_mock("GET", "/api/z/1.0/xchan?address=test%40test.com"); let _res = client().xchan().by_address("test@test.com").fetch().unwrap(); m.assert(); } #[test] fn fetch_xchan_by_hash() { let m = default_mock("GET", "/api/z/1.0/xchan?hash=baffebaff"); let _res = client().xchan().by_hash("baffebaff").fetch().unwrap(); m.assert(); } #[test] fn fetch_xchan_by_guid() { let m = default_mock("GET", "/api/z/1.0/xchan?guid=baffebaff-baff-baff"); let _res = client().xchan().by_guid("baffebaff-baff-baff").fetch().unwrap(); m.assert(); } #[test] fn fetch_connections() { let m = default_mock("GET", "/api/z/1.0/abook"); let _res = client().abook().fetch().unwrap(); m.assert(); } #[test] fn fetch_abconfig() { let m = default_mock("GET", "/api/z/1.0/abconfig"); let _res = client().abconfig().fetch().unwrap(); m.assert(); } #[test] fn fetch_privacy_groups() { let m = default_mock("GET", "/api/z/1.0/group"); let _res = client().group().fetch().unwrap(); m.assert(); } #[test] fn fetch_members_of_group_by_group_id() { let m = mock_with_authorization("GET", "/api/z/1.0/group_members") .match_query(Matcher::UrlEncoded("group_id".to_string(), "42".to_string())) .with_status(200) .with_header("content-type", "application/json") .with_body("{}") .create(); let _res = client().group_members().by_group_id(42).fetch().unwrap(); m.assert(); } #[test] fn fetch_members_of_group_by_group_name() { let m = mock_with_authorization("GET", "/api/z/1.0/group_members") .match_query(Matcher::UrlEncoded("group_name".into(), "Friends+of+pain".into())) .with_status(200) .with_header("content-type", "application/json") .with_body("{}") .create(); let _res = client().group_members().by_group_name("Friends of pain").fetch().unwrap(); m.assert(); }