Page Menu
Home
DevCentral
Search
Configure Global Search
Log In
Files
F3985934
D1871.diff
No One
Temporary
Actions
View File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Flag For Later
Size
6 KB
Referenced Files
None
Subscribers
None
D1871.diff
View Options
diff --git a/Cargo.toml b/Cargo.toml
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -6,8 +6,10 @@
[dependencies]
env_logger = "^0.5.13"
+lazy_static = "1.1.0"
limiting-factor = { path = "../limiting-factor", features = ["minimal"] }
log = "^0.4.5"
+regex = "1.0.5"
rocket = "^0.3.16"
rocket_codegen = "^0.3.16"
rocket_contrib = { version = "*", features = ["json"] }
diff --git a/src/app.rs b/src/app.rs
--- a/src/app.rs
+++ b/src/app.rs
@@ -9,6 +9,7 @@
status,
favicon,
get_registry_stats,
+ get_repository_info,
];
MinimalApplication::start_application(routes);
diff --git a/src/registry.rs b/src/registry.rs
--- a/src/registry.rs
+++ b/src/registry.rs
@@ -1,6 +1,8 @@
+use lazy_static::lazy_static;
+use regex::Regex;
use serde_derive::{Deserialize, Serialize};
-use std::fs::{DirEntry, read_dir};
-use std::io::Result as IOResult;
+use std::fs::{DirEntry, File, read_dir};
+use std::io::{Read, Result as IOResult};
use std::path::Path;
/* -------------------------------------------------------------
@@ -9,7 +11,7 @@
/// Represents a Docker registry
#[serde(rename_all = "camelCase")]
-#[derive(Serialize, Deserialize)]
+#[derive(Serialize, Deserialize, Clone)]
pub struct Registry {
#[serde(skip_serializing)]
pub directory: String,
@@ -40,7 +42,7 @@
}
pub fn count_repositories (&self) -> i32 {
- let path_name = format!("{}/docker/registry/v2/repositories", self.directory);
+ let path_name = self.get_repositories_path();
let path = Path::new(&path_name);
if path.exists() && path.is_dir() {
@@ -60,6 +62,116 @@
0
}
}
+
+ pub fn get_repositories_path (&self) -> String {
+ format!("{}/docker/registry/v2/repositories", self.directory)
+ }
+
+ pub fn get_repository (&self, repository_name: &str) -> Option<Repository> {
+ if !Repository::is_valid_name(repository_name) {
+ return None
+ }
+
+ let path = Path::new(&self.get_repositories_path()).join(repository_name);
+
+ let directory = match path.as_path().to_str() {
+ Some(name) => String::from(name),
+ None => { return None; }
+ };
+
+ let mut repository = Repository {
+ directory,
+ name: String::from(repository_name),
+ tags: Vec::new(),
+ };
+
+ repository.update_tags();
+
+ Some(repository)
+ }
+}
+
+/* -------------------------------------------------------------
+ Repository
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
+
+/// Represents a repository from the Docker registry
+#[serde(rename_all = "camelCase")]
+#[derive(Serialize, Deserialize)]
+pub struct Repository {
+ #[serde(skip_serializing)]
+ pub directory: String,
+
+ pub name: String,
+ pub tags: Vec<Tag>,
+}
+
+impl Repository {
+ pub fn exists(&self) -> bool {
+ let path = Path::new(&self.directory);
+
+ path.exists() && path.is_dir()
+ }
+
+ pub fn is_valid_name(name: &str) -> bool {
+ lazy_static! {
+ static ref RE: Regex = Regex::new("^/?[a-zA-Z0-9_-]+$").unwrap();
+ }
+
+ RE.is_match(name) && name.len() <= 30
+ }
+
+ pub fn update_tags(&mut self) {
+ let path = Path::new(&self.directory).join("_manifests/tags");
+
+ let tag_names = get_subdirectories_names(&path);
+ self.tags = tag_names.iter()
+ .map(|name| Tag {
+ name: name.clone(),
+ hash: self.get_hash_for_tag(&name).unwrap_or(String::new()),
+ })
+ .collect();
+ }
+
+ fn get_hash_for_tag(&self, tag_name: &str) -> IOResult<String> {
+ let mut buffer = String::new();
+
+ let path = Path::new(&self.directory)
+ .join("_manifests/tags")
+ .join(tag_name)
+ .join("current/link");
+
+ let mut f = File::open(path)?;
+ f.read_to_string(&mut buffer)?;
+
+ buffer = Tag::clean_tag(&buffer);
+
+ Ok(buffer)
+ }
+}
+
+/* -------------------------------------------------------------
+ Tag
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
+
+/// Represents a repository tag
+#[serde(rename_all = "camelCase")]
+#[derive(Serialize, Deserialize, Clone)]
+pub struct Tag {
+ pub name: String,
+ pub hash: String,
+}
+
+impl Tag {
+ pub fn clean_tag (tag: &str) -> String {
+ let fragments: Vec<&str> = tag.split(":").collect();
+
+ if fragments.len() == 1 {
+ String::from(tag)
+ } else {
+ String::from(fragments[1])
+ }
+ }
}
@@ -81,3 +193,30 @@
Err(_) => false,
}
}
+
+fn get_entry_name(entry: &IOResult<DirEntry>) -> String {
+ match entry {
+ Ok(e) => match e.file_name().into_string() {
+ Ok(name) => String::from(name),
+ Err(_) => String::new(),
+ }
+ Err(_) => String::new(),
+ }
+}
+
+fn get_subdirectories_names (dir: &Path) -> Vec<String> {
+ match std::fs::read_dir(dir) {
+ Ok(iterator) => {
+ iterator
+ .filter(|entry| is_entry_sub_directory(entry))
+ .map(|entry| get_entry_name(&entry))
+ .filter(|name| name != "")
+ .collect::<Vec<_>>()
+ },
+ Err(_) => {
+ error!("Can't get subdirectories of {:?}", dir);
+
+ Vec::new()
+ }
+ }
+}
diff --git a/src/requests.rs b/src/requests.rs
--- a/src/requests.rs
+++ b/src/requests.rs
@@ -1,9 +1,9 @@
//!
-//! Requests handled by the microservice.
+//! HTTP requests handled by the microservice.
//!
-use crate::registry::Registry;
-use limiting_factor::api::replies::{ApiJsonResponse, ApiResponse};
+use crate::registry::{Registry, Repository};
+use limiting_factor::api::replies::*;
use rocket::response::NamedFile;
#[get("/status")]
@@ -25,3 +25,24 @@
Registry::with_default_location()
.into_json_response()
}
+
+// -------------------------------------------------------------
+// /repository
+// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
+
+#[get("/repository/<repository_name>")]
+pub fn get_repository_info(repository_name: String) -> ApiJsonResponse<Repository> {
+ let repository = Registry::with_default_location()
+ .get_repository(&repository_name);
+
+ match repository {
+ None => Err(build_bad_request_response()),
+ Some(repo) => {
+ if repo.exists() {
+ Ok(repo.into_json_response()?)
+ } else {
+ Err(build_not_found_response())
+ }
+ }
+ }
+}
File Metadata
Details
Attached
Mime Type
text/plain
Expires
Thu, Jan 9, 06:34 (46 m, 20 s)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
2340882
Default Alt Text
D1871.diff (6 KB)
Attached To
Mode
D1871: Provide repository information
Attached
Detach File
Event Timeline
Log In to Comment