snarkos-node-rest 4.6.0

A REST API server for a decentralized virtual machine
Documentation
// Copyright (c) 2019-2026 Provable Inc.
// This file is part of the snarkOS library.

// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at:

// http://www.apache.org/licenses/LICENSE-2.0

// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

use super::RestError;

use axum::{
    extract::{FromRequestParts, path::ErrorKind, rejection::PathRejection},
    http::request::Parts,
};
use serde::de::DeserializeOwned;

struct PathError {
    message: String,
    path: String,
    cause: anyhow::Error,
    location: Option<String>,
}

/// Convert Path errors into the unified REST error type.
impl From<PathError> for RestError {
    fn from(val: PathError) -> Self {
        let err = if let Some(loc) = val.location {
            val.cause.context(format!("Invalid argument \"{loc}\" in path \"{}\": {}", val.path, val.message))
        } else {
            val.cause.context(format!("Invalid path \"{}\": {}", val.path, val.message))
        };

        RestError::bad_request(err)
    }
}

/// Custom Path extractor to improve errors in invalid URLs.
/// Adapted from axum's [customize-path-rejection](https://github.com/tokio-rs/axum/blob/main/examples/customize-path-rejection/src/main.rs)
pub struct Path<T>(pub T);

impl<S, T> FromRequestParts<S> for Path<T>
where
    T: DeserializeOwned + Send,
    S: Send + Sync,
{
    type Rejection = RestError;

    async fn from_request_parts(parts: &mut Parts, state: &S) -> Result<Self, Self::Rejection> {
        match axum::extract::Path::<T>::from_request_parts(parts, state).await {
            Ok(value) => Ok(Self(value.0)),
            Err(rejection) => {
                let err = match rejection {
                    PathRejection::FailedToDeserializePathParams(inner) => {
                        let kind = inner.kind();

                        let (message, location) = match &kind {
                            ErrorKind::WrongNumberOfParameters { .. } => {
                                ("wrong number of parameters".to_string(), None)
                            }
                            ErrorKind::ParseErrorAtKey { key, value, expected_type } => (
                                format!("value `{value}` is not of expected type `{expected_type}`"),
                                Some(key.clone()),
                            ),
                            ErrorKind::ParseErrorAtIndex { index, value, expected_type } => (
                                format!("value `{value}` at index {index} is not of expected type `{expected_type}`"),
                                None,
                            ),
                            ErrorKind::ParseError { value, expected_type } => {
                                (format!("value `{value}` is not of expected type `{expected_type}`"), None)
                            }
                            ErrorKind::InvalidUtf8InPathParam { key } => {
                                ("invalid UTF-8 in parameter".to_string(), Some(key.clone()))
                            }
                            ErrorKind::Message(msg) => (format!("unknown error: {msg}"), None),
                            _ => ("unknown error".to_string(), None),
                        };

                        PathError { message, path: parts.uri.path().to_string(), location, cause: inner.into() }
                    }
                    PathRejection::MissingPathParams(error) => PathError {
                        message: "missing path parameter".to_string(),
                        path: parts.uri.path().to_string(),
                        location: None,
                        cause: error.into(),
                    },
                    _ => PathError {
                        message: "unknown path error".to_string(),
                        path: parts.uri.path().to_string(),
                        location: None,
                        cause: rejection.into(),
                    },
                };

                Err(err.into())
            }
        }
    }
}