Skip to content

Commit

Permalink
Add registry.toml to the source directory template assets
Browse files Browse the repository at this point in the history
  • Loading branch information
Limeth committed Aug 4, 2024
1 parent de9fbd4 commit cfbdc29
Show file tree
Hide file tree
Showing 6 changed files with 235 additions and 71 deletions.
19 changes: 11 additions & 8 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 3 additions & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -24,13 +24,15 @@ derive_more = "0.99.17"
ed25519-dalek = { version = "2.1.1", features = ["rand_core", "pem"] }
futures = "0.3.30"
include_dir = { version = "0.7.4", features = ["nightly"] }
rrr = { git = "https://github.com/recursive-record-registry/rrr.git", rev = "8c1a37007cbded8c144b6ff566656b104a7d0add" }
itertools = "0.13.0"
rrr = { git = "https://github.com/recursive-record-registry/rrr.git", rev = "c5258b43eb1d98a0a8b676d86d6f93b21fb489e2" }
serde = { version = "1.0.203", features = ["derive"] }
serde_bytes = "0.11.14"
serde_with = "3.8.1"
thiserror = "1.0.62"
tokio = { version = "1.37", features = ["full"] }
toml = { version = "0.8.14", features = ["preserve_order"] }
toml_edit = { version = "0.22.20", features = ["serde"] }
tracing = "0.1.40"
# Dependencies of the executable binary
clap = { version = "4.5", features = ["derive"], optional = true }
Expand Down
29 changes: 29 additions & 0 deletions assets/source-directory-template/registry.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
root_record_path = "root"
staging_directory_path = "target/staging"
revisions_directory_path = "target/revisions"
published_directory_path = "target/published"
signing_key_paths = ['keys/key_ed25519.pem']

[hash]
output_length_in_bytes = 32

[hash.algorithm.argon2]
variant = "argon2id"
m_cost = 1024
t_cost = 1024
p_cost = 1

[kdf]
file_name_length_in_bytes = 8
file_tag_length_in_bytes = 32
succession_nonce_length_in_bytes = 32
root_predecessor_nonce = "PLACEHOLDER"

[kdf.algorithm.hkdf]
prf = "sha256"

[default_record_parameters.splitting_strategy.fill]

[default_record_parameters.encryption]
algorithm = "Aes256Gcm"
segment_padding_to_bytes = 1024
41 changes: 40 additions & 1 deletion src/assets.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,43 @@
use include_dir::{include_dir, Dir};
use std::{
collections::HashMap,
path::{Path, PathBuf},
};

use futures::{future::BoxFuture, FutureExt};
use include_dir::{include_dir, Dir, DirEntry};
use rrr::utils::fd_lock::{FileLock, WriteLock};
use tokio::io::AsyncWriteExt;

pub const SOURCE_DIRECTORY_TEMPLATE: Dir<'_> =
include_dir!("$CARGO_MANIFEST_DIR/assets/source-directory-template");

pub(crate) fn extract_with_locks<'a>(
dir: &'a Dir<'_>,
base_path: impl AsRef<Path> + Send + Sync + 'a,
lock_map: &'a mut HashMap<PathBuf, &mut WriteLock>,
) -> BoxFuture<'a, std::io::Result<()>> {
async move {
let base_path = base_path.as_ref();

for entry in dir.entries() {
let path = base_path.join(entry.path());

match entry {
DirEntry::Dir(d) => {
tokio::fs::create_dir_all(&path).await?;
extract_with_locks(d, base_path, lock_map).await?;
}
DirEntry::File(f) => {
if let Some(lock) = lock_map.get_mut(entry.path()) {
lock.file_mut().write_all(f.contents()).await?;
} else {
tokio::fs::write(path, f.contents()).await?;
}
}
}
}

Ok(())
}
.boxed()
}
141 changes: 82 additions & 59 deletions src/owned/registry.rs
Original file line number Diff line number Diff line change
@@ -1,33 +1,33 @@
use aes_gcm::aead::OsRng;
use color_eyre::Result;
use ed25519_dalek::pkcs8::{spki::der::pem::LineEnding, DecodePrivateKey, EncodePrivateKey};
use rrr::crypto::kdf::hkdf::HkdfParams;
use rrr::crypto::kdf::KdfAlgorithm;
use rrr::crypto::password_hash::{argon2::Argon2Params, PasswordHashAlgorithm};
use itertools::Itertools;
use rrr::crypto::signature::{SigningKey, SigningKeyEd25519};
use rrr::record::RecordKey;
use rrr::registry::{RegistryConfig, RegistryConfigHash, RegistryConfigKdf};
use rrr::utils::fd_lock::{FileLock, FileLockType, ReadLock, WriteLock};
use rrr::utils::serde::Secret;
use rrr::{crypto::encryption::EncryptionAlgorithm, record::RecordKey};
use serde::{Deserialize, Serialize};
use std::collections::HashMap;
use std::io::SeekFrom;
use std::{
fmt::Debug,
ops::{Deref, DerefMut},
path::{Path, PathBuf},
};
use tokio::fs::OpenOptions;
use tokio::fs::{DirEntry, OpenOptions};
use tokio::io::AsyncSeekExt;
use tokio::{
fs::File,
io::{AsyncReadExt, AsyncWriteExt},
};
use toml_edit::DocumentMut;

use crate::assets;
use crate::error::Error;
use crate::record::{
OwnedRecordConfigEncryption, OwnedRecordConfigParameters, OwnedRecordConfigParametersUnresolved,
};
use crate::record::OwnedRecordConfigParametersUnresolved;

use super::record::{OwnedRecord, SplittingStrategy};
use super::record::OwnedRecord;

/// Represents a registry with cryptographic credentials for editing.
#[derive(Debug, Serialize, Deserialize, PartialEq, Eq)]
Expand Down Expand Up @@ -68,6 +68,8 @@ pub struct OwnedRegistry<L: FileLock> {
}

impl<L: FileLock> OwnedRegistry<L> {
const FILE_NAME_CONFIG: &str = "registry.toml";

pub async fn load(directory_path: impl Into<PathBuf>) -> Result<Self> {
let directory_path = directory_path.into();
let config_path = Self::get_config_path_from_registry_directory_path(&directory_path);
Expand Down Expand Up @@ -117,6 +119,7 @@ impl<L: FileLock> OwnedRegistry<L> {
pub async fn save_config(&mut self) -> Result<()> {
let config_string = toml::to_string_pretty(&self.config)?;

self.file_lock.file_mut().seek(SeekFrom::Start(0)).await?;
self.file_lock
.file_mut()
.write_all(config_string.as_bytes())
Expand All @@ -126,7 +129,7 @@ impl<L: FileLock> OwnedRegistry<L> {
}

fn get_config_path_from_registry_directory_path(directory_path: impl AsRef<Path>) -> PathBuf {
directory_path.as_ref().join("registry.toml")
directory_path.as_ref().join(Self::FILE_NAME_CONFIG)
}

fn get_config_path(&self) -> PathBuf {
Expand Down Expand Up @@ -221,13 +224,11 @@ impl OwnedRegistry<WriteLock> {
Err(error) => return Err(error.into()),
}

let signing_keys_directory_relative = PathBuf::from("keys");
let signing_keys_directory_absolute = directory_path.join(&signing_keys_directory_relative);
let config_path = Self::get_config_path_from_registry_directory_path(&directory_path);

// TODO: Unify with save_config
tokio::fs::create_dir_all(&directory_path).await?;
let file_lock = {
let mut file_lock = {
let open_options = {
let mut open_options = OpenOptions::new();
open_options.read(true);
Expand All @@ -239,66 +240,88 @@ impl OwnedRegistry<WriteLock> {
};
WriteLock::lock(&config_path, &open_options).await?
};
tokio::task::spawn_blocking({
let directory_path = directory_path.clone();
move || assets::SOURCE_DIRECTORY_TEMPLATE.extract(directory_path)
})
.await??;
tokio::fs::create_dir(&signing_keys_directory_absolute).await?;

{
let mut lock_map = HashMap::new();
lock_map.insert(PathBuf::from(Self::FILE_NAME_CONFIG), &mut file_lock);
assets::extract_with_locks(
&assets::SOURCE_DIRECTORY_TEMPLATE,
&directory_path,
&mut lock_map,
)
.await?;
}

let mut csprng = OsRng;
let signing_keys = vec![SigningKey::Ed25519(Secret(SigningKeyEd25519(
ed25519_dalek::SigningKey::generate(&mut csprng),
)))];
let signing_key_paths = {
let mut signing_key_paths = Vec::new();

for signing_key in &signing_keys {
let signing_key_path_relative = signing_keys_directory_relative
.join(format!("key_{}.pem", signing_key.key_type_name()));
let signing_key_path_absolute = directory_path.join(&signing_key_path_relative);
let pem = signing_key.to_pkcs8_pem(LineEnding::default()).unwrap();
let mut file = File::create_new(&signing_key_path_absolute).await?;

file.write_all(pem.as_bytes()).await?;
signing_key_paths.push(signing_key_path_relative);
}

signing_key_paths
// Patch registry config.
let config = {
let mut config_string = String::new();
file_lock.file_mut().seek(SeekFrom::Start(0)).await?;
file_lock
.file_mut()
.read_to_string(&mut config_string)
.await?;
let mut config_doc = config_string.parse::<DocumentMut>()?;
let root_predecessor_nonce =
RegistryConfigKdf::generate_random_root_predecessor_nonce(csprng, None);
let root_predecessor_nonce_string =
format!("{:02x}", root_predecessor_nonce.iter().format(""));
config_doc["kdf"]["root_predecessor_nonce"] =
toml_edit::value(root_predecessor_nonce_string);
config_string = config_doc.to_string();

// Parse patched config
let config = toml::from_str::<OwnedRegistryConfig>(&config_string)?;

// Store patched config
file_lock.file_mut().seek(SeekFrom::Start(0)).await?;
file_lock
.file_mut()
.write_all(config_string.as_bytes())
.await?;

config
};

let config = OwnedRegistryConfig {
hash: RegistryConfigHash {
algorithm: PasswordHashAlgorithm::Argon2(Argon2Params::default()),
output_length_in_bytes: Default::default(),
},
kdf: RegistryConfigKdf::builder()
.with_algorithm(KdfAlgorithm::Hkdf(HkdfParams::default()))
.build_with_random_root_predecessor_nonce(csprng)?,
default_record_parameters: OwnedRecordConfigParameters {
splitting_strategy: SplittingStrategy::Fill {},
encryption: Some(OwnedRecordConfigEncryption {
algorithm: EncryptionAlgorithm::Aes256Gcm,
segment_padding_to_bytes: 1024, // 1 KiB
}),
fn print(dir: &Path) {
for entry in dir.read_dir().unwrap() {
let entry = entry.unwrap();

dbg!(entry.path());

if entry.metadata().unwrap().is_dir() {
print(&entry.path());
}
}
.into(),
staging_directory_path: PathBuf::from("target/staging"),
revisions_directory_path: PathBuf::from("target/revisions"),
published_directory_path: PathBuf::from("target/published"),
root_record_path: PathBuf::from("root"),
signing_key_paths,
}

print(&directory_path);

// Generate signing keys.
let signing_keys = {
let signing_key = SigningKey::Ed25519(Secret(SigningKeyEd25519(
ed25519_dalek::SigningKey::generate(&mut csprng),
)));
let signing_keys_directory_relative = PathBuf::from("keys");
let signing_key_path_relative = signing_keys_directory_relative
.join(format!("key_{}.pem", signing_key.key_type_name()));
let signing_key_path_absolute = directory_path.join(&signing_key_path_relative);
let pem = signing_key.to_pkcs8_pem(LineEnding::default()).unwrap();
let mut file = File::create_new(&signing_key_path_absolute).await?;

file.write_all(pem.as_bytes()).await?;

vec![signing_key]
};

let mut registry = Self {
let registry = Self {
directory_path,
config,
signing_keys,
file_lock,
};

registry.save_config().await?;

Ok(registry)
}

Expand Down
Loading

0 comments on commit cfbdc29

Please sign in to comment.