2023-02-14 17:38:45 +01:00
|
|
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
|
|
|
use deno_core::error::type_error;
|
|
|
|
use deno_core::error::AnyError;
|
|
|
|
use deno_core::op;
|
|
|
|
use deno_core::OpState;
|
|
|
|
use deno_core::ResourceId;
|
2023-02-20 22:22:28 +05:30
|
|
|
use deno_core::StringOrBuffer;
|
2023-02-14 17:38:45 +01:00
|
|
|
use deno_core::ZeroCopyBuf;
|
|
|
|
use std::rc::Rc;
|
|
|
|
|
2023-02-20 22:22:28 +05:30
|
|
|
use rsa::padding::PaddingScheme;
|
|
|
|
use rsa::pkcs8::DecodePrivateKey;
|
|
|
|
use rsa::pkcs8::DecodePublicKey;
|
|
|
|
use rsa::PublicKey;
|
|
|
|
use rsa::RsaPrivateKey;
|
|
|
|
use rsa::RsaPublicKey;
|
|
|
|
|
2023-02-14 17:38:45 +01:00
|
|
|
mod digest;
|
|
|
|
|
|
|
|
#[op]
|
|
|
|
pub fn op_node_create_hash(
|
|
|
|
state: &mut OpState,
|
|
|
|
algorithm: String,
|
|
|
|
) -> Result<ResourceId, AnyError> {
|
|
|
|
Ok(state.resource_table.add(digest::Context::new(&algorithm)?))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
|
|
|
pub fn op_node_hash_update(
|
|
|
|
state: &mut OpState,
|
|
|
|
rid: ResourceId,
|
|
|
|
data: &[u8],
|
|
|
|
) -> Result<(), AnyError> {
|
|
|
|
let context = state.resource_table.get::<digest::Context>(rid)?;
|
|
|
|
context.update(data);
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
|
|
|
pub fn op_node_hash_digest(
|
|
|
|
state: &mut OpState,
|
|
|
|
rid: ResourceId,
|
|
|
|
) -> Result<ZeroCopyBuf, AnyError> {
|
|
|
|
let context = state.resource_table.take::<digest::Context>(rid)?;
|
|
|
|
let context = Rc::try_unwrap(context)
|
|
|
|
.map_err(|_| type_error("Hash context is already in use"))?;
|
|
|
|
Ok(context.digest()?.into())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
|
|
|
pub fn op_node_hash_clone(
|
|
|
|
state: &mut OpState,
|
|
|
|
rid: ResourceId,
|
|
|
|
) -> Result<ResourceId, AnyError> {
|
|
|
|
let context = state.resource_table.get::<digest::Context>(rid)?;
|
|
|
|
Ok(state.resource_table.add(context.as_ref().clone()))
|
|
|
|
}
|
2023-02-20 22:22:28 +05:30
|
|
|
|
|
|
|
#[op]
|
|
|
|
pub fn op_node_private_encrypt(
|
|
|
|
key: StringOrBuffer,
|
|
|
|
msg: StringOrBuffer,
|
|
|
|
padding: u32,
|
|
|
|
) -> Result<ZeroCopyBuf, AnyError> {
|
|
|
|
let key = RsaPrivateKey::from_pkcs8_pem((&key).try_into()?)?;
|
|
|
|
|
|
|
|
let mut rng = rand::thread_rng();
|
|
|
|
match padding {
|
|
|
|
1 => Ok(
|
|
|
|
key
|
|
|
|
.encrypt(&mut rng, PaddingScheme::new_pkcs1v15_encrypt(), &msg)?
|
|
|
|
.into(),
|
|
|
|
),
|
|
|
|
4 => Ok(
|
|
|
|
key
|
|
|
|
.encrypt(&mut rng, PaddingScheme::new_oaep::<sha1::Sha1>(), &msg)?
|
|
|
|
.into(),
|
|
|
|
),
|
|
|
|
_ => Err(type_error("Unknown padding")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
|
|
|
pub fn op_node_private_decrypt(
|
|
|
|
key: StringOrBuffer,
|
|
|
|
msg: StringOrBuffer,
|
|
|
|
padding: u32,
|
|
|
|
) -> Result<ZeroCopyBuf, AnyError> {
|
|
|
|
let key = RsaPrivateKey::from_pkcs8_pem((&key).try_into()?)?;
|
|
|
|
|
|
|
|
match padding {
|
|
|
|
1 => Ok(
|
|
|
|
key
|
|
|
|
.decrypt(PaddingScheme::new_pkcs1v15_encrypt(), &msg)?
|
|
|
|
.into(),
|
|
|
|
),
|
|
|
|
4 => Ok(
|
|
|
|
key
|
|
|
|
.decrypt(PaddingScheme::new_oaep::<sha1::Sha1>(), &msg)?
|
|
|
|
.into(),
|
|
|
|
),
|
|
|
|
_ => Err(type_error("Unknown padding")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
|
|
|
pub fn op_node_public_encrypt(
|
|
|
|
key: StringOrBuffer,
|
|
|
|
msg: StringOrBuffer,
|
|
|
|
padding: u32,
|
|
|
|
) -> Result<ZeroCopyBuf, AnyError> {
|
|
|
|
let key = RsaPublicKey::from_public_key_pem((&key).try_into()?)?;
|
|
|
|
|
|
|
|
let mut rng = rand::thread_rng();
|
|
|
|
match padding {
|
|
|
|
1 => Ok(
|
|
|
|
key
|
|
|
|
.encrypt(&mut rng, PaddingScheme::new_pkcs1v15_encrypt(), &msg)?
|
|
|
|
.into(),
|
|
|
|
),
|
|
|
|
4 => Ok(
|
|
|
|
key
|
|
|
|
.encrypt(&mut rng, PaddingScheme::new_oaep::<sha1::Sha1>(), &msg)?
|
|
|
|
.into(),
|
|
|
|
),
|
|
|
|
_ => Err(type_error("Unknown padding")),
|
|
|
|
}
|
|
|
|
}
|