Skip to content

Commit

Permalink
Expose wallet migration to node wrapper (#978)
Browse files Browse the repository at this point in the history
* Added capability of migrating wallet through the node.js wrapper

Signed-off-by: Bogdan Mircea <[email protected]>
  • Loading branch information
bobozaur authored Sep 27, 2023
1 parent 62ea7b1 commit ac35fdd
Show file tree
Hide file tree
Showing 10 changed files with 120 additions and 4 deletions.
2 changes: 2 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions libvcx_core/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ uuid = { version = "0.7.4", default-features = false, features = ["v4"] }
agency_client = { path = "../agency_client" }
async-trait = "0.1.61"
url = "2.3.1"
wallet_migrator = { path = "../wallet_migrator" }

[dev-dependencies]
tokio = { version = "1.20", features = [ "rt", "macros" ] }
54 changes: 52 additions & 2 deletions libvcx_core/src/api_vcx/api_global/wallet.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ use aries_vcx::{
base_wallet::BaseWallet,
indy::{
internal::{close_search_wallet, fetch_next_records_wallet, open_search_wallet},
wallet::import,
wallet::{close_wallet, create_and_open_wallet, delete_wallet, import},
IndySdkWallet, IssuerConfig, RestoreWalletConfigs, WalletConfig,
},
},
Expand All @@ -31,7 +31,8 @@ use crate::{
get_main_anoncreds, get_main_indy_ledger_write, get_main_wallet, try_get_main_wallet,
},
errors::{
error::LibvcxResult, mapping_from_ariesvcx::map_ariesvcx_result,
error::{LibvcxError, LibvcxErrorKind, LibvcxResult},
mapping_from_ariesvcx::map_ariesvcx_result,
mapping_from_ariesvcxcore::map_ariesvcx_core_result,
},
};
Expand Down Expand Up @@ -279,6 +280,32 @@ pub async fn wallet_import(config: &RestoreWalletConfigs) -> LibvcxResult<()> {
map_ariesvcx_core_result(import(config).await)
}

pub async fn wallet_migrate(wallet_config: &WalletConfig) -> LibvcxResult<()> {
let src_wallet_handle = get_main_wallet_handle()?;
let dest_wallet_handle = create_and_open_wallet(wallet_config).await?;

let migration_res = wallet_migrator::migrate_wallet(
src_wallet_handle,
dest_wallet_handle,
wallet_migrator::vdrtools2credx::migrate_any_record,
)
.await;

if let Err(e) = migration_res {
close_wallet(dest_wallet_handle).await.ok();
delete_wallet(wallet_config).await.ok();
Err(LibvcxError::from_msg(
LibvcxErrorKind::WalletMigrationFailed,
e,
))
} else {
setup_wallet(dest_wallet_handle)?;
close_wallet(src_wallet_handle).await?;
Ok(())
}
}

#[allow(clippy::unwrap_used)]
pub mod test_utils {
use aries_vcx::{
aries_vcx_core::wallet::indy::WalletConfig,
Expand Down Expand Up @@ -375,6 +402,29 @@ pub mod tests {
errors::error::{LibvcxErrorKind, LibvcxResult},
};

#[tokio::test]
async fn test_wallet_migrate() {
let wallet_name = format!("test_create_wallet_{}", uuid::Uuid::new_v4());
let config: WalletConfig = serde_json::from_value(json!({
"wallet_name": wallet_name,
"wallet_key": DEFAULT_WALLET_KEY,
"wallet_key_derivation": WALLET_KDF_RAW
}))
.unwrap();

create_and_open_as_main_wallet(&config).await.unwrap();

let wallet_name = format!("test_migrate_wallet_{}", uuid::Uuid::new_v4());
let new_config: WalletConfig = serde_json::from_value(json!({
"wallet_name": wallet_name,
"wallet_key": DEFAULT_WALLET_KEY,
"wallet_key_derivation": WALLET_KDF_RAW
}))
.unwrap();

super::wallet_migrate(&new_config).await.unwrap();
}

#[tokio::test]
async fn test_wallet_create() {
let _setup = SetupEmpty::init();
Expand Down
2 changes: 2 additions & 0 deletions libvcx_core/src/errors/error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -143,6 +143,8 @@ pub enum LibvcxErrorKind {
DuplicationWallet,
#[error("Wallet record not found")]
WalletRecordNotFound,
#[error("Wallet migration failed")]
WalletMigrationFailed,
#[error("Record already exists in the wallet")]
DuplicationWalletRecord,
#[error("Wallet not found")]
Expand Down
6 changes: 6 additions & 0 deletions libvdrtools/indy-api-types/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,12 @@ pub type IndyHandle = i32;
pub struct WalletHandle(pub i32);
pub const INVALID_WALLET_HANDLE: WalletHandle = WalletHandle(0);

impl From<i32> for WalletHandle {
fn from(value: i32) -> Self {
Self(value)
}
}

pub type CallbackHandle = i32;

pub type CommandHandle = i32;
Expand Down
16 changes: 16 additions & 0 deletions wrappers/node/src/api/wallet.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,22 @@ export async function createWallet(config: object): Promise<void> {
}
}

export async function migrateWallet(config: object): Promise<void> {
try {
return await ffi.walletMigrate(JSON.stringify(config));
} catch (err: any) {
throw new VCXInternalError(err);
}
}

export async function deleteWallet(config: object): Promise<void> {
try {
return await ffi.walletDelete(JSON.stringify(config));
} catch (err: any) {
throw new VCXInternalError(err);
}
}

export async function configureIssuerWallet(seed: string): Promise<string> {
try {
return await ffi.configureIssuerWallet(seed);
Expand Down
1 change: 1 addition & 0 deletions wrappers/vcx-napi-rs/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ anoncreds_vdrtools = ["libvcx_core/anoncreds_vdrtools"]

[dependencies]
libvcx_core = { path = "../../libvcx_core" }
wallet_migrator = { path = "../../wallet_migrator" }
log = "0.4.16"
napi = { version = "2.10.14", default-features = false, features = [ "async" ] }
napi-derive = { version = "2.10.1" }
Expand Down
2 changes: 2 additions & 0 deletions wrappers/vcx-napi-rs/index.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -195,6 +195,8 @@ export function unpack(data: Buffer): Promise<string>
export function createAndStoreDid(seed?: string | undefined | null): Promise<string>
export function walletImport(config: string): Promise<void>
export function walletExport(path: string, backupKey: string): Promise<void>
export function walletMigrate(walletConfig: string): Promise<void>
export function walletDelete(walletConfig: string): Promise<void>
export function getVerkeyFromWallet(did: string): Promise<string>
export function rotateVerkey(did: string): Promise<void>
export function rotateVerkeyStart(did: string): Promise<string>
Expand Down
4 changes: 3 additions & 1 deletion wrappers/vcx-napi-rs/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -252,7 +252,7 @@ if (!nativeBinding) {
throw new Error(`Failed to load native binding`)
}

const { updateWebhookUrl, createAgencyClientForMainWallet, provisionCloudAgent, messagesUpdateStatus, generatePublicInvitation, connectionCreateInviter, connectionCreateInvitee, connectionGetThreadId, connectionGetPairwiseInfo, connectionGetRemoteDid, connectionGetRemoteVk, connectionGetState, connectionGetInvitation, connectionProcessInvite, connectionProcessRequest, connectionProcessResponse, connectionProcessAck, connectionProcessProblemReport, connectionSendResponse, connectionSendRequest, connectionSendAck, connectionSendGenericMessage, connectionSendAriesMessage, connectionCreateInvite, connectionSerialize, connectionDeserialize, connectionRelease, credentialCreateWithOffer, credentialRelease, credentialSendRequest, credentialDeclineOffer, credentialSerialize, credentialDeserialize, v2CredentialUpdateStateWithMessage, v2CredentialUpdateState, credentialGetState, credentialGetOffers, credentialGetAttributes, credentialGetAttachment, credentialGetTailsLocation, credentialGetTailsHash, credentialGetRevRegId, credentialGetThreadId, credentialdefCreateV2, credentialdefPublish, credentialdefDeserialize, credentialdefRelease, credentialdefSerialize, credentialdefGetCredDefId, credentialdefUpdateState, credentialdefGetState, disclosedProofCreateWithRequest, disclosedProofRelease, disclosedProofSendProof, disclosedProofRejectProof, disclosedProofGetProofMsg, disclosedProofSerialize, disclosedProofDeserialize, v2DisclosedProofUpdateState, v2DisclosedProofUpdateStateWithMessage, disclosedProofGetState, disclosedProofGetRequests, disclosedProofRetrieveCredentials, disclosedProofGetProofRequestAttachment, disclosedProofGenerateProof, disclosedProofDeclinePresentationRequest, disclosedProofGetThreadId, issuerCredentialDeserialize, issuerCredentialSerialize, issuerCredentialUpdateStateV2, issuerCredentialUpdateStateWithMessageV2, issuerCredentialUpdateStateWithMessageNonmediated, issuerCredentialGetState, issuerCredentialGetRevRegId, issuerCredentialCreate, issuerCredentialRevokeLocal, issuerCredentialIsRevokable, issuerCredentialGetRevocationId, issuerCredentialSendCredential, issuerCredentialSendCredentialNonmediated, issuerCredentialSendOfferV2, issuerCredentialSendOfferNonmediated, issuerCredentialBuildOfferMsgV2, issuerCredentialGetOfferMsg, issuerCredentialRelease, issuerCredentialGetThreadId, getLedgerAuthorAgreement, setActiveTxnAuthorAgreementMeta, createService, createServiceV2, getServiceFromLedger, getAttrFromLedger, clearAttrFromLedger, writeEndorserDid, getVerkeyFromLedger, getLedgerTxn, initDefaultLogger, mediatedConnectionGeneratePublicInvite, mediatedConnectionGetPwDid, mediatedConnectionGetTheirPwDid, mediatedConnectionGetThreadId, mediatedConnectionGetState, mediatedConnectionGetSourceId, mediatedConnectionCreate, mediatedConnectionCreateWithInvite, mediatedConnectionSendMessage, mediatedConnectionCreateWithConnectionRequestV2, mediatedConnectionSendHandshakeReuse, mediatedConnectionUpdateStateWithMessage, mediatedConnectionHandleMessage, mediatedConnectionUpdateState, mediatedConnectionDeleteConnection, mediatedConnectionConnect, mediatedConnectionSerialize, mediatedConnectionDeserialize, mediatedConnectionRelease, mediatedConnectionInviteDetails, mediatedConnectionSendPing, mediatedConnectionSendDiscoveryFeatures, mediatedConnectionInfo, mediatedConnectionMessagesDownload, mediatedConnectionSignData, mediatedConnectionVerifySignature, outOfBandBuildHandshakeReuseAcceptedMsg, outOfBandReceiverCreate, outOfBandReceiverExtractMessage, outOfBandReceiverConnectionExists, outOfBandReceiverNonmediatedConnectionExists, outOfBandReceiverBuildConnection, outOfBandReceiverGetThreadId, outOfBandReceiverSerialize, outOfBandReceiverDeserialize, outOfBandReceiverRelease, outOfBandSenderCreate, outOfBandSenderAppendMessage, outOfBandSenderAppendService, outOfBandSenderAppendServiceDid, outOfBandSenderToMessage, outOfBandSenderGetThreadId, outOfBandSenderSerialize, outOfBandSenderDeserialize, outOfBandSenderRelease, openMainPool, closeMainPool, proofCreate, proofGetPresentationMsg, proofGetPresentationRequestAttachment, proofGetPresentationAttachment, proofRelease, proofSendRequest, proofSendRequestNonmediated, proofGetRequestMsg, proofSerialize, proofDeserialize, v2ProofUpdateState, v2ProofUpdateStateWithMessage, proofUpdateStateWithMessageNonmediated, proofGetState, proofGetVerificationStatus, proofGetThreadId, markPresentationRequestMsgSent, revocationRegistryCreate, revocationRegistryPublish, revocationRegistryPublishRevocations, revocationRegistryGetRevRegId, revocationRegistryGetTailsHash, revocationRegistrySerialize, revocationRegistryDeserialize, revocationRegistryRelease, schemaGetAttributes, schemaPrepareForEndorser, schemaCreate, schemaGetSchemaId, schemaDeserialize, schemaSerialize, schemaRelease, schemaUpdateState, schemaGetState, enableMocks, trustpingBuildResponseMsg, trustpingBuildPing, shutdown, getVersion, walletOpenAsMain, walletCreateMain, walletCloseMain, vcxInitIssuerConfig, configureIssuerWallet, unpack, createAndStoreDid, walletImport, walletExport, getVerkeyFromWallet, rotateVerkey, rotateVerkeyStart, rotateVerkeyApply } = nativeBinding
const { updateWebhookUrl, createAgencyClientForMainWallet, provisionCloudAgent, messagesUpdateStatus, generatePublicInvitation, connectionCreateInviter, connectionCreateInvitee, connectionGetThreadId, connectionGetPairwiseInfo, connectionGetRemoteDid, connectionGetRemoteVk, connectionGetState, connectionGetInvitation, connectionProcessInvite, connectionProcessRequest, connectionProcessResponse, connectionProcessAck, connectionProcessProblemReport, connectionSendResponse, connectionSendRequest, connectionSendAck, connectionSendGenericMessage, connectionSendAriesMessage, connectionCreateInvite, connectionSerialize, connectionDeserialize, connectionRelease, credentialCreateWithOffer, credentialRelease, credentialSendRequest, credentialDeclineOffer, credentialSerialize, credentialDeserialize, v2CredentialUpdateStateWithMessage, v2CredentialUpdateState, credentialGetState, credentialGetOffers, credentialGetAttributes, credentialGetAttachment, credentialGetTailsLocation, credentialGetTailsHash, credentialGetRevRegId, credentialGetThreadId, credentialdefCreateV2, credentialdefPublish, credentialdefDeserialize, credentialdefRelease, credentialdefSerialize, credentialdefGetCredDefId, credentialdefUpdateState, credentialdefGetState, disclosedProofCreateWithRequest, disclosedProofRelease, disclosedProofSendProof, disclosedProofRejectProof, disclosedProofGetProofMsg, disclosedProofSerialize, disclosedProofDeserialize, v2DisclosedProofUpdateState, v2DisclosedProofUpdateStateWithMessage, disclosedProofGetState, disclosedProofGetRequests, disclosedProofRetrieveCredentials, disclosedProofGetProofRequestAttachment, disclosedProofGenerateProof, disclosedProofDeclinePresentationRequest, disclosedProofGetThreadId, issuerCredentialDeserialize, issuerCredentialSerialize, issuerCredentialUpdateStateV2, issuerCredentialUpdateStateWithMessageV2, issuerCredentialUpdateStateWithMessageNonmediated, issuerCredentialGetState, issuerCredentialGetRevRegId, issuerCredentialCreate, issuerCredentialRevokeLocal, issuerCredentialIsRevokable, issuerCredentialGetRevocationId, issuerCredentialSendCredential, issuerCredentialSendCredentialNonmediated, issuerCredentialSendOfferV2, issuerCredentialSendOfferNonmediated, issuerCredentialBuildOfferMsgV2, issuerCredentialGetOfferMsg, issuerCredentialRelease, issuerCredentialGetThreadId, getLedgerAuthorAgreement, setActiveTxnAuthorAgreementMeta, createService, createServiceV2, getServiceFromLedger, getAttrFromLedger, clearAttrFromLedger, writeEndorserDid, getVerkeyFromLedger, getLedgerTxn, initDefaultLogger, mediatedConnectionGeneratePublicInvite, mediatedConnectionGetPwDid, mediatedConnectionGetTheirPwDid, mediatedConnectionGetThreadId, mediatedConnectionGetState, mediatedConnectionGetSourceId, mediatedConnectionCreate, mediatedConnectionCreateWithInvite, mediatedConnectionSendMessage, mediatedConnectionCreateWithConnectionRequestV2, mediatedConnectionSendHandshakeReuse, mediatedConnectionUpdateStateWithMessage, mediatedConnectionHandleMessage, mediatedConnectionUpdateState, mediatedConnectionDeleteConnection, mediatedConnectionConnect, mediatedConnectionSerialize, mediatedConnectionDeserialize, mediatedConnectionRelease, mediatedConnectionInviteDetails, mediatedConnectionSendPing, mediatedConnectionSendDiscoveryFeatures, mediatedConnectionInfo, mediatedConnectionMessagesDownload, mediatedConnectionSignData, mediatedConnectionVerifySignature, outOfBandBuildHandshakeReuseAcceptedMsg, outOfBandReceiverCreate, outOfBandReceiverExtractMessage, outOfBandReceiverConnectionExists, outOfBandReceiverNonmediatedConnectionExists, outOfBandReceiverBuildConnection, outOfBandReceiverGetThreadId, outOfBandReceiverSerialize, outOfBandReceiverDeserialize, outOfBandReceiverRelease, outOfBandSenderCreate, outOfBandSenderAppendMessage, outOfBandSenderAppendService, outOfBandSenderAppendServiceDid, outOfBandSenderToMessage, outOfBandSenderGetThreadId, outOfBandSenderSerialize, outOfBandSenderDeserialize, outOfBandSenderRelease, openMainPool, closeMainPool, proofCreate, proofGetPresentationMsg, proofGetPresentationRequestAttachment, proofGetPresentationAttachment, proofRelease, proofSendRequest, proofSendRequestNonmediated, proofGetRequestMsg, proofSerialize, proofDeserialize, v2ProofUpdateState, v2ProofUpdateStateWithMessage, proofUpdateStateWithMessageNonmediated, proofGetState, proofGetVerificationStatus, proofGetThreadId, markPresentationRequestMsgSent, revocationRegistryCreate, revocationRegistryPublish, revocationRegistryPublishRevocations, revocationRegistryGetRevRegId, revocationRegistryGetTailsHash, revocationRegistrySerialize, revocationRegistryDeserialize, revocationRegistryRelease, schemaGetAttributes, schemaPrepareForEndorser, schemaCreate, schemaGetSchemaId, schemaDeserialize, schemaSerialize, schemaRelease, schemaUpdateState, schemaGetState, enableMocks, trustpingBuildResponseMsg, trustpingBuildPing, shutdown, getVersion, walletOpenAsMain, walletCreateMain, walletCloseMain, vcxInitIssuerConfig, configureIssuerWallet, unpack, createAndStoreDid, walletImport, walletExport, walletMigrate, walletDelete, getVerkeyFromWallet, rotateVerkey, rotateVerkeyStart, rotateVerkeyApply } = nativeBinding

module.exports.updateWebhookUrl = updateWebhookUrl
module.exports.createAgencyClientForMainWallet = createAgencyClientForMainWallet
Expand Down Expand Up @@ -446,6 +446,8 @@ module.exports.unpack = unpack
module.exports.createAndStoreDid = createAndStoreDid
module.exports.walletImport = walletImport
module.exports.walletExport = walletExport
module.exports.walletMigrate = walletMigrate
module.exports.walletDelete = walletDelete
module.exports.getVerkeyFromWallet = getVerkeyFromWallet
module.exports.rotateVerkey = rotateVerkey
module.exports.rotateVerkeyStart = rotateVerkeyStart
Expand Down
36 changes: 35 additions & 1 deletion wrappers/vcx-napi-rs/src/api/wallet.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
use libvcx_core::{
api_vcx::api_global::{ledger, settings::settings_init_issuer_config, wallet},
aries_vcx::aries_vcx_core::wallet::indy::{IssuerConfig, RestoreWalletConfigs, WalletConfig},
aries_vcx::aries_vcx_core::wallet::indy::{
wallet::delete_wallet, IssuerConfig, RestoreWalletConfigs, WalletConfig,
},
errors::error::{LibvcxError, LibvcxErrorKind},
serde_json,
serde_json::json,
Expand Down Expand Up @@ -103,6 +105,38 @@ pub async fn wallet_export(path: String, backup_key: String) -> napi::Result<()>
.map_err(to_napi_err)
}

#[napi]
pub async fn wallet_migrate(wallet_config: String) -> napi::Result<()> {
let wallet_config = serde_json::from_str(&wallet_config)
.map_err(|err| {
LibvcxError::from_msg(
LibvcxErrorKind::InvalidConfiguration,
format!("Serialization error: {:?}", err),
)
})
.map_err(to_napi_err)?;

wallet::wallet_migrate(&wallet_config)
.await
.map_err(|e| napi::Error::from_reason(e.to_string()))
}

#[napi]
pub async fn wallet_delete(wallet_config: String) -> napi::Result<()> {
let wallet_config = serde_json::from_str(&wallet_config)
.map_err(|err| {
LibvcxError::from_msg(
LibvcxErrorKind::InvalidConfiguration,
format!("Serialization error: {:?}", err),
)
})
.map_err(to_napi_err)?;

delete_wallet(&wallet_config)
.await
.map_err(|e| napi::Error::from_reason(e.to_string()))
}

#[napi]
pub async fn get_verkey_from_wallet(did: String) -> napi::Result<String> {
wallet::key_for_local_did(&did).await.map_err(to_napi_err)
Expand Down

0 comments on commit ac35fdd

Please sign in to comment.