1
1
//! Functions for interacting with an Atomic Server
2
2
use url:: Url ;
3
3
4
- use crate :: { errors:: AtomicResult , parse:: parse_json_ad_resource, Resource , Storelike } ;
4
+ use crate :: {
5
+ agents:: Agent , commit:: sign_message, errors:: AtomicResult , parse:: parse_json_ad_resource,
6
+ Resource , Storelike ,
7
+ } ;
5
8
6
9
/// Fetches a resource, makes sure its subject matches.
7
10
/// Checks the datatypes for the Values.
8
11
/// Ignores all atoms where the subject is different.
9
12
/// WARNING: Calls store methods, and is called by store methods, might get stuck in a loop!
10
- pub fn fetch_resource ( subject : & str , store : & impl Storelike ) -> AtomicResult < Resource > {
11
- let body = fetch_body ( subject, crate :: parse:: JSON_AD_MIME ) ?;
13
+ pub fn fetch_resource (
14
+ subject : & str ,
15
+ store : & impl Storelike ,
16
+ for_agent : Option < Agent > ,
17
+ ) -> AtomicResult < Resource > {
18
+ let body = fetch_body ( subject, crate :: parse:: JSON_AD_MIME , for_agent) ?;
12
19
let resource = parse_json_ad_resource ( & body, store)
13
20
. map_err ( |e| format ! ( "Error parsing body of {}. {}" , subject, e) ) ?;
14
21
Ok ( resource)
15
22
}
16
23
17
- /// Fetches a URL, returns its body
18
- pub fn fetch_body ( url : & str , content_type : & str ) -> AtomicResult < String > {
24
+ /// Returns the various x-atomic authentication headers, includign agent signature
25
+ pub fn get_authentication_headers ( url : & str , agent : & Agent ) -> AtomicResult < Vec < ( String , String ) > > {
26
+ let mut headers = Vec :: new ( ) ;
27
+ let now = crate :: datetime_helpers:: now ( ) . to_string ( ) ;
28
+ let message = format ! ( "{} {}" , url, now) ;
29
+ let signature = sign_message (
30
+ & message,
31
+ agent
32
+ . private_key
33
+ . as_ref ( )
34
+ . ok_or ( "No private key in agent" ) ?,
35
+ & agent. public_key ,
36
+ ) ?;
37
+ headers. push ( ( "x-atomic-public-key" . into ( ) , agent. public_key . to_string ( ) ) ) ;
38
+ headers. push ( ( "x-atomic-signature" . into ( ) , signature) ) ;
39
+ headers. push ( ( "x-atomic-timestamp" . into ( ) , now) ) ;
40
+ headers. push ( ( "x-atomic-agent" . into ( ) , agent. subject . to_string ( ) ) ) ;
41
+ Ok ( headers)
42
+ }
43
+
44
+ /// Fetches a URL, returns its body.
45
+ /// Uses the store's Agent agent (if set) to sign the request.
46
+ pub fn fetch_body ( url : & str , content_type : & str , for_agent : Option < Agent > ) -> AtomicResult < String > {
19
47
if !url. starts_with ( "http" ) {
20
48
return Err ( format ! ( "Could not fetch url '{}', must start with http." , url) . into ( ) ) ;
21
49
}
50
+ if let Some ( agent) = for_agent {
51
+ get_authentication_headers ( url, & agent) ?;
52
+ }
22
53
let resp = ureq:: get ( url)
23
54
. set ( "Accept" , content_type)
24
55
. timeout_read ( 2000 )
@@ -28,7 +59,7 @@ pub fn fetch_body(url: &str, content_type: &str) -> AtomicResult<String> {
28
59
} ;
29
60
let body = resp
30
61
. into_string ( )
31
- . map_err ( |e| format ! ( "Could not parse response {}: {}" , url, e) ) ?;
62
+ . map_err ( |e| format ! ( "Could not parse HTTP response for {}: {}" , url, e) ) ?;
32
63
Ok ( body)
33
64
}
34
65
@@ -50,7 +81,11 @@ pub fn fetch_tpf(
50
81
if let Some ( val) = q_value {
51
82
url. query_pairs_mut ( ) . append_pair ( "value" , val) ;
52
83
}
53
- let body = fetch_body ( url. as_str ( ) , "application/ad+json" ) ?;
84
+ let body = fetch_body (
85
+ url. as_str ( ) ,
86
+ "application/ad+json" ,
87
+ store. get_default_agent ( ) . ok ( ) ,
88
+ ) ?;
54
89
crate :: parse:: parse_json_ad_array ( & body, store, false )
55
90
}
56
91
@@ -97,7 +132,7 @@ mod test {
97
132
#[ ignore]
98
133
fn fetch_resource_basic ( ) {
99
134
let store = crate :: Store :: init ( ) . unwrap ( ) ;
100
- let resource = fetch_resource ( crate :: urls:: SHORTNAME , & store) . unwrap ( ) ;
135
+ let resource = fetch_resource ( crate :: urls:: SHORTNAME , & store, None ) . unwrap ( ) ;
101
136
let shortname = resource. get ( crate :: urls:: SHORTNAME ) . unwrap ( ) ;
102
137
assert ! ( shortname. to_string( ) == "shortname" ) ;
103
138
}
0 commit comments