1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
use std::path::PathBuf;
use anyhow::Result;
use serde::{Deserialize, Serialize};
use tokio::fs::{self, File};
use tokio::io::{AsyncReadExt, AsyncWriteExt};
use super::utils::home_path;
use crate::commands::auth::types::AuthorizedClient;
use crate::commands::projects::types::Project;
use crate::config::EXEC_NAME;
#[derive(Debug, Serialize, Deserialize, Clone, Default)]
pub struct Context {
pub default_project: Option<String>,
pub default_user: Option<String>,
pub override_api_url: Option<String>,
pub last_version_check: Option<(String, String)>,
#[serde(skip)]
pub current: Option<AuthorizedClient>,
#[serde(skip)]
pub project_override: Option<String>,
}
impl Context {
fn path() -> PathBuf {
home_path(".hop/context.json")
}
pub fn find_project_by_id_or_namespace(&self, id_or_namespace: String) -> Option<Project> {
self.current
.as_ref()
.and_then(|me| {
me.projects.iter().find(|p| {
p.id == id_or_namespace
|| p.namespace.to_lowercase() == id_or_namespace.to_lowercase()
})
})
.cloned()
}
pub fn find_project_by_id_or_namespace_error(&self, id_or_namespace: String) -> Project {
self.find_project_by_id_or_namespace(id_or_namespace)
.expect("Project not found, please check your spelling or switch accounts")
}
pub fn current_project(&self) -> Option<Project> {
match self.project_override.clone() {
Some(project) => Some(
self.find_project_by_id_or_namespace(project.clone())
.unwrap_or_else(|| panic!("Could not find project `{}`", project)),
),
None => self
.default_project
.clone()
.and_then(|id| self.find_project_by_id_or_namespace(id)),
}
}
pub fn current_project_error(self) -> Project {
self.current_project().unwrap_or_else(|| panic!("No project specified, run `{} projects switch` or use --project to specify a project",
EXEC_NAME))
}
pub async fn new() -> Self {
let path = Self::path();
match fs::metadata(path.clone()).await {
Ok(_) => match File::open(path).await {
Ok(mut file) => {
let mut buffer = String::new();
file.read_to_string(&mut buffer)
.await
.expect("Failed to read auth store");
serde_json::from_str(&buffer).unwrap()
}
Err(err) => {
panic!("Error opening auth file: {}", err)
}
},
Err(_) => Self::default().save().await.unwrap(),
}
}
pub async fn save(&mut self) -> Result<Self> {
if let Some(ref authorized) = self.current {
self.default_user = Some(authorized.id.clone());
}
let path = Self::path();
fs::create_dir_all(path.parent().unwrap())
.await
.expect("Failed to create auth store directory");
let mut file = File::create(path.clone())
.await
.expect("Error opening auth file:");
file.write_all(
serde_json::to_string(&self)
.expect("Failed to deserialize auth")
.as_bytes(),
)
.await
.expect("Failed to write auth store");
log::debug!("Saved context to {}", path.display());
Ok(self.clone())
}
pub fn update_command(&self) -> String {
format!("{EXEC_NAME} update")
}
}