melib 0.8.13

library for e-mail clients and other e-mail applications
Documentation
/*
 * meli - jmap module.
 *
 * Copyright 2017 - 2019 Manos Pitsidianakis
 *
 * This file is part of meli.
 *
 * meli is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * meli is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with meli. If not, see <http://www.gnu.org/licenses/>.
 */

use std::sync::Arc;

use futures::lock::Mutex as FutureMutex;
use isahc::AsyncReadResponseExt;

use crate::{
    error::Result,
    jmap::{connection::JmapConnection, methods::download_request_format, Store},
    EnvelopeHash,
};

/// Fetch an envelope by hash as bytes.
#[derive(Clone, Debug)]
pub struct JmapOp {
    hash: EnvelopeHash,
    connection: Arc<FutureMutex<JmapConnection>>,
    store: Arc<Store>,
}

impl JmapOp {
    pub fn new(
        hash: EnvelopeHash,
        connection: Arc<FutureMutex<JmapConnection>>,
        store: Arc<Store>,
    ) -> Self {
        Self {
            hash,
            connection,
            store,
        }
    }

    pub async fn as_bytes(&self) -> Result<Vec<u8>> {
        {
            let byte_lck = self.store.byte_cache.lock().await;
            if let Some(Some(ret)) = byte_lck.get(&self.hash).map(|c| c.bytes.clone()) {
                return Ok(ret.into_bytes());
            }
        }
        let blob_id = self.store.blob_id_store.lock().await[&self.hash].clone();
        let mut conn = self.connection.lock().await;
        conn.connect().await?;
        let (download_url, mail_account_id) = {
            let g = self.store.online_status.session_guard().await?;
            (g.download_url.clone(), g.mail_account_id())
        };
        let res_text = conn
            .get_async(&download_request_format(
                &download_url,
                &mail_account_id,
                &blob_id,
                None,
            )?)
            .await?
            .text()
            .await?;

        self.store
            .byte_cache
            .lock()
            .await
            .entry(self.hash)
            .or_default()
            .bytes = Some(res_text.clone());
        Ok(res_text.into_bytes())
    }
}