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
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
use core::marker::PhantomData;

use crate::{
    build::{Build, Builder},
    error::{BasicError, UniError},
    protocol::{
        lookup_hint, lookup_visit, AnyHint, AnyVisit, Hint, Protocol, ProtocolDescription,
        ProtocolId,
    },
    protocol_list,
    protocols::reference::{self, Reference},
    walk::{Walk, WalkMut, WalkOnce},
    walk_with_hints, HintGiven, WalkerHints,
};

use super::reference::BuilderRefValue;

pub struct Walker<'value, 'ctx: 'value>(&'value str, PhantomData<&'ctx ()>);

impl<'value, 'ctx: 'value, VisitorErr: 'ctx> crate::Walker<'value, 'ctx, VisitorErr>
    for Walker<'value, 'ctx>
{
    type Error = BasicError;

    fn walk(
        &mut self,
        visitor: &mut dyn crate::Visitor<'value, 'ctx, Self::Error, Error = VisitorErr>,
    ) -> Result<crate::WalkStatus, UniError<Self::Error, VisitorErr>> {
        match lookup_visit::<Reference<str>, _, _>(visitor).map_err(UniError::Walker)? {
            Some(visitor) => {
                visitor.visit(reference::Ref::Value(self.0))?;
                Ok(crate::WalkStatus::Continue)
            }
            None => Err(UniError::Walker(BasicError::new(
                "visitor needs the reference protocol for `str`",
            ))),
        }
    }

    fn all_protocols() -> impl Iterator<Item = ProtocolDescription>
    where
        Self: Sized,
    {
        protocol_list![Reference<str>]
    }
}

impl<'value, 'ctx: 'value, VisitorErr: 'ctx> WalkerHints<'value, 'ctx, VisitorErr>
    for Walker<'value, 'ctx>
{
    type Error = BasicError;

    fn protocol(
        &mut self,
        id: ProtocolId,
    ) -> Option<AnyHint<'_, 'value, 'ctx, Self::Error, VisitorErr>> {
        match id {
            id if id == ProtocolId::of::<Reference<str>>() => Some(AnyHint::new(self)),
            _ => None,
        }
    }
}

impl<'value, 'ctx: 'value, VisitorErr: 'ctx> Hint<'value, 'ctx, Reference<str>, VisitorErr>
    for Walker<'value, 'ctx>
{
    type Error = BasicError;

    fn hint(
        &mut self,
        visitor: &mut dyn crate::Visitor<'value, 'ctx, Self::Error, Error = VisitorErr>,
        _hint: <Reference<str> as crate::protocol::Protocol<'value, 'ctx>>::Hint,
    ) -> Result<HintGiven, UniError<Self::Error, VisitorErr>> {
        lookup_visit::<Reference<str>, _, _>(visitor)
            .map_err(UniError::Walker)?
            .ok_or_else(|| {
                UniError::Walker(BasicError::new("visitor is missing the str protocol"))
            })?
            .visit(reference::Ref::Value(self.0))?;

        Ok(HintGiven)
    }

    fn known(
        &mut self,
        _hint: &<Reference<str> as crate::protocol::Protocol<'value, 'ctx>>::Hint,
    ) -> Result<<Reference<str> as crate::protocol::Protocol<'value, 'ctx>>::Known, BasicError>
    {
        Ok(reference::Known {
            len: Some(self.0.len()),
            kind: Some(reference::Kind::Context),
        })
    }
}

impl<'value, 'ctx: 'value, VisitorErr: 'ctx> WalkMut<'value, 'ctx, VisitorErr> for str {
    type ErrorMut = BasicError;
    type WalkerMut = Walker<'value, 'ctx>;

    fn walker_mut(&'value mut self) -> Self::WalkerMut {
        Walker(self, PhantomData)
    }
}

impl<'value, 'ctx: 'value, VisitorErr: 'ctx> Walk<'value, 'ctx, VisitorErr> for str {
    type Error = BasicError;
    type Walker = Walker<'value, 'ctx>;

    fn walker(&'value self) -> Self::Walker {
        Walker(self, PhantomData)
    }
}