Add Tupperbox importing support
This commit is contained in:
parent
9be7514fb9
commit
1e1ef4495f
@ -1,7 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Net;
|
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
@ -27,19 +26,66 @@ namespace PluralKit.Bot.Commands
|
|||||||
{
|
{
|
||||||
var response = await client.GetAsync(url);
|
var response = await client.GetAsync(url);
|
||||||
if (!response.IsSuccessStatusCode) throw Errors.InvalidImportFile;
|
if (!response.IsSuccessStatusCode) throw Errors.InvalidImportFile;
|
||||||
var str = await response.Content.ReadAsStringAsync();
|
var json = await response.Content.ReadAsStringAsync();
|
||||||
|
|
||||||
var data = TryDeserialize(str);
|
var settings = new JsonSerializerSettings
|
||||||
if (!data.HasValue || !data.Value.Valid) throw Errors.InvalidImportFile;
|
|
||||||
|
|
||||||
if (Context.SenderSystem != null && Context.SenderSystem.Hid != data.Value.Id)
|
|
||||||
{
|
{
|
||||||
// TODO: prompt "are you sure you want to import someone else's system?
|
MissingMemberHandling = MissingMemberHandling.Error
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
DataFileSystem data;
|
||||||
|
|
||||||
|
// TODO: can we clean up this mess?
|
||||||
|
try
|
||||||
|
{
|
||||||
|
data = JsonConvert.DeserializeObject<DataFileSystem>(json, settings);
|
||||||
|
}
|
||||||
|
catch (JsonException)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var tupperbox = JsonConvert.DeserializeObject<TupperboxProfile>(json, settings);
|
||||||
|
if (!tupperbox.Valid) throw Errors.InvalidImportFile;
|
||||||
|
|
||||||
|
var res = tupperbox.ToPluralKit();
|
||||||
|
if (res.HadGroups || res.HadMultibrackets || res.HadIndividualTags)
|
||||||
|
{
|
||||||
|
var issueStr =
|
||||||
|
$"{Emojis.Warn} The following potential issues were detected converting your Tupperbox input file:";
|
||||||
|
if (res.HadGroups)
|
||||||
|
issueStr +=
|
||||||
|
"\n- PluralKit does not support member groups. Members will be imported without groups.";
|
||||||
|
if (res.HadMultibrackets)
|
||||||
|
issueStr += "\n- PluralKit does not support members with multiple proxy tags. Only the first pair will be imported.";
|
||||||
|
if (res.HadIndividualTags)
|
||||||
|
issueStr +=
|
||||||
|
"\n- PluralKit does not support per-member system tags. Since you had multiple members with distinct tags, tags will not be imported. You can set your system tag using the `pk;system tag <tag>` command later.";
|
||||||
|
|
||||||
|
var msg = await Context.Channel.SendMessageAsync($"{issueStr}\n\nDo you want to proceed with the import?");
|
||||||
|
if (!await Context.PromptYesNo(msg)) throw Errors.ImportCancelled;
|
||||||
|
}
|
||||||
|
|
||||||
|
data = res.System;
|
||||||
|
}
|
||||||
|
catch (JsonException)
|
||||||
|
{
|
||||||
|
throw Errors.InvalidImportFile;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (!data.Valid) throw Errors.InvalidImportFile;
|
||||||
|
|
||||||
|
if (data.LinkedAccounts != null && !data.LinkedAccounts.Contains(Context.User.Id))
|
||||||
|
{
|
||||||
|
var msg = await Context.Channel.SendMessageAsync($"{Emojis.Warn} You seem to importing a system profile belonging to another account. Are you sure you want to proceed?");
|
||||||
|
if (!await Context.PromptYesNo(msg)) throw Errors.ImportCancelled;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If passed system is null, it'll create a new one
|
// If passed system is null, it'll create a new one
|
||||||
// (and that's okay!)
|
// (and that's okay!)
|
||||||
var result = await DataFiles.ImportSystem(data.Value, Context.SenderSystem);
|
var result = await DataFiles.ImportSystem(data, Context.SenderSystem);
|
||||||
|
|
||||||
if (Context.SenderSystem == null)
|
if (Context.SenderSystem == null)
|
||||||
{
|
{
|
||||||
@ -72,7 +118,7 @@ namespace PluralKit.Bot.Commands
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
return JsonConvert.DeserializeObject<DataFileSystem>(json);
|
|
||||||
}
|
}
|
||||||
catch (JsonException e)
|
catch (JsonException e)
|
||||||
{
|
{
|
||||||
|
@ -61,7 +61,8 @@ namespace PluralKit.Bot {
|
|||||||
public static PKError InvalidTimeZone(string zoneStr) => new PKError($"Invalid time zone ID '{zoneStr}'. To find your time zone ID, use the following website: <https://xske.github.io/tz>");
|
public static PKError InvalidTimeZone(string zoneStr) => new PKError($"Invalid time zone ID '{zoneStr}'. To find your time zone ID, use the following website: <https://xske.github.io/tz>");
|
||||||
public static PKError TimezoneChangeCancelled => new PKError("Time zone change cancelled.");
|
public static PKError TimezoneChangeCancelled => new PKError("Time zone change cancelled.");
|
||||||
public static PKError AmbiguousTimeZone(string zoneStr, int count) => new PKError($"The time zone query '{zoneStr}' resulted in **{count}** different time zone regions. Try being more specific - e.g. pass an exact time zone specifier from the following website: <https://xske.github.io/tz>");
|
public static PKError AmbiguousTimeZone(string zoneStr, int count) => new PKError($"The time zone query '{zoneStr}' resulted in **{count}** different time zone regions. Try being more specific - e.g. pass an exact time zone specifier from the following website: <https://xske.github.io/tz>");
|
||||||
public static Exception NoImportFilePassed => new PKError("You must either pass an URL to a file as a command parameter, or as an attachment to the message containing the command.");
|
public static PKError NoImportFilePassed => new PKError("You must either pass an URL to a file as a command parameter, or as an attachment to the message containing the command.");
|
||||||
public static Exception InvalidImportFile => new PKError("Imported data file invalid. Make sure this is a .json file directly exported from PluralKit or Tupperbox.");
|
public static PKError InvalidImportFile => new PKError("Imported data file invalid. Make sure this is a .json file directly exported from PluralKit or Tupperbox.");
|
||||||
|
public static PKError ImportCancelled => new PKError("Import cancelled.");
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -4,7 +4,6 @@ using System.Threading.Tasks;
|
|||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
using NodaTime;
|
using NodaTime;
|
||||||
using NodaTime.Text;
|
using NodaTime.Text;
|
||||||
using NodaTime.TimeZones;
|
|
||||||
|
|
||||||
namespace PluralKit.Bot
|
namespace PluralKit.Bot
|
||||||
{
|
{
|
||||||
@ -49,7 +48,7 @@ namespace PluralKit.Bot
|
|||||||
Id = member.Hid,
|
Id = member.Hid,
|
||||||
Name = member.Name,
|
Name = member.Name,
|
||||||
Description = member.Description,
|
Description = member.Description,
|
||||||
Birthdate = member.Birthday?.ToString(Formats.DateExportFormat, null),
|
Birthday = member.Birthday?.ToString(Formats.DateExportFormat, null),
|
||||||
Pronouns = member.Pronouns,
|
Pronouns = member.Pronouns,
|
||||||
Color = member.Color,
|
Color = member.Color,
|
||||||
AvatarUrl = member.AvatarUrl,
|
AvatarUrl = member.AvatarUrl,
|
||||||
@ -67,19 +66,19 @@ namespace PluralKit.Bot
|
|||||||
|
|
||||||
public async Task<ImportResult> ImportSystem(DataFileSystem data, PKSystem system)
|
public async Task<ImportResult> ImportSystem(DataFileSystem data, PKSystem system)
|
||||||
{
|
{
|
||||||
var result = new ImportResult { AddedNames = new List<string>(), ModifiedNames = new List<string>() };
|
var result = new ImportResult {AddedNames = new List<string>(), ModifiedNames = new List<string>()};
|
||||||
|
|
||||||
// If we don't already have a system to save to, create one
|
// If we don't already have a system to save to, create one
|
||||||
if (system == null) system = await _systems.Create(data.Name);
|
if (system == null) system = await _systems.Create(data.Name);
|
||||||
|
|
||||||
// Apply system info
|
// Apply system info
|
||||||
system.Name = data.Name;
|
system.Name = data.Name;
|
||||||
system.Description = data.Description;
|
if (data.Description != null) system.Description = data.Description;
|
||||||
system.Tag = data.Tag;
|
if (data.Tag != null) system.Tag = data.Tag;
|
||||||
system.AvatarUrl = data.AvatarUrl;
|
if (data.AvatarUrl != null) system.AvatarUrl = data.AvatarUrl;
|
||||||
system.UiTz = data.TimeZone ?? "UTC";
|
if (data.TimeZone != null) system.UiTz = data.TimeZone ?? "UTC";
|
||||||
await _systems.Save(system);
|
await _systems.Save(system);
|
||||||
|
|
||||||
// Apply members
|
// Apply members
|
||||||
// TODO: parallelize?
|
// TODO: parallelize?
|
||||||
foreach (var dataMember in data.Members)
|
foreach (var dataMember in data.Members)
|
||||||
@ -93,10 +92,10 @@ namespace PluralKit.Bot
|
|||||||
// ...but if it's a different system's member, we just make a new one anyway
|
// ...but if it's a different system's member, we just make a new one anyway
|
||||||
if (member != null && member.System != system.Id) member = null;
|
if (member != null && member.System != system.Id) member = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try to look up by name, too
|
// Try to look up by name, too
|
||||||
if (member == null) member = await _members.GetByName(system, dataMember.Name);
|
if (member == null) member = await _members.GetByName(system, dataMember.Name);
|
||||||
|
|
||||||
// And if all else fails (eg. fresh import from Tupperbox, etc) we just make a member lol
|
// And if all else fails (eg. fresh import from Tupperbox, etc) we just make a member lol
|
||||||
if (member == null)
|
if (member == null)
|
||||||
{
|
{
|
||||||
@ -107,20 +106,28 @@ namespace PluralKit.Bot
|
|||||||
{
|
{
|
||||||
result.ModifiedNames.Add(dataMember.Name);
|
result.ModifiedNames.Add(dataMember.Name);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Apply member info
|
// Apply member info
|
||||||
member.Name = dataMember.Name;
|
member.Name = dataMember.Name;
|
||||||
member.Description = dataMember.Description;
|
if (dataMember.Description != null) member.Description = dataMember.Description;
|
||||||
member.Color = dataMember.Color;
|
if (dataMember.Color != null) member.Color = dataMember.Color;
|
||||||
member.AvatarUrl = dataMember.AvatarUrl;
|
if (dataMember.AvatarUrl != null) member.AvatarUrl = dataMember.AvatarUrl;
|
||||||
member.Prefix = dataMember.Prefix;
|
if (dataMember.Prefix != null || dataMember.Suffix != null)
|
||||||
member.Suffix = dataMember.Suffix;
|
{
|
||||||
|
member.Prefix = dataMember.Prefix;
|
||||||
|
member.Suffix = dataMember.Suffix;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dataMember.Birthday != null)
|
||||||
|
{
|
||||||
|
var birthdayParse = LocalDatePattern.CreateWithInvariantCulture(Formats.DateExportFormat)
|
||||||
|
.Parse(dataMember.Birthday);
|
||||||
|
member.Birthday = birthdayParse.Success ? (LocalDate?) birthdayParse.Value : null;
|
||||||
|
}
|
||||||
|
|
||||||
var birthdayParse = LocalDatePattern.CreateWithInvariantCulture(Formats.DateExportFormat).Parse(dataMember.Birthdate);
|
|
||||||
member.Birthday = birthdayParse.Success ? (LocalDate?) birthdayParse.Value : null;
|
|
||||||
await _members.Save(member);
|
await _members.Save(member);
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: import switches, too?
|
// TODO: import switches, too?
|
||||||
|
|
||||||
result.System = system;
|
result.System = system;
|
||||||
@ -137,87 +144,128 @@ namespace PluralKit.Bot
|
|||||||
|
|
||||||
public struct DataFileSystem
|
public struct DataFileSystem
|
||||||
{
|
{
|
||||||
[JsonProperty("id")]
|
[JsonProperty("id")] public string Id;
|
||||||
public string Id;
|
[JsonProperty("name")] public string Name;
|
||||||
|
[JsonProperty("description")] public string Description;
|
||||||
[JsonProperty("name")]
|
[JsonProperty("tag")] public string Tag;
|
||||||
public string Name;
|
[JsonProperty("avatar_url")] public string AvatarUrl;
|
||||||
|
[JsonProperty("timezone")] public string TimeZone;
|
||||||
[JsonProperty("description")]
|
[JsonProperty("members")] public ICollection<DataFileMember> Members;
|
||||||
public string Description;
|
[JsonProperty("switches")] public ICollection<DataFileSwitch> Switches;
|
||||||
|
[JsonProperty("accounts")] public ICollection<ulong> LinkedAccounts;
|
||||||
[JsonProperty("tag")]
|
[JsonProperty("created")] public string Created;
|
||||||
public string Tag;
|
|
||||||
|
|
||||||
[JsonProperty("avatar_url")]
|
|
||||||
public string AvatarUrl;
|
|
||||||
|
|
||||||
[JsonProperty("timezone")]
|
|
||||||
public string TimeZone;
|
|
||||||
|
|
||||||
[JsonProperty("members")]
|
|
||||||
public ICollection<DataFileMember> Members;
|
|
||||||
|
|
||||||
[JsonProperty("switches")]
|
|
||||||
public ICollection<DataFileSwitch> Switches;
|
|
||||||
|
|
||||||
[JsonProperty("accounts")]
|
|
||||||
public ICollection<ulong> LinkedAccounts;
|
|
||||||
|
|
||||||
[JsonProperty("created")]
|
|
||||||
public string Created;
|
|
||||||
|
|
||||||
private bool TimeZoneValid => TimeZone == null || DateTimeZoneProviders.Tzdb.GetZoneOrNull(TimeZone) != null;
|
private bool TimeZoneValid => TimeZone == null || DateTimeZoneProviders.Tzdb.GetZoneOrNull(TimeZone) != null;
|
||||||
|
|
||||||
[JsonIgnore]
|
[JsonIgnore] public bool Valid => TimeZoneValid && Members != null && Members.All(m => m.Valid);
|
||||||
public bool Valid => TimeZoneValid && Members.All(m => m.Valid);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public struct DataFileMember
|
public struct DataFileMember
|
||||||
{
|
{
|
||||||
[JsonProperty("id")]
|
[JsonProperty("id")] public string Id;
|
||||||
public string Id;
|
[JsonProperty("name")] public string Name;
|
||||||
|
[JsonProperty("description")] public string Description;
|
||||||
[JsonProperty("name")]
|
[JsonProperty("birthday")] public string Birthday;
|
||||||
public string Name;
|
[JsonProperty("pronouns")] public string Pronouns;
|
||||||
|
[JsonProperty("color")] public string Color;
|
||||||
[JsonProperty("description")]
|
[JsonProperty("avatar_url")] public string AvatarUrl;
|
||||||
public string Description;
|
[JsonProperty("prefix")] public string Prefix;
|
||||||
|
[JsonProperty("suffix")] public string Suffix;
|
||||||
[JsonProperty("birthday")]
|
[JsonProperty("message_count")] public int MessageCount;
|
||||||
public string Birthdate;
|
[JsonProperty("created")] public string Created;
|
||||||
|
|
||||||
[JsonProperty("pronouns")]
|
|
||||||
public string Pronouns;
|
|
||||||
|
|
||||||
[JsonProperty("color")]
|
|
||||||
public string Color;
|
|
||||||
|
|
||||||
[JsonProperty("avatar_url")]
|
|
||||||
public string AvatarUrl;
|
|
||||||
|
|
||||||
[JsonProperty("prefix")]
|
|
||||||
public string Prefix;
|
|
||||||
|
|
||||||
[JsonProperty("suffix")]
|
|
||||||
public string Suffix;
|
|
||||||
|
|
||||||
[JsonProperty("message_count")]
|
|
||||||
public int MessageCount;
|
|
||||||
|
|
||||||
[JsonProperty("created")]
|
[JsonIgnore] public bool Valid => Name != null;
|
||||||
public string Created;
|
|
||||||
|
|
||||||
[JsonIgnore]
|
|
||||||
public bool Valid => Name != null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public struct DataFileSwitch
|
public struct DataFileSwitch
|
||||||
{
|
{
|
||||||
[JsonProperty("timestamp")]
|
[JsonProperty("timestamp")] public string Timestamp;
|
||||||
public string Timestamp;
|
[JsonProperty("members")] public ICollection<string> Members;
|
||||||
|
}
|
||||||
[JsonProperty("members")]
|
|
||||||
public ICollection<string> Members;
|
public struct TupperboxConversionResult
|
||||||
|
{
|
||||||
|
public bool HadGroups;
|
||||||
|
public bool HadIndividualTags;
|
||||||
|
public bool HadMultibrackets;
|
||||||
|
public DataFileSystem System;
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct TupperboxProfile
|
||||||
|
{
|
||||||
|
[JsonProperty("tuppers")] public ICollection<TupperboxTupper> Tuppers;
|
||||||
|
[JsonProperty("groups")] public ICollection<TupperboxGroup> Groups;
|
||||||
|
|
||||||
|
[JsonIgnore] public bool Valid => Tuppers != null && Groups != null && Tuppers.All(t => t.Valid) && Groups.All(g => g.Valid);
|
||||||
|
|
||||||
|
public TupperboxConversionResult ToPluralKit()
|
||||||
|
{
|
||||||
|
// Set by member conversion function
|
||||||
|
string lastSetTag = null;
|
||||||
|
|
||||||
|
TupperboxConversionResult output = default(TupperboxConversionResult);
|
||||||
|
|
||||||
|
output.System = new DataFileSystem
|
||||||
|
{
|
||||||
|
Members = Tuppers.Select(t => t.ToPluralKit(ref lastSetTag, ref output.HadMultibrackets,
|
||||||
|
ref output.HadGroups, ref output.HadMultibrackets)).ToList(),
|
||||||
|
|
||||||
|
// If we haven't had multiple tags set, use the last (and only) one we set as the system tag
|
||||||
|
Tag = !output.HadIndividualTags ? lastSetTag : null
|
||||||
|
};
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct TupperboxTupper
|
||||||
|
{
|
||||||
|
[JsonProperty("name")] public string Name;
|
||||||
|
[JsonProperty("avatar_url")] public string AvatarUrl;
|
||||||
|
[JsonProperty("brackets")] public ICollection<string> Brackets;
|
||||||
|
[JsonProperty("posts")] public int Posts; // Not supported by PK
|
||||||
|
[JsonProperty("show_brackets")] public bool ShowBrackets; // Not supported by PK
|
||||||
|
[JsonProperty("birthday")] public string Birthday;
|
||||||
|
[JsonProperty("description")] public string Description;
|
||||||
|
[JsonProperty("tag")] public string Tag; // Not supported by PK
|
||||||
|
[JsonProperty("group_id")] public string GroupId; // Not supported by PK
|
||||||
|
[JsonProperty("group_pos")] public int? GroupPos; // Not supported by PK
|
||||||
|
|
||||||
|
[JsonIgnore] public bool Valid => Name != null && Brackets != null && Brackets.Count % 2 == 0;
|
||||||
|
|
||||||
|
public DataFileMember ToPluralKit(ref string lastSetTag, ref bool multipleTags, ref bool hasGroup, ref bool hasMultiBrackets)
|
||||||
|
{
|
||||||
|
// If we've set a tag before and it's not the same as this one,
|
||||||
|
// then we have multiple unique tags and we pass that flag back to the caller
|
||||||
|
if (Tag != null && lastSetTag != null && lastSetTag != Tag) multipleTags = true;
|
||||||
|
lastSetTag = Tag;
|
||||||
|
|
||||||
|
// If this member is in a group, we have a (used) group and we flag that
|
||||||
|
if (GroupId != null) hasGroup = true;
|
||||||
|
|
||||||
|
// Brackets in Tupperbox format are arranged as a single array
|
||||||
|
// [prefix1, suffix1, prefix2, suffix2, prefix3... etc]
|
||||||
|
// If there are more than two entries this member has multiple brackets and we flag that
|
||||||
|
if (Brackets.Count > 2) hasMultiBrackets = true;
|
||||||
|
|
||||||
|
return new DataFileMember
|
||||||
|
{
|
||||||
|
Name = Name,
|
||||||
|
AvatarUrl = AvatarUrl,
|
||||||
|
Birthday = Birthday,
|
||||||
|
Description = Description,
|
||||||
|
Prefix = Brackets.FirstOrDefault(),
|
||||||
|
Suffix = Brackets.Skip(1).FirstOrDefault() // TODO: can Tupperbox members have no proxies at all?
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public struct TupperboxGroup
|
||||||
|
{
|
||||||
|
[JsonProperty("id")] public int Id;
|
||||||
|
[JsonProperty("name")] public string Name;
|
||||||
|
[JsonProperty("description")] public string Description;
|
||||||
|
[JsonProperty("tag")] public string Tag;
|
||||||
|
|
||||||
|
[JsonIgnore] public bool Valid => true;
|
||||||
}
|
}
|
||||||
}
|
}
|
Loading…
Reference in New Issue
Block a user