// Copyright (C) 2015, The Duplicati Team // http://www.duplicati.com, info@duplicati.com // // This library is free software; you can redistribute it and/or modify // it under the terms of the GNU Lesser General Public License as // published by the Free Software Foundation; either version 2.1 of the // License, or (at your option) any later version. // // This library 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 // Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public // License along with this library; if not, write to the Free Software // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA using System; using System.Net; using System.Text; using Newtonsoft.Json; using Duplicati.Library.Utility; namespace Duplicati.Library.Backend.Backblaze { public class B2AuthHelper : JSONWebHelper { private readonly string m_credentials; private AuthResponse m_config; private DateTime m_configExpires; private const string AUTH_URL = "https://api.backblazeb2.com/b2api/v1/b2_authorize_account"; public B2AuthHelper(string userid, string password) : base() { m_credentials = Convert.ToBase64String(Encoding.UTF8.GetBytes(userid + ":" + password)); } public override HttpWebRequest CreateRequest(string url, string method = null) { var r = base.CreateRequest(url, method); r.Headers["Authorization"] = AuthorizationToken; return r; } public string AuthorizationToken { get { return Config.AuthorizationToken; } } public string APIUrl { get { return Config.APIUrl; } } public string DownloadUrl { get { return Config.DownloadUrl; } } public string AccountID { get { return Config.AccountID; } } private string DropTrailingSlashes(string url) { while(url.EndsWith("/", StringComparison.Ordinal)) url = url.Substring(0, url.Length - 1); return url; } private AuthResponse Config { get { if (m_config == null || m_configExpires < DateTime.UtcNow) { var retries = 0; while(true) { try { var req = base.CreateRequest(AUTH_URL); req.Headers.Add("Authorization", string.Format("Basic {0}", m_credentials)); req.ContentType = "application/json; charset=utf-8"; using(var resp = (HttpWebResponse)new AsyncHttpRequest(req).GetResponse()) m_config = ReadJSONResponse(resp); m_config.APIUrl = DropTrailingSlashes(m_config.APIUrl); m_config.DownloadUrl = DropTrailingSlashes(m_config.DownloadUrl); m_configExpires = DateTime.UtcNow + TimeSpan.FromHours(1); return m_config; } catch (Exception ex) { var msg = ex.Message; var clienterror = false; try { // Only retry once on client errors if (ex is WebException && (ex as WebException).Response is HttpWebResponse) { var sc = (int)((ex as WebException).Response as HttpWebResponse).StatusCode; clienterror = (sc >= 400 && sc <= 499); } } catch { } if (retries >= (clienterror ? 1 : 5)) { AttemptParseAndThrowException(ex); throw; } System.Threading.Thread.Sleep(TimeSpan.FromSeconds(Math.Pow(2, retries))); retries++; } } } return m_config; } } public static void AttemptParseAndThrowException(Exception ex) { Exception newex = null; try { if (ex is WebException && (ex as WebException).Response is HttpWebResponse) { string rawdata = null; var hs = (ex as WebException).Response as HttpWebResponse; using(var rs = Library.Utility.AsyncHttpRequest.TrySetTimeout(hs.GetResponseStream())) using(var sr = new System.IO.StreamReader(rs)) rawdata = sr.ReadToEnd(); newex = new Exception("Raw message: " + rawdata); var msg = JsonConvert.DeserializeObject(rawdata); newex = new Exception(string.Format("{0} - {1}: {2}", msg.Status, msg.Code, msg.Message)); } } catch { } if (newex != null) throw newex; } protected override void ParseException(Exception ex) { AttemptParseAndThrowException(ex); } public static HttpStatusCode GetExceptionStatusCode(Exception ex) { if (ex is WebException && (ex as WebException).Response is HttpWebResponse) return ((ex as WebException).Response as HttpWebResponse).StatusCode; else return (HttpStatusCode)0; } private class ErrorResponse { [JsonProperty("code")] public string Code { get; set; } [JsonProperty("message")] public string Message { get; set; } [JsonProperty("status")] public long Status { get; set; } } private class AuthResponse { [JsonProperty("accountId")] public string AccountID { get; set; } [JsonProperty("apiUrl")] public string APIUrl { get; set; } [JsonProperty("authorizationToken")] public string AuthorizationToken { get; set; } [JsonProperty("downloadUrl")] public string DownloadUrl { get; set; } } } }