From 9191b1c68b8321f6e2b507d813574ef01c290856 Mon Sep 17 00:00:00 2001 From: hoslo Date: Wed, 24 Jan 2024 22:56:35 +0800 Subject: [PATCH] feat(services/pcloud): setup test for pcloud --- .github/services/pcloud/pcloud/action.yml | 32 ++++++++ bindings/java/Cargo.toml | 2 + bindings/nodejs/Cargo.toml | 2 + bindings/python/Cargo.toml | 2 + core/Cargo.toml | 1 + core/src/services/pcloud/backend.rs | 15 +++- core/src/services/pcloud/core.rs | 92 +++++++++++++++-------- core/src/services/pcloud/error.rs | 24 +++++- core/src/services/pcloud/lister.rs | 4 +- core/src/services/pcloud/writer.rs | 55 ++++++++++---- 10 files changed, 178 insertions(+), 51 deletions(-) create mode 100644 .github/services/pcloud/pcloud/action.yml diff --git a/.github/services/pcloud/pcloud/action.yml b/.github/services/pcloud/pcloud/action.yml new file mode 100644 index 000000000000..0d01853d1de3 --- /dev/null +++ b/.github/services/pcloud/pcloud/action.yml @@ -0,0 +1,32 @@ +# Licensed to the Apache Software Foundation (ASF) under one +# or more contributor license agreements. See the NOTICE file +# distributed with this work for additional information +# regarding copyright ownership. The ASF licenses this file +# to you under the Apache License, Version 2.0 (the +# "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, +# software distributed under the License is distributed on an +# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +# KIND, either express or implied. See the License for the +# specific language governing permissions and limitations +# under the License. + +name: pcloud +description: 'Behavior test for Pcloud.' + +runs: + using: "composite" + steps: + - name: Setup + uses: 1password/load-secrets-action@v1 + with: + export-env: true + env: + OPENDAL_PCLOUD_ROOT: / + OPENDAL_PCLOUD_ENDPOINT: op://services/pcloud/endpoint + OPENDAL_PCLOUD_USERNAME: op://services/pcloud/username + OPENDAL_PCLOUD_PASSWORD: op://services/pcloud/password diff --git a/bindings/java/Cargo.toml b/bindings/java/Cargo.toml index 9d199342c31b..f91b64e41253 100644 --- a/bindings/java/Cargo.toml +++ b/bindings/java/Cargo.toml @@ -71,6 +71,7 @@ services-all = [ "services-onedrive", "services-persy", "services-postgresql", + "services-pcloud", "services-koofr", "services-mysql", "services-redb", @@ -135,6 +136,7 @@ services-mysql = ["opendal/services-mysql"] services-onedrive = ["opendal/services-onedrive"] services-persy = ["opendal/services-persy"] services-postgresql = ["opendal/services-postgresql"] +services-pcloud = ["opendal/services-pcloud"] services-redb = ["opendal/services-redb"] services-redis = ["opendal/services-redis"] services-rocksdb = ["opendal/services-rocksdb"] diff --git a/bindings/nodejs/Cargo.toml b/bindings/nodejs/Cargo.toml index f9182e3a88e1..afa30bcd0f85 100644 --- a/bindings/nodejs/Cargo.toml +++ b/bindings/nodejs/Cargo.toml @@ -74,6 +74,7 @@ services-all = [ "services-onedrive", "services-persy", "services-postgresql", + "services-pcloud", "services-mysql", "services-redb", "services-redis", @@ -130,6 +131,7 @@ services-mysql = ["opendal/services-mysql"] services-onedrive = ["opendal/services-onedrive"] services-persy = ["opendal/services-persy"] services-postgresql = ["opendal/services-postgresql"] +services-pcloud = ["opendal/services-pcloud"] services-redb = ["opendal/services-redb"] services-redis = ["opendal/services-redis"] services-rocksdb = ["opendal/services-rocksdb"] diff --git a/bindings/python/Cargo.toml b/bindings/python/Cargo.toml index f2145c8ee539..c7ccc6565ef8 100644 --- a/bindings/python/Cargo.toml +++ b/bindings/python/Cargo.toml @@ -68,6 +68,7 @@ services-all = [ "services-onedrive", "services-persy", "services-postgresql", + "services-pcloud", "services-mysql", "services-redb", "services-redis", @@ -132,6 +133,7 @@ services-mysql = ["opendal/services-mysql"] services-onedrive = ["opendal/services-onedrive"] services-persy = ["opendal/services-persy"] services-postgresql = ["opendal/services-postgresql"] +services-pcloud = ["opendal/services-pcloud"] services-redb = ["opendal/services-redb"] services-redis = ["opendal/services-redis"] services-rocksdb = ["opendal/services-rocksdb"] diff --git a/core/Cargo.toml b/core/Cargo.toml index 880f4ff3da1c..b8c4c2ea2fef 100644 --- a/core/Cargo.toml +++ b/core/Cargo.toml @@ -52,6 +52,7 @@ default = [ "services-webdav", "services-webhdfs", "services-azfile", + "services-pcloud" ] # Build test utils or not. diff --git a/core/src/services/pcloud/backend.rs b/core/src/services/pcloud/backend.rs index 8283ae884610..2696704e26aa 100644 --- a/core/src/services/pcloud/backend.rs +++ b/core/src/services/pcloud/backend.rs @@ -27,6 +27,7 @@ use serde::Deserialize; use super::core::*; use super::error::parse_error; +use super::error::parse_result; use super::error::PcloudError; use super::lister::PcloudLister; use super::writer::PcloudWriter; @@ -277,6 +278,9 @@ impl Accessor for PcloudBackend { let resp: StatResponse = serde_json::from_slice(&bs).map_err(new_json_deserialize_error)?; let result = resp.result; + + parse_result(result)?; + if result == 2010 || result == 2055 || result == 2002 { return Err(Error::new(ErrorKind::NotFound, &format!("{resp:?}"))); } @@ -338,9 +342,10 @@ impl Accessor for PcloudBackend { let resp: PcloudError = serde_json::from_slice(&bs).map_err(new_json_deserialize_error)?; let result = resp.result; - - // pCloud returns 2005 or 2009 if the file or folder is not found - if result != 0 && result != 2005 && result != 2009 { + parse_result(result)?; + // pCloud returns 2005 if the folder is not found. + // And 2009 or 2002 if the file is not found. + if result != 0 && result != 2005 && result != 2009 && result != 2002 { return Err(Error::new(ErrorKind::Unexpected, &format!("{resp:?}"))); } @@ -372,6 +377,8 @@ impl Accessor for PcloudBackend { let resp: PcloudError = serde_json::from_slice(&bs).map_err(new_json_deserialize_error)?; let result = resp.result; + + parse_result(result)?; if result == 2009 || result == 2010 || result == 2055 || result == 2002 { return Err(Error::new(ErrorKind::NotFound, &format!("{resp:?}"))); } @@ -402,6 +409,8 @@ impl Accessor for PcloudBackend { let resp: PcloudError = serde_json::from_slice(&bs).map_err(new_json_deserialize_error)?; let result = resp.result; + + parse_result(result)?; if result == 2009 || result == 2010 || result == 2055 || result == 2002 { return Err(Error::new(ErrorKind::NotFound, &format!("{resp:?}"))); } diff --git a/core/src/services/pcloud/core.rs b/core/src/services/pcloud/core.rs index 21d3dfedd780..ee134d83d5fe 100644 --- a/core/src/services/pcloud/core.rs +++ b/core/src/services/pcloud/core.rs @@ -19,12 +19,14 @@ use std::fmt::Debug; use std::fmt::Formatter; use bytes::Bytes; +use http::header; use http::Request; use http::Response; use http::StatusCode; use serde::Deserialize; use super::error::parse_error; +use super::error::parse_result; use super::error::PcloudError; use crate::raw::*; use crate::*; @@ -68,8 +70,8 @@ impl PcloudCore { "{}/getfilelink?path=/{}&username={}&password={}", self.endpoint, percent_encode_path(&path), - self.username, - self.password + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); let req = Request::get(url); @@ -88,6 +90,9 @@ impl PcloudCore { let resp: GetFileLinkResponse = serde_json::from_slice(&bs).map_err(new_json_deserialize_error)?; let result = resp.result; + + parse_result(result)?; + if result == 2010 || result == 2055 || result == 2002 { return Err(Error::new(ErrorKind::NotFound, &format!("{resp:?}"))); } @@ -136,6 +141,9 @@ impl PcloudCore { let resp: PcloudError = serde_json::from_slice(&bs).map_err(new_json_deserialize_error)?; let result = resp.result; + + parse_result(result)?; + if result == 2010 || result == 2055 || result == 2002 { return Err(Error::new(ErrorKind::NotFound, &format!("{resp:?}"))); } @@ -161,8 +169,8 @@ impl PcloudCore { "{}/createfolderifnotexists?path=/{}&username={}&password={}", self.endpoint, percent_encode_path(path), - self.username, - self.password + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); let req = Request::post(url); @@ -184,8 +192,8 @@ impl PcloudCore { self.endpoint, percent_encode_path(&from), percent_encode_path(&to), - self.username, - self.password + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); let req = Request::post(url); @@ -206,8 +214,8 @@ impl PcloudCore { self.endpoint, percent_encode_path(&from), percent_encode_path(&to), - self.username, - self.password + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); let req = Request::post(url); @@ -227,8 +235,8 @@ impl PcloudCore { "{}/deletefolder?path=/{}&username={}&password={}", self.endpoint, percent_encode_path(&path), - self.username, - self.password + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); let req = Request::post(url); @@ -248,8 +256,8 @@ impl PcloudCore { "{}/deletefile?path=/{}&username={}&password={}", self.endpoint, percent_encode_path(&path), - self.username, - self.password + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); let req = Request::post(url); @@ -271,8 +279,8 @@ impl PcloudCore { self.endpoint, percent_encode_path(&from), percent_encode_path(&to), - self.username, - self.password + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); let req = Request::post(url); @@ -294,8 +302,8 @@ impl PcloudCore { self.endpoint, percent_encode_path(&from), percent_encode_path(&to), - self.username, - self.password + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); let req = Request::post(url); @@ -317,8 +325,8 @@ impl PcloudCore { "{}/stat?path=/{}&username={}&password={}", self.endpoint, percent_encode_path(path), - self.username, - self.password + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); let req = Request::post(url); @@ -337,20 +345,28 @@ impl PcloudCore { let (name, path) = (get_basename(&path), get_parent(&path).trim_end_matches('/')); let url = format!( - "{}/uploadfile?path=/{}&filename={}&username={}&password={}", + "{}/uploadfile?path=/{}&nopartial=1&mtime={}&username={}&password={}", self.endpoint, percent_encode_path(path), - percent_encode_path(name), - self.username, - self.password + chrono::Local::now().timestamp(), + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); - let req = Request::put(url); + let req = Request::post(url); - // set body - let req = req - .body(AsyncBody::Bytes(bs)) - .map_err(new_request_build_error)?; + let file_part = FormDataPart::new("file") + .header( + header::CONTENT_DISPOSITION, + format!("form-data; name=\"file\"; filename=\"{name}\"") + .parse() + .unwrap(), + ) + .content(bs); + + let multipart = Multipart::new().part(file_part); + + let req = multipart.apply(req)?; self.send(req).await } @@ -366,8 +382,8 @@ impl PcloudCore { "{}/listfolder?path={}&username={}&password={}", self.endpoint, percent_encode_path(path), - self.username, - self.password + percent_encode_path(&self.username), + percent_encode_path(&self.password), ); let req = Request::get(url); @@ -423,14 +439,14 @@ pub(super) fn parse_list_metadata(content: ListMetadata) -> Result { #[derive(Debug, Deserialize)] pub struct GetFileLinkResponse { - pub result: u64, + pub result: u32, pub path: Option, pub hosts: Option>, } #[derive(Debug, Deserialize)] pub struct StatResponse { - pub result: u64, + pub result: u32, pub metadata: Option, } @@ -445,7 +461,7 @@ pub struct StatMetadata { #[derive(Debug, Deserialize)] pub struct ListFolderResponse { - pub result: u64, + pub result: u32, pub metadata: Option, } @@ -459,3 +475,15 @@ pub struct ListMetadata { pub contenttype: Option, pub contents: Option>, } + +#[derive(Debug, Deserialize)] +pub struct UploadFileResponse { + pub result: u32, + pub metadata: Vec, +} + +#[derive(Debug, Deserialize)] +pub struct UploadProgressResponse { + pub result: u32, + pub finished: Option, +} diff --git a/core/src/services/pcloud/error.rs b/core/src/services/pcloud/error.rs index e59bcb672cc8..054859b8e112 100644 --- a/core/src/services/pcloud/error.rs +++ b/core/src/services/pcloud/error.rs @@ -42,16 +42,38 @@ impl Debug for PcloudError { } } +/// Deal with error response result. +pub fn parse_result(result: u32) -> Result<()> { + match result / 1000 { + 4 | 5 => { + let mut err = Error::new(ErrorKind::Unexpected, "Pcloud service returns an error."); + err = err.set_temporary(); + Err(err) + } + _ => Ok(()), + } +} + /// Parse error response into Error. pub async fn parse_error(resp: Response) -> Result { let (parts, body) = resp.into_parts(); + + let (kind, retryable) = match parts.status.as_u16() { + 429 | 500 | 502 | 503 | 504 | 509 => (ErrorKind::Unexpected, true), + _ => (ErrorKind::Unexpected, false), + }; + let bs = body.bytes().await?; let message = String::from_utf8_lossy(&bs).into_owned(); - let mut err = Error::new(ErrorKind::Unexpected, &message); + let mut err = Error::new(kind, &message); err = with_error_response_context(err, parts); + if retryable { + err = err.set_temporary(); + } + Ok(err) } diff --git a/core/src/services/pcloud/lister.rs b/core/src/services/pcloud/lister.rs index e4dd65c38127..7334a7ea4a3e 100644 --- a/core/src/services/pcloud/lister.rs +++ b/core/src/services/pcloud/lister.rs @@ -21,7 +21,7 @@ use async_trait::async_trait; use http::StatusCode; use super::core::*; -use super::error::parse_error; +use super::error::{parse_error, parse_result}; use crate::raw::oio::Entry; use crate::raw::*; use crate::*; @@ -56,6 +56,8 @@ impl oio::PageList for PcloudLister { serde_json::from_slice(&bs).map_err(new_json_deserialize_error)?; let result = resp.result; + parse_result(result)?; + if result == 2005 { ctx.done = true; return Ok(()); diff --git a/core/src/services/pcloud/writer.rs b/core/src/services/pcloud/writer.rs index 108080ff172a..1a56b4664d0b 100644 --- a/core/src/services/pcloud/writer.rs +++ b/core/src/services/pcloud/writer.rs @@ -21,8 +21,10 @@ use async_trait::async_trait; use http::StatusCode; use super::core::PcloudCore; +use super::core::StatResponse; +use super::core::UploadFileResponse; use super::error::parse_error; -use super::error::PcloudError; +use super::error::parse_result; use crate::raw::*; use crate::*; @@ -46,24 +48,49 @@ impl oio::OneShotWrite for PcloudWriter { self.core.ensure_dir_exists(&self.path).await?; - let resp = self.core.upload_file(&self.path, bs).await?; + let mut exist = false; - let status = resp.status(); + while !exist { + let resp = self.core.upload_file(&self.path, bs.clone()).await?; - match status { - StatusCode::OK => { - let bs = resp.into_body().bytes().await?; - let resp: PcloudError = - serde_json::from_slice(&bs).map_err(new_json_deserialize_error)?; - let result = resp.result; + let status = resp.status(); - if result != 0 { - return Err(Error::new(ErrorKind::Unexpected, &format!("{resp:?}"))); - } + match status { + StatusCode::OK => { + let bs = resp.into_body().bytes().await?; + let resp: UploadFileResponse = + serde_json::from_slice(&bs).map_err(new_json_deserialize_error)?; + let result = resp.result; + + parse_result(result)?; + + if result != 0 { + return Err(Error::new(ErrorKind::Unexpected, &format!("{resp:?}"))); + } + + if resp.metadata.len() != 1 { + return Err(Error::new(ErrorKind::Unexpected, &format!("{resp:?}"))); + } - Ok(()) + let resp = self.core.stat(&self.path).await?; + + let bs = resp.into_body().bytes().await?; + let resp: StatResponse = + serde_json::from_slice(&bs).map_err(new_json_deserialize_error)?; + let result = resp.result; + + parse_result(result)?; + + if result == 2010 || result == 2055 || result == 2002 { + let mut err = Error::new(ErrorKind::Unexpected, &format!("{resp:?}")); + err = err.set_temporary(); + return Err(err); + } + exist = true; + } + _ => return Err(parse_error(resp).await?), } - _ => Err(parse_error(resp).await?), } + Ok(()) } }