1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
use crate::steam::SteamResult;
use crate::string_ext::FromUtf8NulTruncating;
use crate::{AppId, Client, SteamId};
use futures::Future;
use snafu::{ensure, ResultExt};
use std::ffi::CString;
use steamworks_sys as sys;

#[derive(Debug, Copy, Clone, Hash, Eq, PartialEq, Ord, PartialOrd)]
pub struct UgcHandle(sys::UGCHandle_t);

impl UgcHandle {
    pub fn download_to_location(
        self,
        client: Client,
        location: impl Into<Vec<u8>>,
        priority: u32,
    ) -> impl Future<Output = Result<DownloadUGCResult, UgcDownloadToLocationError>> + Send {
        let location = CString::new(location.into());
        async move {
            let location = location.context(NulSnafu)?;

            let response: sys::RemoteStorageDownloadUGCResult_t = unsafe {
                let handle = sys::SteamAPI_ISteamRemoteStorage_UGCDownloadToLocation(
                    *client.0.remote_storage,
                    self.0,
                    location.as_ptr(),
                    priority,
                );

                client.register_for_call_result(handle).await
            };

            {
                let result = SteamResult::from_inner(response.m_eResult);

                ensure!(
                    result == SteamResult::OK,
                    UGCDownloadToLocationSnafu {
                        steam_result: result,
                    }
                );
            }

            Ok(DownloadUGCResult {
                app_id: response.m_nAppID.into(),
                size_in_bytes: response.m_nSizeInBytes,
                filename: String::from_utf8_nul_truncating(&response.m_pchFileName[..]).expect(
                    "Filename returned in RemoteStorageDownloadUGCResult_t was not valid UTF-8",
                ),
                steam_id_owner: SteamId::new(response.m_ulSteamIDOwner),
            })
        }
    }

    pub(crate) fn from_inner(handle: sys::UGCHandle_t) -> Option<Self> {
        if handle == sys::k_UGCHandleInvalid {
            None
        } else {
            Some(UgcHandle(handle))
        }
    }
}

#[derive(Debug, Clone, Hash, Eq, PartialEq, Ord, PartialOrd)]
pub struct DownloadUGCResult {
    app_id: AppId,
    size_in_bytes: i32,
    filename: String,
    steam_id_owner: SteamId,
}

#[derive(Debug, snafu::Snafu)]
pub enum UgcDownloadToLocationError {
    /// The location provided contains nul byte(s)
    #[snafu(display("The location provided contained nul byte(s): {}", source))]
    Nul { source: std::ffi::NulError },

    /// `UGCDownloadToLocation()` failed
    #[snafu(display("UGCDownloadToLocation() failed: {}", steam_result))]
    UGCDownloadToLocation { steam_result: SteamResult },
}