mirror of
https://github.com/duplicati/duplicati.git
synced 2025-11-28 11:30:24 +08:00
105 lines
5.2 KiB
C#
105 lines
5.2 KiB
C#
// Copyright (C) 2025, The Duplicati Team
|
|
// https://duplicati.com, hello@duplicati.com
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the "Software"),
|
|
// to deal in the Software without restriction, including without limitation
|
|
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
// and/or sell copies of the Software, and to permit persons to whom the
|
|
// Software is furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
// DEALINGS IN THE SOFTWARE.
|
|
|
|
using Duplicati.Library.Backend.MicrosoftGraph;
|
|
using Duplicati.Library.Interface;
|
|
|
|
namespace Duplicati.Library.Backend
|
|
{
|
|
public class MicrosoftGroup : MicrosoftGraphBackend
|
|
{
|
|
private const string GROUP_EMAIL_OPTION = "group-email";
|
|
private const string GROUP_ID_OPTION = "group-id";
|
|
private const string PROTOCOL_KEY = "msgroup";
|
|
|
|
private readonly Dictionary<string, string?> options;
|
|
private string? drivePath;
|
|
|
|
public MicrosoftGroup()
|
|
{
|
|
// Constructor needed for dynamic loading to find it
|
|
options = null!;
|
|
}
|
|
|
|
public MicrosoftGroup(string url, Dictionary<string, string?> options)
|
|
: base(url, PROTOCOL_KEY, options)
|
|
{
|
|
this.options = options;
|
|
}
|
|
|
|
public override string ProtocolKey => PROTOCOL_KEY;
|
|
|
|
public override string DisplayName => Strings.MicrosoftGroup.DisplayName;
|
|
|
|
protected override async Task<string> GetDrivePath(CancellationToken cancelToken)
|
|
{
|
|
if (string.IsNullOrEmpty(drivePath))
|
|
{
|
|
string? groupId = null;
|
|
var groupEmail = options.GetValueOrDefault(GROUP_EMAIL_OPTION);
|
|
if (!string.IsNullOrWhiteSpace(groupEmail))
|
|
groupId = await this.GetGroupIdFromEmailAsync(groupEmail, cancelToken).ConfigureAwait(false);
|
|
|
|
var groupIdOption = options.GetValueOrDefault(GROUP_ID_OPTION);
|
|
if (!string.IsNullOrWhiteSpace(groupIdOption))
|
|
{
|
|
if (!string.IsNullOrEmpty(groupId) && !string.Equals(groupId, groupIdOption))
|
|
throw new UserInformationException(Strings.MicrosoftGroup.ConflictingGroupId(groupIdOption, groupId), "MicrosoftGroupConflictingGroupId");
|
|
|
|
groupId = groupIdOption;
|
|
}
|
|
|
|
if (string.IsNullOrWhiteSpace(groupId))
|
|
throw new UserInformationException(Strings.MicrosoftGroup.MissingGroupIdAndEmailAddress, "MicrosoftGroupMissingGroupIdAndEmailAddress");
|
|
|
|
drivePath = string.Format("/groups/{0}/drive", groupId);
|
|
}
|
|
|
|
return drivePath;
|
|
}
|
|
|
|
protected override DescriptionTemplateDelegate DescriptionTemplate => Strings.MicrosoftGroup.Description;
|
|
|
|
protected override IList<ICommandLineArgument> AdditionalSupportedCommands => [
|
|
new CommandLineArgument(GROUP_ID_OPTION, CommandLineArgument.ArgumentType.String, Strings.MicrosoftGroup.GroupIdShort, Strings.MicrosoftGroup.GroupIdLong),
|
|
new CommandLineArgument(GROUP_EMAIL_OPTION, CommandLineArgument.ArgumentType.String, Strings.MicrosoftGroup.GroupEmailShort, Strings.MicrosoftGroup.GroupEmailLong),
|
|
];
|
|
|
|
private async Task<string> GetGroupIdFromEmailAsync(string email, CancellationToken cancelToken)
|
|
{
|
|
// We can get all groups that have the given email as one of their addresses with:
|
|
// https://graph.microsoft.com/v1.0/groups?$filter=mail eq '{email}' or proxyAddresses/any(x:x eq 'smtp:{email}')
|
|
string request = string.Format("{0}/groups?$filter=mail eq '{1}' or proxyAddresses/any(x:x eq 'smtp:{1}')", this.ApiVersion, email);
|
|
var groups = await Utility.Utility.WithTimeout(m_timeouts.ShortTimeout, cancelToken, ct => GetAsync<GraphCollection<Group>>(request, ct)).ConfigureAwait(false);
|
|
if (groups.Value == null || groups.Value.Length == 0)
|
|
throw new UserInformationException(Strings.MicrosoftGroup.NoGroupsWithEmail(email), "MicrosoftGroupNoGroupsWithEmail");
|
|
|
|
if (groups.Value.Length > 1)
|
|
throw new UserInformationException(Strings.MicrosoftGroup.MultipleGroupsWithEmail(email), "MicrosoftGroupMultipleGroupsWithEmail");
|
|
|
|
var id = groups.Value.Single().Id;
|
|
if (string.IsNullOrEmpty(id))
|
|
throw new UserInformationException(Strings.MicrosoftGroup.NoGroupsWithEmail(email), "MicrosoftGroupNoGroupsWithEmail");
|
|
|
|
return id;
|
|
}
|
|
}
|
|
}
|