sui_package_management/
lib.rsuse anyhow::{bail, Context};
use std::collections::HashMap;
use std::fs::File;
use std::path::{Path, PathBuf};
use std::str::FromStr;
use move_core_types::account_address::AccountAddress;
use move_package::{
lock_file::{self, schema::ManagedPackage, LockFile},
resolution::resolution_graph::Package,
source_package::layout::SourcePackageLayout,
};
use move_symbol_pool::Symbol;
use sui_json_rpc_types::{get_new_package_obj_from_response, SuiTransactionBlockResponse};
use sui_sdk::wallet_context::WalletContext;
use sui_types::base_types::ObjectID;
pub mod system_package_versions;
const PUBLISHED_AT_MANIFEST_FIELD: &str = "published-at";
pub enum LockCommand {
Publish,
Upgrade,
}
#[derive(thiserror::Error, Debug, Clone)]
pub enum PublishedAtError {
#[error("The 'published-at' field in Move.toml or Move.lock is invalid: {0:?}")]
Invalid(String),
#[error("The 'published-at' field is not present in Move.toml or Move.lock")]
NotPresent,
#[error(
"Conflicting 'published-at' addresses between Move.toml -- {id_manifest} -- and \
Move.lock -- {id_lock}"
)]
Conflict {
id_lock: ObjectID,
id_manifest: ObjectID,
},
}
pub async fn update_lock_file(
context: &WalletContext,
command: LockCommand,
install_dir: Option<PathBuf>,
lock_file: Option<PathBuf>,
response: &SuiTransactionBlockResponse,
) -> Result<(), anyhow::Error> {
let chain_identifier = context
.get_client()
.await
.context("Network issue: couldn't use client to connect to chain when updating Move.lock")?
.read_api()
.get_chain_identifier()
.await
.context("Network issue: couldn't determine chain identifier for updating Move.lock")?;
let (original_id, version, _) = get_new_package_obj_from_response(response).context(
"Expected a valid published package response but didn't see \
one when attempting to update the `Move.lock`.",
)?;
let Some(lock_file) = lock_file else {
bail!(
"Expected a `Move.lock` file to exist after publishing \
package, but none found. Consider running `sui move build` to \
generate the `Move.lock` file in the package directory."
)
};
let install_dir = install_dir.unwrap_or(PathBuf::from("."));
let env = context.config.get_active_env().context(
"Could not resolve environment from active wallet context. \
Try ensure `sui client active-env` is valid.",
)?;
let mut lock = LockFile::from(install_dir.clone(), &lock_file)?;
match command {
LockCommand::Publish => lock_file::schema::update_managed_address(
&mut lock,
&env.alias,
lock_file::schema::ManagedAddressUpdate::Published {
chain_id: chain_identifier,
original_id: original_id.to_string(),
},
),
LockCommand::Upgrade => lock_file::schema::update_managed_address(
&mut lock,
&env.alias,
lock_file::schema::ManagedAddressUpdate::Upgraded {
latest_id: original_id.to_string(),
version: version.into(),
},
),
}?;
lock.commit(lock_file)?;
Ok(())
}
pub fn set_package_id(
package_path: &Path,
install_dir: Option<PathBuf>,
chain_id: &String,
id: AccountAddress,
) -> Result<Option<AccountAddress>, anyhow::Error> {
let lock_file_path = package_path.join(SourcePackageLayout::Lock.path());
let Ok(mut lock_file) = File::open(lock_file_path.clone()) else {
return Ok(None);
};
let managed_package = ManagedPackage::read(&mut lock_file)
.ok()
.and_then(|m| m.into_iter().find(|(_, v)| v.chain_id == *chain_id));
let Some((env, v)) = managed_package else {
return Ok(None);
};
let install_dir = install_dir.unwrap_or(PathBuf::from("."));
let lock_for_update = LockFile::from(install_dir.clone(), &lock_file_path);
let Ok(mut lock_for_update) = lock_for_update else {
return Ok(None);
};
lock_file::schema::set_original_id(&mut lock_for_update, &env, &id.to_canonical_string(true))?;
lock_for_update.commit(lock_file_path)?;
let id = AccountAddress::from_str(&v.original_published_id)?;
Ok(Some(id))
}
pub fn resolve_published_id(
package: &Package,
chain_id: Option<String>,
) -> Result<ObjectID, PublishedAtError> {
let published_id_in_manifest = manifest_published_at(package);
match published_id_in_manifest {
Ok(_) | Err(PublishedAtError::NotPresent) => { }
Err(e) => {
return Err(e);
}
}
let lock = package.package_path.join(SourcePackageLayout::Lock.path());
let Ok(mut lock_file) = File::open(lock.clone()) else {
return published_id_in_manifest;
};
let id_in_lock_for_chain_id =
lock_published_at(ManagedPackage::read(&mut lock_file).ok(), chain_id.as_ref());
match (id_in_lock_for_chain_id, published_id_in_manifest) {
(Ok(id_lock), Ok(id_manifest)) if id_lock != id_manifest => {
Err(PublishedAtError::Conflict {
id_lock,
id_manifest,
})
}
(Ok(id), _) | (_, Ok(id)) => Ok(id),
(from_lock, Err(_)) => from_lock,
}
}
fn manifest_published_at(package: &Package) -> Result<ObjectID, PublishedAtError> {
let Some(value) = package
.source_package
.package
.custom_properties
.get(&Symbol::from(PUBLISHED_AT_MANIFEST_FIELD))
else {
return Err(PublishedAtError::NotPresent);
};
let id =
ObjectID::from_str(value.as_str()).map_err(|_| PublishedAtError::Invalid(value.clone()))?;
if id == ObjectID::ZERO {
Err(PublishedAtError::NotPresent)
} else {
Ok(id)
}
}
fn lock_published_at(
lock: Option<HashMap<String, ManagedPackage>>,
chain_id: Option<&String>,
) -> Result<ObjectID, PublishedAtError> {
let (Some(lock), Some(chain_id)) = (lock, chain_id) else {
return Err(PublishedAtError::NotPresent);
};
let managed_package = lock
.into_values()
.find(|v| v.chain_id == *chain_id)
.ok_or(PublishedAtError::NotPresent)?;
let id = ObjectID::from_str(managed_package.latest_published_id.as_str())
.map_err(|_| PublishedAtError::Invalid(managed_package.latest_published_id.clone()))?;
if id == ObjectID::ZERO {
Err(PublishedAtError::NotPresent)
} else {
Ok(id)
}
}