1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
use alloc::borrow::Cow;
use serde_json::Value;

use crate::{Error, Pointer};

/// Assign is implemented by types which can internally assign a
/// `serde_json::Value` by a JSON Pointer.
pub trait Assign {
    /// Error associated with `Assign`
    type Error;
    /// Assign a value of based on the path provided by a JSON Pointer.
    fn assign<V>(&mut self, ptr: &Pointer, value: V) -> Result<Assignment, Self::Error>
    where
        V: Into<Value>;
}

impl Assign for Value {
    type Error = Error;
    fn assign<V>(&mut self, ptr: &Pointer, value: V) -> Result<Assignment, Error>
    where
        V: Into<Value>,
    {
        ptr.assign(self, value)
    }
}
#[derive(Debug)]
/// The data structure returned from a successful call to `assign`.
pub struct Assignment<'a> {
    /// The value that was assigned.
    ///
    /// In the event a path is created, this will be the `serde_json::Value`
    /// encompassing the new branch.
    pub assigned: Cow<'a, Value>,

    /// The value that was replaced.
    ///
    /// Note: `serde_json::Value::Null` is utilized if the value did not
    /// previously exist.
    pub replaced: Value,
    /// The path which was created or replaced.
    ///
    /// For example, if you had the json `{ "foo": { "bar": "baz" } }` and you
    /// assigned `"new_value"` to `"/foo/qux/quux"`, then `created_or_mutated`
    /// would be `Some("/foo/qux")` as `"qux"` is the top-level value assigned.
    ///
    /// The resulting json would have the following structure:
    /// ```json
    /// {
    ///     "foo": {
    ///        "bar": "baz",
    ///         "qux": {
    ///             "quux": "new_value"
    ///         }
    ///     }
    /// }
    /// ```
    ///
    /// Note: if a portion of the path contains a leaf node that is to be
    /// overridden by an object or an array, then the path will be leaf that is
    /// replaced.
    ///
    /// For example, if you had the json:
    /// ```json
    /// { "foo:" "bar" }
    /// ```
    /// and you assigned `"new_value"` to `"/foo/bar/baz"`, then the value would
    /// be `Some("/foo/bar")`.
    pub created_or_mutated: Pointer,
    /// A `Pointer` consisting of the path which was assigned.
    ///
    /// ## Example
    /// ```rust
    /// use serde_json::json;
    /// use jsonptr::{Pointer, Assign};
    /// let mut data = json!({ "foo": ["zero"] });
    /// let mut ptr = Pointer::try_from("/foo/-").unwrap();
    /// let assignment = data.assign(&mut ptr, "one").unwrap();
    /// assert_eq!(assignment.assigned_to, Pointer::try_from("/foo/1").unwrap());
    /// ```
    pub assigned_to: Pointer,
}