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
// Copyright (c) 2017-present PyO3 Project and Contributors

use defs;
use func::impl_method_proto;
use method::FnSpec;
use proc_macro2::TokenStream;
use py_method;
use quote::ToTokens;
use syn;

pub fn build_py_proto(ast: &mut syn::ItemImpl) -> TokenStream {
    if let Some((_, ref mut path, _)) = ast.trait_ {
        let proto = if let Some(ref mut segment) = path.segments.last() {
            match segment.value().ident.to_string().as_str() {
                "PyObjectProtocol" => &defs::OBJECT,
                "PyAsyncProtocol" => &defs::ASYNC,
                "PyMappingProtocol" => &defs::MAPPING,
                "PyIterProtocol" => &defs::ITER,
                "PyContextProtocol" => &defs::CONTEXT,
                "PySequenceProtocol" => &defs::SEQ,
                "PyNumberProtocol" => &defs::NUM,
                "PyDescrProtocol" => &defs::DESCR,
                "PyBufferProtocol" => &defs::BUFFER,
                "PyGCProtocol" => &defs::GC,
                _ => panic!("#[pyproto] can not be used with this block"),
            }
        } else {
            panic!("#[pyproto] can only be used with protocol trait implementations")
        };

        let tokens = impl_proto_impl(&ast.self_ty, &mut ast.items, proto);

        // attach lifetime
        let mut seg = path.segments.pop().unwrap().into_value();
        seg.arguments = syn::PathArguments::AngleBracketed(parse_quote!{<'p>});
        path.segments.push(seg);
        ast.generics.params = parse_quote!{'p};

        tokens
    } else {
        panic!("#[pyproto] can only be used with protocol trait implementations")
    }
}

fn impl_proto_impl(
    ty: &syn::Type,
    impls: &mut Vec<syn::ImplItem>,
    proto: &defs::Proto,
) -> TokenStream {
    let mut tokens = TokenStream::new();
    let mut py_methods = Vec::new();

    for iimpl in impls.iter_mut() {
        match iimpl {
            syn::ImplItem::Method(ref mut met) => {
                for m in proto.methods {
                    if m.eq(met.sig.ident.to_string().as_str()) {
                        impl_method_proto(ty, &mut met.sig, m).to_tokens(&mut tokens);
                    }
                }
                for m in proto.py_methods {
                    let ident = met.sig.ident.clone();
                    if m.name == ident.to_string().as_str() {
                        let name: syn::Ident = syn::parse_str(m.name).unwrap();
                        let proto: syn::Path = syn::parse_str(m.proto).unwrap();

                        let fn_spec = FnSpec::parse(&ident, &mut met.sig, &mut met.attrs);
                        let meth = py_method::impl_proto_wrap(ty, &ident, &fn_spec);

                        py_methods.push(quote! {
                            impl #proto for #ty
                            {
                                #[inline]
                                fn #name() -> Option<::pyo3::class::methods::PyMethodDef> {
                                    #meth

                                    Some(::pyo3::class::PyMethodDef {
                                        ml_name: stringify!(#name),
                                        ml_meth: ::pyo3::class::PyMethodType::PyCFunctionWithKeywords(__wrap),
                                        ml_flags: ::pyo3::ffi::METH_VARARGS | ::pyo3::ffi::METH_KEYWORDS,
                                        ml_doc: ""})
                                }
                            }
                        });
                    }
                }
            }
            _ => (),
        }
    }

    quote! {
        #tokens

        #(#py_methods)*
    }
}