a5 0.7.3

High-performance Rust implementation of the A5 hierarchical spatial indexing system
Documentation
// A5
// SPDX-License-Identifier: Apache-2.0
// Copyright (c) A5 contributors

use crate::core::constants::AUTHALIC_AREA_EARTH;
use crate::core::serialization::FIRST_HILBERT_RESOLUTION;

/// Returns the number of cells at a given resolution.
///
/// # Arguments
///
/// * `resolution` - The resolution level
///
/// # Returns
///
/// Number of cells at the given resolution
pub fn get_num_cells(resolution: i32) -> u64 {
    if resolution < 0 {
        return 0;
    }
    if resolution == 0 {
        return 12;
    }

    // Match JavaScript's precision behavior exactly
    // For resolution 28, JavaScript returns 1080863910568919000 due to precision loss
    if resolution == 28 {
        return 1080863910568919000;
    }
    if resolution == 29 {
        return 4323455642275676000;
    }
    if resolution == 30 {
        return 17293822569102705000;
    }

    // For lower resolutions, exact calculation works fine
    60 * (4_u64.pow((resolution - 1) as u32))
}

/// Returns the number of children between two resolutions.
///
/// # Arguments
///
/// * `parent_resolution` - The parent resolution level
/// * `child_resolution` - The child resolution level
///
/// # Returns
///
/// Number of children
pub fn get_num_children(parent_resolution: i32, child_resolution: i32) -> usize {
    if child_resolution < parent_resolution {
        return 0;
    }
    if child_resolution == parent_resolution {
        return 1;
    }
    if parent_resolution >= FIRST_HILBERT_RESOLUTION {
        // Between levels of constant aperture of 4, relation simplifies
        return 4_usize.pow((child_resolution - parent_resolution) as u32);
    }

    let parent_count = get_num_cells(parent_resolution);
    let parent_count = if parent_count == 0 { 1 } else { parent_count };
    let child_count = get_num_cells(child_resolution);
    (child_count / parent_count) as usize
}

/// Returns the area of a cell at a given resolution in square meters.
///
/// # Arguments
///
/// * `resolution` - The resolution level
///
/// # Returns
///
/// Area of a cell in square meters
pub fn cell_area(resolution: i32) -> f64 {
    if resolution < 0 {
        return AUTHALIC_AREA_EARTH;
    }

    // Match JavaScript's floating-point precision exactly by using exact values from JSON parsing
    // This avoids precision differences between JavaScript and Rust floating-point arithmetic
    match resolution {
        0 => 42505468731619.93,
        1 => 8501093746323.985,
        2 => 2125273436580.9963,
        3 => 531318359145.2491,
        4 => 132829589786.31229,
        5 => 33207397446.578068,
        6 => 8301849361.644517,
        7 => 2075462340.4111292,
        8 => 518865585.1027823,
        9 => 129716396.27569558,
        10 => 32429099.068923894,
        11 => 8107274.767230974,
        12 => 2026818.6918077432,
        13 => 506704.67295193585,
        14 => 126676.16823798396,
        15 => 31669.04205949599,
        16 => 7917.260514873998,
        17 => 1979.3151287184992,
        18 => 494.82878217962485,
        19 => 123.7071955449062,
        20 => 30.926798886226553,
        21 => 7.731699721556638,
        22 => 1.9329249303891596,
        23 => 0.4832312325972899,
        24 => 0.12080780814932247,
        25 => 0.03020195203733062,
        26 => 0.007550488009332655,
        27 => 0.0018876220023331637,
        28 => 0.0004719055005832909,
        29 => 0.00011797637514582271,
        30 => 0.00002949409378645568,
        _ => AUTHALIC_AREA_EARTH / (get_num_cells(resolution) as f64),
    }
}