remote-diff draft
This commit is contained in:
parent
dfd42389f3
commit
30004ebd8b
@ -5,3 +5,4 @@ pub mod reset;
|
|||||||
pub mod clone;
|
pub mod clone;
|
||||||
pub mod push;
|
pub mod push;
|
||||||
pub mod config;
|
pub mod config;
|
||||||
|
pub mod remote_diff;
|
||||||
|
94
src/commands/remote_diff.rs
Normal file
94
src/commands/remote_diff.rs
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
use crate::services::api::ApiError;
|
||||||
|
use crate::services::req_props::{ReqProps, ObjProps};
|
||||||
|
use crate::store::object::Object;
|
||||||
|
use crate::utils::path;
|
||||||
|
use std::fs::canonicalize;
|
||||||
|
use std::path::PathBuf;
|
||||||
|
|
||||||
|
pub struct RemoteDiffArgs {
|
||||||
|
pub path: Option<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn remote_diff(args: RemoteDiffArgs) {
|
||||||
|
let path = {
|
||||||
|
if let Some(path) = args.path {
|
||||||
|
let mut cur = path::current().unwrap();
|
||||||
|
cur.push(path);
|
||||||
|
let canonic = canonicalize(cur).ok().unwrap();
|
||||||
|
dbg!(&canonic);
|
||||||
|
let ok = canonic.strip_prefix(path::repo_root());
|
||||||
|
dbg!(&ok);
|
||||||
|
|
||||||
|
PathBuf::from("/")
|
||||||
|
} else {
|
||||||
|
PathBuf::from("/")
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut folders: Vec<ObjProps> = vec![ObjProps {
|
||||||
|
contentlength: None,
|
||||||
|
href: None,
|
||||||
|
lastmodified: None,
|
||||||
|
relative_s: Some(path.to_str().unwrap().to_owned()),
|
||||||
|
}];
|
||||||
|
let mut files: Vec<ObjProps> = vec![];
|
||||||
|
|
||||||
|
while folders.len() > 0 {
|
||||||
|
let folder = folders.pop().unwrap();
|
||||||
|
|
||||||
|
let res = ReqProps::new()
|
||||||
|
.set_url(&folder.relative_s.unwrap())
|
||||||
|
.gethref()
|
||||||
|
.getlastmodified()
|
||||||
|
.send_req_multiple();
|
||||||
|
|
||||||
|
let objs = match res {
|
||||||
|
Ok(o) => o,
|
||||||
|
Err(ApiError::IncorrectRequest(err)) => {
|
||||||
|
eprintln!("fatal: {}", err.status());
|
||||||
|
std::process::exit(1);
|
||||||
|
},
|
||||||
|
Err(ApiError::EmptyError(_)) => {
|
||||||
|
eprintln!("Failed to get body");
|
||||||
|
vec![]
|
||||||
|
}
|
||||||
|
Err(ApiError::RequestError(err)) => {
|
||||||
|
eprintln!("fatal: {}", err);
|
||||||
|
std::process::exit(1);
|
||||||
|
},
|
||||||
|
Err(ApiError::Unexpected(_)) => todo!()
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
let mut iter = objs.iter();
|
||||||
|
// todo opti store date of root
|
||||||
|
let root = iter.next();
|
||||||
|
|
||||||
|
for obj in iter {
|
||||||
|
let mut o = Object::new(&obj.clone().relative_s.unwrap());
|
||||||
|
let exist = o.exists();
|
||||||
|
|
||||||
|
let should_pull = {
|
||||||
|
if exist {
|
||||||
|
o.read()
|
||||||
|
.is_older(obj.lastmodified.unwrap().timestamp())
|
||||||
|
} else {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if should_pull {
|
||||||
|
println!("should pull {}", obj.clone().relative_s.unwrap());
|
||||||
|
if obj.href.clone().unwrap().chars().last().unwrap() == '/' {
|
||||||
|
folders.push(obj.clone());
|
||||||
|
} else {
|
||||||
|
files.push(obj.clone());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
31
src/main.rs
31
src/main.rs
@ -1,6 +1,8 @@
|
|||||||
use clap::{App, Arg, SubCommand};
|
use clap::{App, Arg, SubCommand};
|
||||||
use textwrap::{fill, Options};
|
use textwrap::{fill, Options};
|
||||||
|
|
||||||
use crate::commands::add::AddArgs;
|
use crate::commands::add::AddArgs;
|
||||||
|
use crate::commands::remote_diff::RemoteDiffArgs;
|
||||||
|
|
||||||
mod commands;
|
mod commands;
|
||||||
mod utils;
|
mod utils;
|
||||||
@ -96,6 +98,20 @@ fn main() {
|
|||||||
.value_name("VALUE")
|
.value_name("VALUE")
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
.subcommand(
|
||||||
|
SubCommand::with_name("remote-diff")
|
||||||
|
.arg(
|
||||||
|
Arg::with_name("path")
|
||||||
|
.required(false)
|
||||||
|
.takes_value(true)
|
||||||
|
.value_name("PATH")
|
||||||
|
.help("The path to pull."),
|
||||||
|
)
|
||||||
|
.about("Fetch new and modifed files from the nextcloud server.")
|
||||||
|
)
|
||||||
|
.subcommand(
|
||||||
|
SubCommand::with_name("test")
|
||||||
|
)
|
||||||
.get_matches();
|
.get_matches();
|
||||||
|
|
||||||
if let Some(matches) = matches.subcommand_matches("init") {
|
if let Some(matches) = matches.subcommand_matches("init") {
|
||||||
@ -134,6 +150,21 @@ fn main() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else if let Some(matches) = matches.subcommand_matches("remote-diff") {
|
||||||
|
commands::remote_diff::remote_diff(RemoteDiffArgs {
|
||||||
|
path: {
|
||||||
|
if let Some(mut path) = matches.values_of("path") {
|
||||||
|
match path.next() {
|
||||||
|
Some(p) => Some(String::from(p)),
|
||||||
|
None => None,
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} else if let Some(_) = matches.subcommand_matches("test") {
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -63,7 +63,9 @@ impl ApiBuilder {
|
|||||||
url.push_str(username.unwrap());
|
url.push_str(username.unwrap());
|
||||||
url.push_str(&root);
|
url.push_str(&root);
|
||||||
url.push_str("/");
|
url.push_str("/");
|
||||||
|
if path != "/" {
|
||||||
url.push_str(path);
|
url.push_str(path);
|
||||||
|
}
|
||||||
self.request = Some(self.client.request(method, url));
|
self.request = Some(self.client.request(method, url));
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
@ -75,7 +77,9 @@ impl ApiBuilder {
|
|||||||
url.push_str(&api_props.username);
|
url.push_str(&api_props.username);
|
||||||
url.push_str(&api_props.root);
|
url.push_str(&api_props.root);
|
||||||
url.push_str("/");
|
url.push_str("/");
|
||||||
|
if p != "/" {
|
||||||
url.push_str(p);
|
url.push_str(p);
|
||||||
|
}
|
||||||
self.request = Some(self.client.request(meth, url));
|
self.request = Some(self.client.request(meth, url));
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
@ -3,6 +3,8 @@ use chrono::{Utc, DateTime};
|
|||||||
use reqwest::{Method, Response, Error};
|
use reqwest::{Method, Response, Error};
|
||||||
use xml::reader::{EventReader, XmlEvent};
|
use xml::reader::{EventReader, XmlEvent};
|
||||||
use reqwest::header::HeaderValue;
|
use reqwest::header::HeaderValue;
|
||||||
|
use crate::commands::clone::get_url_props;
|
||||||
|
use crate::commands::config;
|
||||||
use crate::utils::time::parse_timestamp;
|
use crate::utils::time::parse_timestamp;
|
||||||
use crate::utils::api::{get_relative_s, ApiProps};
|
use crate::utils::api::{get_relative_s, ApiProps};
|
||||||
use crate::services::api::{ApiBuilder, ApiError};
|
use crate::services::api::{ApiBuilder, ApiError};
|
||||||
@ -55,6 +57,19 @@ impl ReqProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_url(&mut self, url: &str) -> &mut ReqProps {
|
pub fn set_url(&mut self, url: &str) -> &mut ReqProps {
|
||||||
|
let remote = match config::get("remote") {
|
||||||
|
Some(r) => r,
|
||||||
|
None => {
|
||||||
|
eprintln!("fatal: unable to find a remote");
|
||||||
|
std::process::exit(1);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
let (host, username, root) = get_url_props(&remote);
|
||||||
|
self.api_props = Some(ApiProps {
|
||||||
|
host,
|
||||||
|
username: username.unwrap().to_owned(),
|
||||||
|
root: root.to_owned(),
|
||||||
|
});
|
||||||
self.api_builder.build_request(Method::from_bytes(b"PROPFIND").unwrap(), url);
|
self.api_builder.build_request(Method::from_bytes(b"PROPFIND").unwrap(), url);
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
@ -1,10 +1,16 @@
|
|||||||
use std::fs::OpenOptions;
|
use std::fs::OpenOptions;
|
||||||
|
use std::path::PathBuf;
|
||||||
use std::io::{self, Write};
|
use std::io::{self, Write};
|
||||||
use crate::utils::{read, path};
|
use crate::utils::{read, path};
|
||||||
|
|
||||||
pub fn add_line(line: String) -> io::Result<()> {
|
pub fn path() -> PathBuf {
|
||||||
let mut root = path::nextsync();
|
let mut root = path::nextsync();
|
||||||
root.push("HEAD");
|
root.push("HEAD");
|
||||||
|
root
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn add_line(line: String) -> io::Result<()> {
|
||||||
|
let root = path();
|
||||||
|
|
||||||
let mut file = OpenOptions::new()
|
let mut file = OpenOptions::new()
|
||||||
.read(true)
|
.read(true)
|
||||||
@ -17,8 +23,7 @@ pub fn add_line(line: String) -> io::Result<()> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn rm_line(line: &str) -> io::Result<()> {
|
pub fn rm_line(line: &str) -> io::Result<()> {
|
||||||
let mut root = path::nextsync();
|
let root = path();
|
||||||
root.push("HEAD");
|
|
||||||
read::rm_line(root, line)?;
|
read::rm_line(root, line)?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -4,11 +4,84 @@ use std::fs::{self, OpenOptions};
|
|||||||
use crypto::sha1::Sha1;
|
use crypto::sha1::Sha1;
|
||||||
use crypto::digest::Digest;
|
use crypto::digest::Digest;
|
||||||
use std::io::{Seek, SeekFrom, Read};
|
use std::io::{Seek, SeekFrom, Read};
|
||||||
|
use crate::utils::time::parse_timestamp;
|
||||||
|
use crate::store::head;
|
||||||
use crate::utils::{read, path};
|
use crate::utils::{read, path};
|
||||||
|
|
||||||
pub mod tree;
|
pub mod tree;
|
||||||
pub mod blob;
|
pub mod blob;
|
||||||
|
|
||||||
|
pub struct Object {
|
||||||
|
path: PathBuf,
|
||||||
|
hash: String,
|
||||||
|
obj_p: PathBuf,
|
||||||
|
ts: Option<i64>
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Object {
|
||||||
|
pub fn new(path: &str) -> Object {
|
||||||
|
let path = match path.chars().next_back() == "/".chars().next() {
|
||||||
|
true => {
|
||||||
|
let mut new = path.chars();
|
||||||
|
new.next_back();
|
||||||
|
new.as_str()
|
||||||
|
},
|
||||||
|
false => path,
|
||||||
|
};
|
||||||
|
if path == "" {
|
||||||
|
return Object {
|
||||||
|
path: PathBuf::from("/"),
|
||||||
|
hash: String::from(""),
|
||||||
|
obj_p: head::path(),
|
||||||
|
ts: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut hasher = Sha1::new();
|
||||||
|
hasher.input_str(path);
|
||||||
|
let hash = hasher.result_str();
|
||||||
|
|
||||||
|
let (dir, res) = hash.split_at(2);
|
||||||
|
|
||||||
|
let mut obj_p = path::objects();
|
||||||
|
obj_p.push(dir);
|
||||||
|
obj_p.push(res);
|
||||||
|
|
||||||
|
Object {
|
||||||
|
path: PathBuf::from(path),
|
||||||
|
hash,
|
||||||
|
obj_p,
|
||||||
|
ts: None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn read(&mut self) -> &mut Object {
|
||||||
|
match read::read_lines(&self.obj_p) {
|
||||||
|
Ok(mut reader) => {
|
||||||
|
if let Some(Ok(line)) = reader.next() {
|
||||||
|
let mut data = line.rsplit(' ');
|
||||||
|
if data.clone().count() >= 2 {
|
||||||
|
self.ts = Some(data.next().unwrap().parse::<i64>().unwrap())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Err(err) => {
|
||||||
|
eprintln!("error reading object {}: {}", self.obj_p.display(), err);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn exists(&mut self) -> bool {
|
||||||
|
self.obj_p.exists()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn is_older(&mut self, ts: i64) -> bool {
|
||||||
|
// todo be aware of the diff of ts format
|
||||||
|
ts > self.ts.expect("Should be read before used") / 1000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns (line, hash, name)
|
/// Returns (line, hash, name)
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
|
Loading…
Reference in New Issue
Block a user