async-graphql 1.6.3

The GraphQL server library implemented by rust
Documentation
use crate::validation::visitor::{Visitor, VisitorContext};
use graphql_parser::query::{OperationDefinition, VariableDefinition};
use graphql_parser::schema::Value;
use graphql_parser::Pos;
use std::collections::HashSet;

#[derive(Default)]
pub struct NoUnusedVariables<'a> {
    vars: HashSet<(&'a str, Pos)>,
    used_vars: HashSet<&'a str>,
}

impl<'a> NoUnusedVariables<'a> {
    fn travel_value(&mut self, value: &'a Value) {
        match value {
            Value::Variable(var_name) => {
                self.used_vars.insert(var_name.as_str());
            }
            Value::List(values) => {
                for value in values {
                    self.travel_value(value);
                }
            }
            Value::Object(obj) => {
                for value in obj.values() {
                    self.travel_value(value);
                }
            }
            _ => {}
        }
    }
}

impl<'a> Visitor<'a> for NoUnusedVariables<'a> {
    fn enter_operation_definition(
        &mut self,
        _ctx: &mut VisitorContext<'a>,
        _operation_definition: &'a OperationDefinition,
    ) {
        self.used_vars.clear();
        self.vars.clear();
    }

    fn exit_operation_definition(
        &mut self,
        ctx: &mut VisitorContext<'a>,
        _operation_definition: &'a OperationDefinition,
    ) {
        for (name, pos) in &self.vars {
            if !self.used_vars.contains(name) {
                ctx.report_error(vec![*pos], format!("Variable \"${}\" is not used", name));
            }
        }
    }

    fn enter_variable_definition(
        &mut self,
        _ctx: &mut VisitorContext<'a>,
        variable_definition: &'a VariableDefinition,
    ) {
        self.vars
            .insert((&variable_definition.name, variable_definition.position));
    }

    fn enter_argument(
        &mut self,
        _ctx: &mut VisitorContext<'a>,
        _pos: Pos,
        _name: &'a str,
        value: &'a Value,
    ) {
        self.travel_value(value);
    }
}