nu_command/network/
version_check.rsuse nu_engine::command_prelude::*;
use serde::Deserialize;
use update_informer::{
http_client::{GenericHttpClient, HttpClient},
registry, Check, Package, Registry, Result as UpdateResult,
};
#[derive(Clone)]
pub struct VersionCheck;
impl Command for VersionCheck {
fn name(&self) -> &str {
"version check"
}
fn description(&self) -> &str {
"Checks to see if you have the latest version of nushell."
}
fn extra_description(&self) -> &str {
"If you're running nushell nightly, `version check` will check to see if you are running the latest nightly version. If you are running the nushell release, `version check` will check to see if you're running the latest release version."
}
fn signature(&self) -> Signature {
Signature::build("version check")
.category(Category::Platform)
.input_output_types(vec![(Type::Nothing, Type::String)])
}
fn examples(&self) -> Vec<Example> {
vec![Example {
description: "Check if you have the latest version of nushell",
example: "version check",
result: None,
}]
}
fn run(
&self,
_engine_state: &EngineState,
_stack: &mut Stack,
_call: &Call,
_input: PipelineData,
) -> Result<PipelineData, ShellError> {
let version_check = check_for_latest_nushell_version();
Ok(version_check.into_pipeline_data())
}
}
pub struct NuShellNightly;
impl Registry for NuShellNightly {
const NAME: &'static str = "nushell/nightly";
fn get_latest_version<T: HttpClient>(
http_client: GenericHttpClient<T>,
pkg: &Package,
) -> UpdateResult<Option<String>> {
#[derive(Deserialize, Debug)]
struct Response {
tag_name: String,
}
let url = format!("https://api.github.com/repos/{}/releases", pkg);
let versions = http_client
.add_header("Accept", "application/vnd.github.v3+json")
.add_header("User-Agent", "update-informer")
.get::<Vec<Response>>(&url)?;
if let Some(v) = versions.first() {
let up_through_plus = match v.tag_name.split('+').next() {
Some(v) => v,
None => &v.tag_name,
};
return Ok(Some(up_through_plus.to_string()));
}
Ok(None)
}
}
struct NativeTlsHttpClient;
impl HttpClient for NativeTlsHttpClient {
fn get<T: serde::de::DeserializeOwned>(
url: &str,
timeout: std::time::Duration,
headers: update_informer::http_client::HeaderMap,
) -> update_informer::Result<T> {
let agent = ureq::AgentBuilder::new()
.tls_connector(std::sync::Arc::new(native_tls::TlsConnector::new()?))
.build();
let mut req = agent.get(url).timeout(timeout);
for (header, value) in headers {
req = req.set(header, value);
}
let json = req.call()?.into_json()?;
Ok(json)
}
}
pub fn check_for_latest_nushell_version() -> Value {
let current_version = env!("CARGO_PKG_VERSION").to_string();
let mut rec = Record::new();
if current_version.contains("nightly") {
rec.push("channel", Value::test_string("nightly"));
let nightly_pkg_name = "nushell/nightly";
let informer =
update_informer::new(NuShellNightly, nightly_pkg_name, current_version.clone())
.http_client(NativeTlsHttpClient)
.interval(std::time::Duration::ZERO);
if let Ok(Some(new_version)) = informer.check_version() {
rec.push("current", Value::test_bool(false));
rec.push("latest", Value::test_string(format!("{}", new_version)));
Value::test_record(rec)
} else {
rec.push("current", Value::test_bool(true));
rec.push("latest", Value::test_string(current_version.clone()));
Value::test_record(rec)
}
} else {
rec.push("channel", Value::test_string("release"));
let normal_pkg_name = "nushell/nushell";
let informer =
update_informer::new(registry::GitHub, normal_pkg_name, current_version.clone())
.interval(std::time::Duration::ZERO);
if let Ok(Some(new_version)) = informer.check_version() {
rec.push("current", Value::test_bool(false));
rec.push("latest", Value::test_string(format!("{}", new_version)));
Value::test_record(rec)
} else {
rec.push("current", Value::test_bool(true));
rec.push("latest", Value::test_string(current_version.clone()));
Value::test_record(rec)
}
}
}