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
use pyo3::{FromPyObject, PyAny, PyResult};
use crate::codegen::Node;
use proc_macro2::TokenStream;
use quote::{quote, format_ident};

use crate::tree::{ExprType, Name};
use crate::codegen::{CodeGen, PythonOptions, CodeGenContext};
use crate::symbols::{SymbolTableScopes, SymbolTableNode};

use serde::{Serialize, Deserialize};

#[derive(Clone, Debug, Serialize, Deserialize, PartialEq)]
pub struct Assign {
    pub targets: Vec<Name>,
    pub value: ExprType,
    pub type_comment: Option<String>,
}

impl<'a> FromPyObject<'a> for Assign {
    fn extract(ob: &'a PyAny) -> PyResult<Self> {
        let targets: Vec<Name> = ob.getattr("targets").expect(
            ob.error_message("<unknown>", "error getting unary operator").as_str()
        ).extract().expect("1");

        let python_value = ob.getattr("value").expect(
            ob.error_message("<unknown>", "assignment statement value not found").as_str()
        );

        let value = ExprType::extract(python_value).expect(
            ob.error_message("<unknown>", "error getting value of assignment statement").as_str()
        );

        Ok(Assign{
            targets: targets,
            value: value,
            type_comment: None,
        })
    }
}

impl<'a> CodeGen for Assign {
    type Context = CodeGenContext;
    type Options = PythonOptions;
    type SymbolTable = SymbolTableScopes;

    fn find_symbols(self, symbols: Self::SymbolTable) -> Self::SymbolTable {
        let mut symbols = symbols;
        let mut position = 0;
        for target in self.targets {
            symbols.insert(target.id, SymbolTableNode::Assign{position: position, value: self.value.clone()});
            position += 1;
        }
        symbols
    }

    fn to_rust(self, ctx: Self::Context, options: Self::Options, symbols: Self::SymbolTable) -> Result<TokenStream, Box<dyn std::error::Error>> {
        let mut stream = TokenStream::new();
        for target in self.targets.into_iter().map(|n| n.id) {
            let ident = format_ident!("{}", target);
            stream.extend(quote!(#ident));
        };
        let value = self.value.to_rust(ctx, options, symbols)?;
        Ok(quote!(#stream = #value))
    }
}