devoyage-subgraph 0.0.15

Subgraph is a CLI that instantly generates a GraphQL API around Mongo, SQL, and HTTP APIs.
Documentation
use async_graphql::{
    dynamic::{Field, FieldFuture, FieldValue, InputValue, Object, TypeRef},
    Value,
};
use biscuit_auth::{Biscuit, KeyPair};
use log::{debug, error};
use serde::{Deserialize, Serialize};
use webauthn_rs::prelude::PublicKeyCredential;

use crate::{data_sources::DataSources, graphql::schema::ServiceSchema};

#[derive(Debug, Serialize, Deserialize, Clone)]
pub struct AuthenticateSuccess {
    pub token: String,
    pub user_uuid: String,
    pub user_identifier: String,
}

impl ServiceSchema {
    pub fn create_authenticate_finish(mut self) -> Self {
        debug!("Creating authenticate finish");

        let auth_config = match self.subgraph_config.service.auth.clone() {
            Some(auth) => auth,
            None => {
                panic!("Auth config not found.");
            }
        };

        let resolver = Field::new(
            "authenticate_finish",
            TypeRef::named_nn("authenticate_success"),
            move |ctx| {
                debug!("Resolving authenticate finish");
                let auth_config = auth_config.clone();

                FieldFuture::new(async move {
                    let data_sources = ctx.data_unchecked::<DataSources>();
                    let data_source = DataSources::get_data_source_by_name(
                        &data_sources,
                        &auth_config.data_source,
                    );
                    let key_pair = match ctx.data_unchecked::<Option<KeyPair>>() {
                        Some(key_pair) => key_pair,
                        None => {
                            error!("Failed to get key pair.");
                            return Err(async_graphql::Error::new(format!(
                                "Failed to get key pair."
                            )));
                        }
                    };

                    let identifier = match ctx.args.try_get("identifier") {
                        Ok(input) => input.deserialize::<String>().map_err(|e| {
                            error!("Failed to get input: {:?}", e);
                            async_graphql::Error::new(format!("Failed to get input: {:?}", e))
                        })?,
                        Err(e) => {
                            error!("Failed to get input: {:?}", e);
                            return Err(async_graphql::Error::new(format!(
                                "Failed to get input: {:?}",
                                e
                            )));
                        }
                    };
                    let pub_key = match ctx.args.try_get("public_key") {
                        Ok(input) => input.deserialize::<String>().map_err(|e| {
                            error!("Failed to deserialize: {:?}", e);
                            async_graphql::Error::new(format!("Failed to deserialize: {:?}", e))
                        })?,
                        Err(e) => {
                            error!("Failed to get input: {:?}", e);
                            return Err(async_graphql::Error::new(format!(
                                "Failed to get input: {:?}",
                                e
                            )));
                        }
                    };
                    let pub_key: Result<PublicKeyCredential, async_graphql::Error> =
                        serde_json::from_str(&pub_key).map_err(|e| {
                            async_graphql::Error::new(format!("Failed to deserialize: {:?}", e))
                        });

                    let pub_key = match pub_key {
                        Ok(pk) => pk,
                        Err(error) => {
                            error!("Failed to deserialize public key: {:?}", error);
                            return Err(error);
                        }
                    };

                    let user = ServiceSchema::get_user(&data_source, &identifier).await?;

                    let user = match user {
                        Some(user) => {
                            if user.passkey.is_none() {
                                error!("User does not have a passkey.");
                                return Err(async_graphql::Error::new(format!(
                                    "User does not have a passkey."
                                )));
                            };

                            if user.authentication_state.is_none() {
                                error!("User does not have an authentication state.");
                                return Err(async_graphql::Error::new(format!(
                                    "User does not have an authentication state."
                                )));
                            };
                            user
                        }
                        None => {
                            error!("User not found.");
                            return Err(async_graphql::Error::new(format!("User not found.")));
                        }
                    };

                    debug!("Authenticating Config: {:?}", &auth_config);

                    let webauthn = ServiceSchema::build_webauthn(&auth_config).map_err(|e| {
                        error!("Failed to build webauthn: {:?}", e);
                        async_graphql::Error::new(format!("Failed to build webauthn: {:?}", e))
                    })?;

                    webauthn
                        .finish_passkey_authentication(
                            &pub_key,
                            &user.authentication_state.unwrap(),
                        )
                        .map_err(|e| {
                            error!("Failed to finish authentication: {:?}", e);
                            async_graphql::Error::new(format!(
                                "Failed to finish authentication: {:?}",
                                e
                            ))
                        })?;

                    let user_uuid = user.uuid.clone().to_string();

                    let mut biscuit = Biscuit::builder();
                    biscuit
                        .add_fact(format!("user(\"{}\", \"{}\")", identifier, user_uuid).as_str())
                        .map_err(|e| {
                            error!("Failed to add fact: {:?}", e);
                            async_graphql::Error::new(format!("Failed to add fact: {:?}", e))
                        })?;
                    let biscuit = biscuit.build(key_pair).map_err(|e| {
                        error!("Failed to build biscuit: {:?}", e);
                        async_graphql::Error::new(format!("Failed to build biscuit: {:?}", e))
                    })?;
                    let base64 = biscuit.to_base64().map_err(|e| {
                        error!("Failed to convert to base64: {:?}", e);
                        async_graphql::Error::new(format!("Failed to convert to base64: {:?}", e))
                    })?;

                    let response_value = serde_json::to_value(AuthenticateSuccess {
                        token: base64.clone(),
                        user_uuid: user.uuid.clone().to_string(),
                        user_identifier: identifier.clone(),
                    })
                    .map_err(|e| {
                        error!("Failed to serialize: {:?}", e);
                        async_graphql::Error::new(format!("Failed to serialize: {:?}", e))
                    })?;

                    Ok(Some(FieldValue::owned_any(response_value)))
                })
            },
        )
        .argument(InputValue::new(
            "identifier",
            TypeRef::named_nn(TypeRef::STRING),
        ))
        .argument(InputValue::new(
            "public_key",
            TypeRef::named_nn(TypeRef::STRING),
        ));

        let authentication_success = Object::new("authenticate_success")
            .field(Field::new(
                "token",
                TypeRef::named_nn(TypeRef::STRING),
                move |ctx| {
                    FieldFuture::new(async move {
                        let parent_value = ctx
                            .parent_value
                            .try_downcast_ref::<serde_json::Value>()
                            .map_err(|e| {
                                error!("Failed to downcast: {:?}", e);
                                async_graphql::Error::new(format!("Failed to downcast: {:?}", e))
                            })?;
                        let token = parent_value["token"].as_str().ok_or_else(|| {
                            error!("Failed to get token.");
                            async_graphql::Error::new(format!("Failed to get token."))
                        })?;

                        Ok(Some(Value::from(token)))
                    })
                },
            ))
            .field(Field::new(
                "user_uuid",
                TypeRef::named_nn(TypeRef::STRING),
                move |ctx| {
                    FieldFuture::new(async move {
                        let parent_value = ctx
                            .parent_value
                            .try_downcast_ref::<serde_json::Value>()
                            .map_err(|e| {
                                error!("Failed to downcast: {:?}", e);
                                async_graphql::Error::new(format!("Failed to downcast: {:?}", e))
                            })?;
                        let user_uuid = parent_value["user_uuid"].as_str().ok_or_else(|| {
                            async_graphql::Error::new(format!("Failed to get user_uuid."))
                        })?;

                        Ok(Some(Value::from(user_uuid)))
                    })
                },
            ))
            .field(Field::new(
                "user_identifier",
                TypeRef::named_nn(TypeRef::STRING),
                move |ctx| {
                    FieldFuture::new(async move {
                        let parent_value = ctx
                            .parent_value
                            .try_downcast_ref::<serde_json::Value>()
                            .map_err(|e| {
                                error!("Failed to downcast: {:?}", e);
                                async_graphql::Error::new(format!("Failed to downcast: {:?}", e))
                            })?;
                        let user_identifier =
                            parent_value["user_identifier"].as_str().ok_or_else(|| {
                                async_graphql::Error::new(format!("Failed to get user_identifier."))
                            })?;

                        Ok(Some(Value::from(user_identifier)))
                    })
                },
            ));

        self = self.register_types(vec![authentication_success]);
        self.mutation = self.mutation.field(resolver);
        self
    }
}