Skip to content

Commit dc5ecfe

Browse files
Expose play purchase API via gRPC
1 parent c651385 commit dc5ecfe

File tree

5 files changed

+107
-9
lines changed

5 files changed

+107
-9
lines changed

mullvad-daemon/src/management_interface.rs

+54
Original file line numberDiff line numberDiff line change
@@ -949,6 +949,60 @@ impl ManagementService for ManagementServiceImpl {
949949
let blob = self.wait_for_result(rx).await??;
950950
Ok(Response::new(blob))
951951
}
952+
953+
#[cfg(target_os = "android")]
954+
async fn init_play_purchase(
955+
&self,
956+
_request: Request<()>,
957+
) -> ServiceResult<types::PlayPurchasePaymentToken> {
958+
log::debug!("init_play_purchase");
959+
960+
let (tx, rx) = oneshot::channel();
961+
self.send_command_to_daemon(DaemonCommand::InitPlayPurchase(tx))?;
962+
963+
let payment_token = self
964+
.wait_for_result(rx)
965+
.await?
966+
.map(types::PlayPurchasePaymentToken::from)
967+
.map_err(map_daemon_error)?;
968+
969+
Ok(Response::new(payment_token))
970+
}
971+
972+
/// On non-Android platforms, the return value will be useless.
973+
#[cfg(not(target_os = "android"))]
974+
async fn init_play_purchase(
975+
&self,
976+
_: Request<()>,
977+
) -> ServiceResult<types::PlayPurchasePaymentToken> {
978+
log::error!("Called `init_play_purchase` on non-Android platform");
979+
Ok(Response::new(types::PlayPurchasePaymentToken::from(
980+
String::default(),
981+
)))
982+
}
983+
984+
#[cfg(target_os = "android")]
985+
async fn verify_play_purchase(
986+
&self,
987+
request: Request<types::PlayPurchase>,
988+
) -> ServiceResult<()> {
989+
log::debug!("verify_play_purchase");
990+
991+
let (tx, rx) = oneshot::channel();
992+
let play_purchase = mullvad_types::account::PlayPurchase::try_from(request.into_inner())?;
993+
994+
self.send_command_to_daemon(DaemonCommand::VerifyPlayPurchase(tx, play_purchase))?;
995+
996+
self.wait_for_result(rx).await?.map_err(map_daemon_error)?;
997+
998+
Ok(Response::new(()))
999+
}
1000+
1001+
#[cfg(not(target_os = "android"))]
1002+
async fn verify_play_purchase(&self, _: Request<types::PlayPurchase>) -> ServiceResult<()> {
1003+
log::error!("Called `verify_play_purchase` on non-Android platform");
1004+
Ok(Response::new(()))
1005+
}
9521006
}
9531007

9541008
impl ManagementServiceImpl {

mullvad-jni/src/daemon_interface.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
use futures::{channel::oneshot, executor::block_on};
22
use mullvad_daemon::{device, DaemonCommand, DaemonCommandSender};
33
use mullvad_types::{
4-
account::{AccountData, AccountToken, PlayPurchase, VoucherSubmission},
4+
account::{AccountData, AccountToken, VoucherSubmission},
55
custom_list::CustomList,
66
device::{Device, DeviceState},
77
relay_constraints::{ObfuscationSettings, RelaySettings},

mullvad-jni/src/lib.rs

+12-8
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ mod is_null;
66
mod problem_report;
77
mod talpid_vpn_service;
88

9-
use crate::{daemon_interface::DaemonInterface};
9+
use crate::daemon_interface::DaemonInterface;
1010
use jnix::{
1111
jni::{
1212
objects::{GlobalRef, JObject, JString, JValue},
@@ -18,11 +18,11 @@ use jnix::{
1818
};
1919
use mullvad_api::{rest::Error as RestError, StatusCode};
2020
use mullvad_daemon::{
21-
cleanup_old_rpc_socket, device, exception_logging, logging, runtime::new_multi_thread, version, Daemon,
22-
DaemonCommandChannel,
21+
cleanup_old_rpc_socket, device, exception_logging, logging, runtime::new_multi_thread, version,
22+
Daemon, DaemonCommandChannel,
2323
};
2424
use mullvad_types::{
25-
account::{AccountData, PlayPurchase, VoucherSubmission},
25+
account::{AccountData, VoucherSubmission},
2626
custom_list::CustomList,
2727
settings::DnsOptions,
2828
};
@@ -276,7 +276,7 @@ fn spawn_daemon(
276276
command_channel: DaemonCommandChannel,
277277
android_context: AndroidContext,
278278
) -> Result<(), Error> {
279-
// let listener = JniEventListener::spawn(env, this).map_err(Error::SpawnJniEventListener)?;
279+
// let listener = JniEventListener::spawn(env, this).map_err(Error::SpawnJniEventListener)?;
280280
let daemon_object = env
281281
.new_global_ref(*this)
282282
.map_err(Error::CreateGlobalReference)?;
@@ -307,7 +307,9 @@ fn spawn_daemon(
307307

308308
runtime.block_on(cleanup_old_rpc_socket());
309309

310-
let event_listener = match runtime.block_on(async {spawn_management_interface(command_channel.sender()) }) {
310+
let event_listener = match runtime
311+
.block_on(async { spawn_management_interface(command_channel.sender()) })
312+
{
311313
Ok(event_listener) => event_listener,
312314
Err(error) => {
313315
let _ = tx.send(Err(error));
@@ -356,7 +358,10 @@ fn spawn_management_interface(
356358
) -> Result<ManagementInterfaceEventBroadcaster, Error> {
357359
let (socket_path, event_broadcaster) = ManagementInterfaceServer::start(command_sender)
358360
.map_err(|error| {
359-
log::error!("{}", error.display_chain_with_msg("Unable to start management interface server"));
361+
log::error!(
362+
"{}",
363+
error.display_chain_with_msg("Unable to start management interface server")
364+
);
360365
Error::SpawnManagementInterface(error)
361366
})?;
362367

@@ -493,7 +498,6 @@ pub extern "system" fn Java_net_mullvad_mullvadvpn_service_MullvadDaemon_shutdow
493498
}
494499
}
495500

496-
497501
fn log_request_error(request: &str, error: &daemon_interface::Error) {
498502
match error {
499503
daemon_interface::Error::Api(RestError::Aborted) => {

mullvad-management-interface/proto/management_interface.proto

+11
Original file line numberDiff line numberDiff line change
@@ -99,6 +99,10 @@ service ManagementService {
9999
rpc SetSplitTunnelState(google.protobuf.BoolValue) returns (google.protobuf.Empty) {}
100100
rpc GetExcludedProcesses(google.protobuf.Empty) returns (ExcludedProcessList) {}
101101

102+
// Play payment (Android)
103+
rpc InitPlayPurchase(google.protobuf.Empty) returns (PlayPurchasePaymentToken) {}
104+
rpc VerifyPlayPurchase(PlayPurchase) returns (google.protobuf.Empty) {}
105+
102106
// Notify the split tunnel monitor that a volume was mounted or dismounted
103107
// (Windows).
104108
rpc CheckVolumes(google.protobuf.Empty) returns (google.protobuf.Empty) {}
@@ -701,3 +705,10 @@ message RemoveDeviceEvent {
701705
string account_token = 1;
702706
repeated Device new_device_list = 2;
703707
}
708+
709+
message PlayPurchase {
710+
string product_id = 1;
711+
PlayPurchasePaymentToken purchase_token = 2;
712+
}
713+
714+
message PlayPurchasePaymentToken { string token = 1; }

mullvad-management-interface/src/types/conversions/account.rs

+29
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
use crate::types;
22
use chrono::TimeZone;
33
use mullvad_types::account::{AccountData, VoucherSubmission};
4+
#[cfg(target_os = "android")]
5+
use mullvad_types::account::{PlayPurchase, PlayPurchasePaymentToken};
46

57
use super::FromProtobufTypeError;
68

@@ -62,3 +64,30 @@ impl TryFrom<types::AccountData> for AccountData {
6264
})
6365
}
6466
}
67+
68+
#[cfg(target_os = "android")]
69+
impl TryFrom<types::PlayPurchase> for PlayPurchase {
70+
type Error = FromProtobufTypeError;
71+
72+
fn try_from(value: types::PlayPurchase) -> Result<Self, Self::Error> {
73+
let product_id = value.product_id;
74+
let purchase_token = value
75+
.purchase_token
76+
.ok_or(FromProtobufTypeError::InvalidArgument(
77+
"Missing purchase token",
78+
))?
79+
.token;
80+
81+
Ok(Self {
82+
product_id,
83+
purchase_token,
84+
})
85+
}
86+
}
87+
88+
#[cfg(target_os = "android")]
89+
impl From<PlayPurchasePaymentToken> for types::PlayPurchasePaymentToken {
90+
fn from(token: PlayPurchasePaymentToken) -> Self {
91+
Self { token }
92+
}
93+
}

0 commit comments

Comments
 (0)