diff --git a/CHANGELOG.md b/CHANGELOG.md
index 69a34d68e1..e0338ead57 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -11,6 +11,7 @@ All notable changes to this project will be documented in this file.
- [GUI] Add import downloads menu item to GUI (#2246 by: HebaruSan; reviewed: politas)
- [Core] Accept header and infrastructure for auth tokens (#2263 by: HebaruSan; reviewed: dbent)
- [CLI] Add Cmdline import command (#2264 by: HebaruSan; reviewed: politas)
+- [Multiple] User interfaces for auth tokens (#2266 by: HebaruSan; reviewed: politas)
### Bugfixes
@@ -21,6 +22,7 @@ All notable changes to this project will be documented in this file.
- [Core] Fix missing filename in install -c log message (No PR, by: HebaruSan)
- [GUI] Leave out children already shown in ancestor node (#2258 by: HebaruSan; reviewed: politas)
- [GUI] Resolve provides for install-from-ckan-file (#2259 by: HebaruSan; reviewed: politas)
+- [Build] Use arch=32 for OSX (#2270 by: HebaruSan; reviewed: techman83)
## v1.24.0-PRE1 (McCandless)
diff --git a/Cmdline/Action/AuthToken.cs b/Cmdline/Action/AuthToken.cs
new file mode 100644
index 0000000000..5bb806fc5b
--- /dev/null
+++ b/Cmdline/Action/AuthToken.cs
@@ -0,0 +1,175 @@
+using System;
+using System.Collections.Generic;
+using CommandLine;
+using CommandLine.Text;
+using log4net;
+
+namespace CKAN.CmdLine
+{
+ ///
+ /// Subcommand for managing authentication tokens
+ ///
+ public class AuthToken : ISubCommand
+ {
+ ///
+ /// Initialize the subcommand
+ ///
+ public AuthToken() { }
+
+ ///
+ /// Run the subcommand
+ ///
+ /// Command line parameters not yet handled by parser
+ ///
+ /// Exit code
+ ///
+ public int RunSubCommand(SubCommandOptions unparsed)
+ {
+ string[] args = unparsed.options.ToArray();
+ int exitCode = Exit.OK;
+
+ Parser.Default.ParseArgumentsStrict(args, new AuthTokenSubOptions(), (string option, object suboptions) =>
+ {
+ if (!string.IsNullOrEmpty(option) && suboptions != null)
+ {
+ CommonOptions options = (CommonOptions)suboptions;
+ user = new ConsoleUser(options.Headless);
+ manager = new KSPManager(user);
+ exitCode = options.Handle(manager, user);
+ if (exitCode == Exit.OK)
+ {
+ switch (option)
+ {
+ case "list":
+ exitCode = listAuthTokens(options);
+ break;
+ case "add":
+ exitCode = addAuthToken((AddAuthTokenOptions)options);
+ break;
+ case "remove":
+ exitCode = removeAuthToken((RemoveAuthTokenOptions)options);
+ break;
+ }
+ }
+ }
+ }, () => { exitCode = MainClass.AfterHelp(); });
+ return exitCode;
+ }
+
+ private int listAuthTokens(CommonOptions opts)
+ {
+ List hosts = new List(Win32Registry.GetAuthTokenHosts());
+ if (hosts.Count > 0)
+ {
+ int longestHostLen = hostHeader.Length;
+ int longestTokenLen = tokenHeader.Length;
+ foreach (string host in hosts)
+ {
+ longestHostLen = Math.Max(longestHostLen, host.Length);
+ string token;
+ if (Win32Registry.TryGetAuthToken(host, out token))
+ {
+ longestTokenLen = Math.Max(longestTokenLen, token.Length);
+ }
+ }
+ // Create format string: {0,-longestHostLen} {1,-longestTokenLen}
+ string fmt = string.Format("{0}0,-{2}{1} {0}1,-{3}{1}",
+ "{", "}", longestHostLen, longestTokenLen);
+ user.RaiseMessage(fmt, hostHeader, tokenHeader);
+ user.RaiseMessage(fmt,
+ new string('-', longestHostLen),
+ new string('-', longestTokenLen)
+ );
+ foreach (string host in hosts)
+ {
+ string token;
+ if (Win32Registry.TryGetAuthToken(host, out token))
+ {
+ user.RaiseMessage(fmt, host, token);
+ }
+ }
+ }
+ return Exit.OK;
+ }
+
+ private int addAuthToken(AddAuthTokenOptions opts)
+ {
+ if (Uri.CheckHostName(opts.host) != UriHostNameType.Unknown)
+ {
+ Win32Registry.SetAuthToken(opts.host, opts.token);
+ }
+ else
+ {
+ user.RaiseError("Invalid host name: {0}", opts.host);
+ }
+ return Exit.OK;
+ }
+
+ private int removeAuthToken(RemoveAuthTokenOptions opts)
+ {
+ Win32Registry.SetAuthToken(opts.host, null);
+ return Exit.OK;
+ }
+
+ private const string hostHeader = "Host";
+ private const string tokenHeader = "Token";
+
+ private IUser user;
+ private KSPManager manager;
+ private static readonly ILog log = LogManager.GetLogger(typeof(AuthToken));
+ }
+
+ internal class AuthTokenSubOptions : VerbCommandOptions
+ {
+ [VerbOption("list", HelpText = "List auth tokens")]
+ public CommonOptions ListOptions { get; set; }
+
+ [VerbOption("add", HelpText = "Add an auth token")]
+ public AddAuthTokenOptions AddOptions { get; set; }
+
+ [VerbOption("remove", HelpText = "Delete an auth token")]
+ public RemoveAuthTokenOptions RemoveOptions { get; set; }
+
+ [HelpVerbOption]
+ public string GetUsage(string verb)
+ {
+ HelpText ht = HelpText.AutoBuild(this, verb);
+ // Add a usage prefix line
+ ht.AddPreOptionsLine(" ");
+ if (string.IsNullOrEmpty(verb))
+ {
+ ht.AddPreOptionsLine("ckan authtoken - Manage authentication tokens");
+ ht.AddPreOptionsLine($"Usage: ckan authtoken [options]");
+ }
+ else
+ {
+ ht.AddPreOptionsLine("authtoken " + verb + " - " + GetDescription(verb));
+ switch (verb)
+ {
+ case "add":
+ ht.AddPreOptionsLine($"Usage: ckan authtoken {verb} [options] host token");
+ break;
+ case "remove":
+ ht.AddPreOptionsLine($"Usage: ckan authtoken {verb} [options] host");
+ break;
+ case "list":
+ ht.AddPreOptionsLine($"Usage: ckan authtoken {verb} [options]");
+ break;
+ }
+ }
+ return ht;
+ }
+ }
+
+ internal class AddAuthTokenOptions : CommonOptions
+ {
+ [ValueOption(0)] public string host { get; set; }
+ [ValueOption(1)] public string token { get; set; }
+ }
+
+ internal class RemoveAuthTokenOptions : CommonOptions
+ {
+ [ValueOption(0)] public string host { get; set; }
+ }
+
+}
diff --git a/Cmdline/CKAN-cmdline.csproj b/Cmdline/CKAN-cmdline.csproj
index 2ca70b235e..b2750f681b 100644
--- a/Cmdline/CKAN-cmdline.csproj
+++ b/Cmdline/CKAN-cmdline.csproj
@@ -53,6 +53,7 @@
Properties\GlobalAssemblyInfo.cs
+
diff --git a/Cmdline/Main.cs b/Cmdline/Main.cs
index a35b916f92..906e736bab 100644
--- a/Cmdline/Main.cs
+++ b/Cmdline/Main.cs
@@ -63,20 +63,19 @@ public static int Main(string[] args)
switch (args[0])
{
case "repair":
- var repair = new Repair();
- return repair.RunSubCommand(new SubCommandOptions(args));
+ return (new Repair()).RunSubCommand(new SubCommandOptions(args));
case "ksp":
- var ksp = new KSP();
- return ksp.RunSubCommand(new SubCommandOptions(args));
+ return (new KSP()).RunSubCommand(new SubCommandOptions(args));
case "compat":
- var compat = new CompatSubCommand();
- return compat.RunSubCommand(new SubCommandOptions(args));
+ return (new CompatSubCommand()).RunSubCommand(new SubCommandOptions(args));
case "repo":
- var repo = new Repo();
- return repo.RunSubCommand(new SubCommandOptions(args));
+ return (new Repo()).RunSubCommand(new SubCommandOptions(args));
+
+ case "authtoken":
+ return (new AuthToken()).RunSubCommand(new SubCommandOptions(args));
}
}
catch (NoGameInstanceKraken)
diff --git a/Cmdline/Options.cs b/Cmdline/Options.cs
index 064553b015..14ad13f1d7 100644
--- a/Cmdline/Options.cs
+++ b/Cmdline/Options.cs
@@ -91,6 +91,9 @@ internal class Actions : VerbCommandOptions
[VerbOption("ksp", HelpText = "Manage KSP installs")]
public SubCommandOptions KSP { get; set; }
+ [VerbOption("authtoken", HelpText = "Manage authentication tokens")]
+ public AuthTokenSubOptions AuthToken { get; set; }
+
[VerbOption("compat", HelpText = "Manage KSP version compatibility")]
public SubCommandOptions Compat { get; set; }
diff --git a/ConsoleUI/AuthTokenAddDialog.cs b/ConsoleUI/AuthTokenAddDialog.cs
new file mode 100644
index 0000000000..2a608b5879
--- /dev/null
+++ b/ConsoleUI/AuthTokenAddDialog.cs
@@ -0,0 +1,86 @@
+using System;
+using CKAN.ConsoleUI.Toolkit;
+
+namespace CKAN.ConsoleUI {
+
+ ///
+ /// Popup for adding a new authentication token.
+ ///
+ public class AuthTokenAddDialog : ConsoleDialog {
+
+ ///
+ /// Initialize the popup.
+ ///
+ public AuthTokenAddDialog() : base()
+ {
+ CenterHeader = () => "Create Authentication Key";
+
+ int top = (Console.WindowHeight - height) / 2;
+ SetDimensions(6, top, -6, top + height - 1);
+
+ int l = GetLeft(),
+ r = GetRight(),
+ t = GetTop(),
+ b = GetBottom();
+
+ AddObject(new ConsoleLabel(
+ l + 2, t + 2, l + 2 + labelW,
+ () => "Host:",
+ () => ConsoleTheme.Current.PopupBg,
+ () => ConsoleTheme.Current.PopupFg
+ ));
+
+ hostEntry = new ConsoleField(
+ l + 2 + labelW + wPad, t + 2, r - 3
+ ) {
+ GhostText = () => ""
+ };
+ AddObject(hostEntry);
+
+ AddObject(new ConsoleLabel(
+ l + 2, t + 4, l + 2 + labelW,
+ () => "Token:",
+ () => ConsoleTheme.Current.PopupBg,
+ () => ConsoleTheme.Current.PopupFg
+ ));
+
+ tokenEntry = new ConsoleField(
+ l + 2 + labelW + wPad, t + 4, r - 3
+ ) {
+ GhostText = () => ""
+ };
+ AddObject(tokenEntry);
+
+ AddTip("Esc", "Cancel");
+ AddBinding(Keys.Escape, (object sender) => false);
+
+ AddTip("Enter", "Accept", validKey);
+ AddBinding(Keys.Enter, (object sender) => {
+ if (validKey()) {
+ Win32Registry.SetAuthToken(hostEntry.Value, tokenEntry.Value);
+ return false;
+ } else {
+ // Don't close window on Enter unless adding a key
+ return true;
+ }
+ });
+ }
+
+ private bool validKey()
+ {
+ string token;
+ return hostEntry.Value.Length > 0
+ && tokenEntry.Value.Length > 0
+ && Uri.CheckHostName(hostEntry.Value) != UriHostNameType.Unknown
+ && !Win32Registry.TryGetAuthToken(hostEntry.Value, out token);
+ }
+
+ private ConsoleField hostEntry;
+ private ConsoleField tokenEntry;
+
+ private const int wPad = 2;
+ private const int labelW = 6;
+ private const int height = 7;
+ }
+
+}
diff --git a/ConsoleUI/AuthTokenListScreen.cs b/ConsoleUI/AuthTokenListScreen.cs
new file mode 100644
index 0000000000..b043540611
--- /dev/null
+++ b/ConsoleUI/AuthTokenListScreen.cs
@@ -0,0 +1,96 @@
+using System;
+using System.ComponentModel;
+using System.Collections.Generic;
+using CKAN.ConsoleUI.Toolkit;
+
+namespace CKAN.ConsoleUI {
+
+ ///
+ /// Screen for display and editing of authentication tokens.
+ ///
+ public class AuthTokenScreen : ConsoleScreen {
+
+ ///
+ /// Initialize the screen.
+ ///
+ public AuthTokenScreen() : base()
+ {
+ LeftHeader = () => $"CKAN {Meta.GetVersion()}";
+ CenterHeader = () => "Authentication Tokens";
+ mainMenu = new ConsolePopupMenu(new List() {
+ new ConsoleMenuOption("Make a GitHub API token", "",
+ "Open the web page for creating GitHub API authentication tokens",
+ true, openGitHubURL)
+ });
+
+ AddObject(new ConsoleLabel(
+ 1, 2, -1,
+ () => "Authentication tokens for downloads:"
+ ));
+
+ tokenList = new ConsoleListBox(
+ 1, 4, -1, -2,
+ new List(Win32Registry.GetAuthTokenHosts()),
+ new List>() {
+ new ConsoleListBoxColumn() {
+ Header = "Host",
+ Width = 20,
+ Renderer = (string s) => s
+ },
+ new ConsoleListBoxColumn() {
+ Header = "Token",
+ Width = 50,
+ Renderer = (string s) => {
+ string token;
+ return Win32Registry.TryGetAuthToken(s, out token)
+ ? token
+ : missingTokenValue;
+ }
+ }
+ },
+ 0, 0, ListSortDirection.Descending
+ );
+ AddObject(tokenList);
+
+ AddObject(new ConsoleLabel(
+ 3, -1, -1,
+ () => "NOTE: These values are private! Do not share screenshots of this screen!",
+ null,
+ () => ConsoleTheme.Current.AlertFrameFg
+ ));
+
+ AddTip("Esc", "Back");
+ AddBinding(Keys.Escape, (object sender) => false);
+
+ tokenList.AddTip("A", "Add");
+ tokenList.AddBinding(Keys.A, (object sender) => {
+ AuthTokenAddDialog ad = new AuthTokenAddDialog();
+ ad.Run();
+ DrawBackground();
+ tokenList.SetData(new List(Win32Registry.GetAuthTokenHosts()));
+ return true;
+ });
+
+ tokenList.AddTip("R", "Remove", () => tokenList.Selection != null);
+ tokenList.AddBinding(Keys.R, (object sender) => {
+ if (tokenList.Selection != null) {
+ Win32Registry.SetAuthToken(tokenList.Selection, null);
+ tokenList.SetData(new List(Win32Registry.GetAuthTokenHosts()));
+ }
+ return true;
+ });
+ }
+
+ private bool openGitHubURL()
+ {
+ ModInfoScreen.LaunchURL(githubTokenURL);
+ return true;
+ }
+
+ private ConsoleListBox tokenList;
+
+ private const string missingTokenValue = "";
+ private static readonly Uri githubTokenURL = new Uri("https://github.com/settings/tokens");
+ }
+
+}
diff --git a/ConsoleUI/CKAN-ConsoleUI.csproj b/ConsoleUI/CKAN-ConsoleUI.csproj
index 5483d77a78..5d719bed9f 100644
--- a/ConsoleUI/CKAN-ConsoleUI.csproj
+++ b/ConsoleUI/CKAN-ConsoleUI.csproj
@@ -64,6 +64,8 @@
Properties\GlobalAssemblyInfo.cs
+
+
diff --git a/ConsoleUI/ModInfoScreen.cs b/ConsoleUI/ModInfoScreen.cs
index e2fbb4f19d..2742dd1802 100644
--- a/ConsoleUI/ModInfoScreen.cs
+++ b/ConsoleUI/ModInfoScreen.cs
@@ -179,7 +179,14 @@ private bool ViewMetadata()
return true;
}
- private bool LaunchURL(Uri u)
+ ///
+ /// Launch a URL in the system browser.
+ ///
+ /// URL to launch
+ ///
+ /// True.
+ ///
+ public static bool LaunchURL(Uri u)
{
// I'm getting error output on Linux, because this runs xdg-open which
// calls chromium-browser which prints a bunch of stuff about plugins that
diff --git a/ConsoleUI/ModListScreen.cs b/ConsoleUI/ModListScreen.cs
index a2b70444de..ae6fe10635 100644
--- a/ConsoleUI/ModListScreen.cs
+++ b/ConsoleUI/ModListScreen.cs
@@ -256,6 +256,9 @@ public ModListScreen(KSPManager mgr, bool dbg)
new ConsoleMenuOption("Select KSP install...", "",
"Switch to a different game instance",
true, SelectInstall),
+ new ConsoleMenuOption("Authentication tokens...", "",
+ "Edit authentication tokens sent to download servers",
+ true, EditAuthTokens),
null,
new ConsoleMenuOption("Help", helpKey,
"Tips & tricks",
@@ -436,6 +439,12 @@ private bool SelectInstall()
return true;
}
+ private bool EditAuthTokens()
+ {
+ LaunchSubScreen(new AuthTokenScreen());
+ return true;
+ }
+
private void RefreshList()
{
moduleList.SetData(GetAllMods(true));
diff --git a/Core/Net/AutoUpdate.cs b/Core/Net/AutoUpdate.cs
index 38623e30f4..270710f3cf 100644
--- a/Core/Net/AutoUpdate.cs
+++ b/Core/Net/AutoUpdate.cs
@@ -63,8 +63,10 @@ private static bool CanWrite(string path)
///
/// Report whether it's possible to run the auto-updater.
/// Checks whether we can overwrite the running ckan.exe.
+ /// Windows doesn't let us check this because it locks the EXE
+ /// for a running process, so assume we can always overwrite on Windows.
///
- public static readonly bool CanUpdate = CanWrite(exePath);
+ public static readonly bool CanUpdate = Platform.IsWindows || CanWrite(exePath);
///
/// Our metadata is considered fetched if we have a latest version, release notes,
@@ -86,12 +88,15 @@ public void FetchLatestReleaseInfo()
try
{
- fetchedCkanUrl = RetrieveUrl(response, 0);
// Check whether the release includes the auto updater
foreach (var asset in response.assets)
{
string url = asset.browser_download_url.ToString();
- if (url.EndsWith("AutoUpdater.exe"))
+ if (url.EndsWith("ckan.exe"))
+ {
+ fetchedCkanUrl = new Tuple(new Uri(url), (long)asset.size);
+ }
+ else if (url.EndsWith("AutoUpdater.exe"))
{
fetchedUpdaterUrl = new Tuple(new Uri(url), (long)asset.size);
break;
diff --git a/Core/Net/NetAsyncDownloader.cs b/Core/Net/NetAsyncDownloader.cs
index 13d14afd3d..ae6bc963d1 100644
--- a/Core/Net/NetAsyncDownloader.cs
+++ b/Core/Net/NetAsyncDownloader.cs
@@ -54,7 +54,7 @@ public NetAsyncDownloaderDownloadPart(Net.DownloadTarget target, string path = n
{
log.InfoFormat("Using auth token for {0}", this.url.Host);
// Send our auth token to the GitHub API (or whoever else needs one)
- agent.Headers.Add("Authentication", $"token {token}");
+ agent.Headers.Add("Authorization", $"token {token}");
}
}
}
diff --git a/Core/Win32Registry.cs b/Core/Win32Registry.cs
index 42f5677796..334550b2df 100644
--- a/Core/Win32Registry.cs
+++ b/Core/Win32Registry.cs
@@ -110,11 +110,22 @@ public static IEnumerable GetAuthTokenHosts()
/// Set an auth token in the registry
///
/// Host for which to set the token
- /// Token to set
+ /// Token to set, or null to delete
public static void SetAuthToken(string host, string token)
{
ConstructKey(authTokenKeyNoPrefix);
- Microsoft.Win32.Registry.SetValue(authTokenKey, host, token);
+ if (!string.IsNullOrEmpty(host))
+ {
+ if (string.IsNullOrEmpty(token))
+ {
+ RegistryKey key = Microsoft.Win32.Registry.CurrentUser.OpenSubKey(authTokenKeyNoPrefix, true);
+ key.DeleteValue(host);
+ }
+ else
+ {
+ Microsoft.Win32.Registry.SetValue(authTokenKey, host, token);
+ }
+ }
}
private static string StripPrefixKey(string keyname)
diff --git a/GUI/SettingsDialog.Designer.cs b/GUI/SettingsDialog.Designer.cs
index de51834fb6..c022fa3f77 100644
--- a/GUI/SettingsDialog.Designer.cs
+++ b/GUI/SettingsDialog.Designer.cs
@@ -35,30 +35,35 @@ private void InitializeComponent()
this.UpRepoButton = new System.Windows.Forms.Button();
this.DeleteRepoButton = new System.Windows.Forms.Button();
this.ReposListBox = new System.Windows.Forms.ListBox();
+ this.AuthTokensGroupBox = new System.Windows.Forms.GroupBox();
+ this.AuthTokensListBox = new System.Windows.Forms.ListBox();
+ this.NewAuthTokenButton = new System.Windows.Forms.Button();
+ this.DeleteAuthTokenButton = new System.Windows.Forms.Button();
this.CacheGroupBox = new System.Windows.Forms.GroupBox();
this.ClearCKANCacheButton = new System.Windows.Forms.Button();
this.CKANCacheLabel = new System.Windows.Forms.Label();
- this.groupBox2 = new System.Windows.Forms.GroupBox();
+ this.AutoUpdateGroupBox = new System.Windows.Forms.GroupBox();
this.HideEpochsCheckbox = new System.Windows.Forms.CheckBox();
this.RefreshOnStartupCheckbox = new System.Windows.Forms.CheckBox();
this.CheckUpdateOnLaunchCheckbox = new System.Windows.Forms.CheckBox();
this.InstallUpdateButton = new System.Windows.Forms.Button();
this.LatestVersionLabel = new System.Windows.Forms.Label();
- this.label4 = new System.Windows.Forms.Label();
+ this.LatestVersionLabelLabel = new System.Windows.Forms.Label();
this.LocalVersionLabel = new System.Windows.Forms.Label();
- this.label3 = new System.Windows.Forms.Label();
+ this.LocalVersionLabelLabel = new System.Windows.Forms.Label();
this.CheckForUpdatesButton = new System.Windows.Forms.Button();
this.RepositoryGroupBox.SuspendLayout();
+ this.AuthTokensGroupBox.SuspendLayout();
this.CacheGroupBox.SuspendLayout();
- this.groupBox2.SuspendLayout();
+ this.AutoUpdateGroupBox.SuspendLayout();
this.SuspendLayout();
//
// NewRepoButton
//
this.NewRepoButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
- this.NewRepoButton.Location = new System.Drawing.Point(6, 224);
+ this.NewRepoButton.Location = new System.Drawing.Point(6, 94);
this.NewRepoButton.Name = "NewRepoButton";
- this.NewRepoButton.Size = new System.Drawing.Size(56, 26);
+ this.NewRepoButton.Size = new System.Drawing.Size(56, 23);
this.NewRepoButton.TabIndex = 6;
this.NewRepoButton.Text = "New";
this.NewRepoButton.UseVisualStyleBackColor = true;
@@ -74,18 +79,18 @@ private void InitializeComponent()
this.RepositoryGroupBox.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
this.RepositoryGroupBox.Location = new System.Drawing.Point(12, 12);
this.RepositoryGroupBox.Name = "RepositoryGroupBox";
- this.RepositoryGroupBox.Size = new System.Drawing.Size(476, 261);
+ this.RepositoryGroupBox.Size = new System.Drawing.Size(476, 127);
this.RepositoryGroupBox.TabIndex = 12;
this.RepositoryGroupBox.TabStop = false;
- this.RepositoryGroupBox.Text = "Metadata repositories";
+ this.RepositoryGroupBox.Text = "Metadata Repositories";
//
// DownRepoButton
//
this.DownRepoButton.Enabled = false;
this.DownRepoButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
- this.DownRepoButton.Location = new System.Drawing.Point(130, 224);
+ this.DownRepoButton.Location = new System.Drawing.Point(130, 94);
this.DownRepoButton.Name = "DownRepoButton";
- this.DownRepoButton.Size = new System.Drawing.Size(56, 26);
+ this.DownRepoButton.Size = new System.Drawing.Size(56, 23);
this.DownRepoButton.TabIndex = 11;
this.DownRepoButton.Text = "Down";
this.DownRepoButton.UseVisualStyleBackColor = true;
@@ -95,9 +100,9 @@ private void InitializeComponent()
//
this.UpRepoButton.Enabled = false;
this.UpRepoButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
- this.UpRepoButton.Location = new System.Drawing.Point(68, 224);
+ this.UpRepoButton.Location = new System.Drawing.Point(68, 94);
this.UpRepoButton.Name = "UpRepoButton";
- this.UpRepoButton.Size = new System.Drawing.Size(56, 26);
+ this.UpRepoButton.Size = new System.Drawing.Size(56, 23);
this.UpRepoButton.TabIndex = 10;
this.UpRepoButton.Text = "Up";
this.UpRepoButton.UseVisualStyleBackColor = true;
@@ -107,9 +112,9 @@ private void InitializeComponent()
//
this.DeleteRepoButton.Enabled = false;
this.DeleteRepoButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
- this.DeleteRepoButton.Location = new System.Drawing.Point(414, 224);
+ this.DeleteRepoButton.Location = new System.Drawing.Point(414, 94);
this.DeleteRepoButton.Name = "DeleteRepoButton";
- this.DeleteRepoButton.Size = new System.Drawing.Size(56, 26);
+ this.DeleteRepoButton.Size = new System.Drawing.Size(56, 23);
this.DeleteRepoButton.TabIndex = 9;
this.DeleteRepoButton.Text = "Delete";
this.DeleteRepoButton.UseVisualStyleBackColor = true;
@@ -120,10 +125,55 @@ private void InitializeComponent()
this.ReposListBox.FormattingEnabled = true;
this.ReposListBox.Location = new System.Drawing.Point(6, 19);
this.ReposListBox.Name = "ReposListBox";
- this.ReposListBox.Size = new System.Drawing.Size(464, 199);
+ this.ReposListBox.Size = new System.Drawing.Size(464, 72);
this.ReposListBox.TabIndex = 9;
this.ReposListBox.SelectedIndexChanged += new System.EventHandler(this.ReposListBox_SelectedIndexChanged);
//
+ // AuthTokensGroupBox
+ //
+ this.AuthTokensGroupBox.Controls.Add(this.AuthTokensListBox);
+ this.AuthTokensGroupBox.Controls.Add(this.NewAuthTokenButton);
+ this.AuthTokensGroupBox.Controls.Add(this.DeleteAuthTokenButton);
+ this.AuthTokensGroupBox.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
+ this.AuthTokensGroupBox.Location = new System.Drawing.Point(12, 145);
+ this.AuthTokensGroupBox.Name = "AuthTokensGroupBox";
+ this.AuthTokensGroupBox.Size = new System.Drawing.Size(476, 127);
+ this.AuthTokensGroupBox.TabIndex = 13;
+ this.AuthTokensGroupBox.TabStop = false;
+ this.AuthTokensGroupBox.Text = "Authentication Tokens";
+ //
+ // AuthTokensListBox
+ //
+ this.AuthTokensListBox.FormattingEnabled = true;
+ this.AuthTokensListBox.Location = new System.Drawing.Point(6, 19);
+ this.AuthTokensListBox.Name = "AuthTokensListBox";
+ this.AuthTokensListBox.Size = new System.Drawing.Size(464, 72);
+ this.AuthTokensListBox.TabIndex = 14;
+ this.AuthTokensListBox.SelectedIndexChanged += new System.EventHandler(this.AuthTokensListBox_SelectedIndexChanged);
+ //
+ // NewAuthTokenButton
+ //
+ this.NewAuthTokenButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
+ this.NewAuthTokenButton.Location = new System.Drawing.Point(6, 94);
+ this.NewAuthTokenButton.Name = "NewAuthTokenButton";
+ this.NewAuthTokenButton.Size = new System.Drawing.Size(56, 23);
+ this.NewAuthTokenButton.TabIndex = 15;
+ this.NewAuthTokenButton.Text = "New";
+ this.NewAuthTokenButton.UseVisualStyleBackColor = true;
+ this.NewAuthTokenButton.Click += new System.EventHandler(this.NewAuthTokenButton_Click);
+ //
+ // DeleteAuthTokenButton
+ //
+ this.DeleteAuthTokenButton.Enabled = false;
+ this.DeleteAuthTokenButton.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
+ this.DeleteAuthTokenButton.Location = new System.Drawing.Point(414, 94);
+ this.DeleteAuthTokenButton.Name = "DeleteAuthTokenButton";
+ this.DeleteAuthTokenButton.Size = new System.Drawing.Size(56, 23);
+ this.DeleteAuthTokenButton.TabIndex = 16;
+ this.DeleteAuthTokenButton.Text = "Delete";
+ this.DeleteAuthTokenButton.UseVisualStyleBackColor = true;
+ this.DeleteAuthTokenButton.Click += new System.EventHandler(this.DeleteAuthTokenButton_Click);
+ //
// CacheGroupBox
//
this.CacheGroupBox.Controls.Add(this.ClearCKANCacheButton);
@@ -134,7 +184,7 @@ private void InitializeComponent()
this.CacheGroupBox.Size = new System.Drawing.Size(476, 49);
this.CacheGroupBox.TabIndex = 10;
this.CacheGroupBox.TabStop = false;
- this.CacheGroupBox.Text = "CKAN Cache";
+ this.CacheGroupBox.Text = "Cache";
//
// ClearCKANCacheButton
//
@@ -156,27 +206,27 @@ private void InitializeComponent()
this.CKANCacheLabel.TabIndex = 0;
this.CKANCacheLabel.Text = "There are currently N files in the cache, taking up M MB";
//
- // groupBox2
+ // AutoUpdateGroupBox
//
- this.groupBox2.Controls.Add(this.HideEpochsCheckbox);
- this.groupBox2.Controls.Add(this.RefreshOnStartupCheckbox);
+ this.AutoUpdateGroupBox.Controls.Add(this.HideEpochsCheckbox);
+ this.AutoUpdateGroupBox.Controls.Add(this.RefreshOnStartupCheckbox);
if (AutoUpdate.CanUpdate)
{
- this.groupBox2.Controls.Add(this.CheckUpdateOnLaunchCheckbox);
- this.groupBox2.Controls.Add(this.InstallUpdateButton);
- this.groupBox2.Controls.Add(this.LatestVersionLabel);
- this.groupBox2.Controls.Add(this.label4);
- this.groupBox2.Controls.Add(this.LocalVersionLabel);
- this.groupBox2.Controls.Add(this.label3);
- this.groupBox2.Controls.Add(this.CheckForUpdatesButton);
+ this.AutoUpdateGroupBox.Controls.Add(this.CheckUpdateOnLaunchCheckbox);
+ this.AutoUpdateGroupBox.Controls.Add(this.InstallUpdateButton);
+ this.AutoUpdateGroupBox.Controls.Add(this.LatestVersionLabel);
+ this.AutoUpdateGroupBox.Controls.Add(this.LatestVersionLabelLabel);
+ this.AutoUpdateGroupBox.Controls.Add(this.LocalVersionLabel);
+ this.AutoUpdateGroupBox.Controls.Add(this.LocalVersionLabelLabel);
+ this.AutoUpdateGroupBox.Controls.Add(this.CheckForUpdatesButton);
}
- this.groupBox2.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
- this.groupBox2.Location = new System.Drawing.Point(12, 334);
- this.groupBox2.Name = "groupBox2";
- this.groupBox2.Size = new System.Drawing.Size(476, 105);
- this.groupBox2.TabIndex = 11;
- this.groupBox2.TabStop = false;
- this.groupBox2.Text = "CKAN auto-update";
+ this.AutoUpdateGroupBox.FlatStyle = System.Windows.Forms.FlatStyle.Flat;
+ this.AutoUpdateGroupBox.Location = new System.Drawing.Point(12, 334);
+ this.AutoUpdateGroupBox.Name = "AutoUpdateGroupBox";
+ this.AutoUpdateGroupBox.Size = new System.Drawing.Size(476, 105);
+ this.AutoUpdateGroupBox.TabIndex = 11;
+ this.AutoUpdateGroupBox.TabStop = false;
+ this.AutoUpdateGroupBox.Text = "Auto-Updates";
//
// HideEpochsCheckbox
//
@@ -232,14 +282,14 @@ private void InitializeComponent()
this.LatestVersionLabel.TabIndex = 4;
this.LatestVersionLabel.Text = "???";
//
- // label4
+ // LatestVersionLabelLabel
//
- this.label4.AutoSize = true;
- this.label4.Location = new System.Drawing.Point(7, 43);
- this.label4.Name = "label4";
- this.label4.Size = new System.Drawing.Size(76, 13);
- this.label4.TabIndex = 3;
- this.label4.Text = "Latest version:";
+ this.LatestVersionLabelLabel.AutoSize = true;
+ this.LatestVersionLabelLabel.Location = new System.Drawing.Point(7, 43);
+ this.LatestVersionLabelLabel.Name = "LatestVersionLabelLabel";
+ this.LatestVersionLabelLabel.Size = new System.Drawing.Size(76, 13);
+ this.LatestVersionLabelLabel.TabIndex = 3;
+ this.LatestVersionLabelLabel.Text = "Latest version:";
//
// LocalVersionLabel
//
@@ -250,14 +300,14 @@ private void InitializeComponent()
this.LocalVersionLabel.TabIndex = 2;
this.LocalVersionLabel.Text = "v0.0.0";
//
- // label3
+ // LocalVersionLabelLabel
//
- this.label3.AutoSize = true;
- this.label3.Location = new System.Drawing.Point(7, 20);
- this.label3.Name = "label3";
- this.label3.Size = new System.Drawing.Size(73, 13);
- this.label3.TabIndex = 1;
- this.label3.Text = "Local version:";
+ this.LocalVersionLabelLabel.AutoSize = true;
+ this.LocalVersionLabelLabel.Location = new System.Drawing.Point(7, 20);
+ this.LocalVersionLabelLabel.Name = "LocalVersionLabelLabel";
+ this.LocalVersionLabelLabel.Size = new System.Drawing.Size(73, 13);
+ this.LocalVersionLabelLabel.TabIndex = 1;
+ this.LocalVersionLabelLabel.Text = "Local version:";
//
// CheckForUpdatesButton
//
@@ -275,19 +325,21 @@ private void InitializeComponent()
this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F);
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
this.ClientSize = new System.Drawing.Size(495, 442);
- this.Controls.Add(this.groupBox2);
+ this.Controls.Add(this.AutoUpdateGroupBox);
this.Controls.Add(this.CacheGroupBox);
this.Controls.Add(this.RepositoryGroupBox);
+ this.Controls.Add(this.AuthTokensGroupBox);
this.FormBorderStyle = System.Windows.Forms.FormBorderStyle.FixedDialog;
this.Icon = ((System.Drawing.Icon)(resources.GetObject("$this.Icon")));
this.Name = "SettingsDialog";
this.Text = "Settings";
this.Load += new System.EventHandler(this.SettingsDialog_Load);
this.RepositoryGroupBox.ResumeLayout(false);
+ this.AuthTokensGroupBox.ResumeLayout(false);
this.CacheGroupBox.ResumeLayout(false);
this.CacheGroupBox.PerformLayout();
- this.groupBox2.ResumeLayout(false);
- this.groupBox2.PerformLayout();
+ this.AutoUpdateGroupBox.ResumeLayout(false);
+ this.AutoUpdateGroupBox.PerformLayout();
this.ResumeLayout(false);
}
@@ -303,11 +355,15 @@ private void InitializeComponent()
private System.Windows.Forms.ListBox ReposListBox;
private System.Windows.Forms.Button UpRepoButton;
private System.Windows.Forms.Button DownRepoButton;
- private System.Windows.Forms.GroupBox groupBox2;
+ private System.Windows.Forms.GroupBox AuthTokensGroupBox;
+ private System.Windows.Forms.ListBox AuthTokensListBox;
+ private System.Windows.Forms.Button NewAuthTokenButton;
+ private System.Windows.Forms.Button DeleteAuthTokenButton;
+ private System.Windows.Forms.GroupBox AutoUpdateGroupBox;
private System.Windows.Forms.Label LatestVersionLabel;
- private System.Windows.Forms.Label label4;
+ private System.Windows.Forms.Label LatestVersionLabelLabel;
private System.Windows.Forms.Label LocalVersionLabel;
- private System.Windows.Forms.Label label3;
+ private System.Windows.Forms.Label LocalVersionLabelLabel;
private System.Windows.Forms.Button CheckForUpdatesButton;
private System.Windows.Forms.Button InstallUpdateButton;
private System.Windows.Forms.CheckBox CheckUpdateOnLaunchCheckbox;
diff --git a/GUI/SettingsDialog.cs b/GUI/SettingsDialog.cs
index 60d958bade..6bb4d61580 100644
--- a/GUI/SettingsDialog.cs
+++ b/GUI/SettingsDialog.cs
@@ -2,6 +2,7 @@
using System.Collections.Generic;
using System.IO;
using System.Windows.Forms;
+using System.Drawing;
using log4net;
namespace CKAN
@@ -29,6 +30,7 @@ private void SettingsDialog_Load(object sender, EventArgs e)
public void UpdateDialog()
{
RefreshReposListBox();
+ RefreshAuthTokensListBox();
LocalVersionLabel.Text = Meta.GetVersion();
@@ -225,6 +227,149 @@ private void DownRepoButton_Click(object sender, EventArgs e)
RefreshReposListBox();
}
+ private void RefreshAuthTokensListBox()
+ {
+ AuthTokensListBox.Items.Clear();
+ foreach (string host in Win32Registry.GetAuthTokenHosts())
+ {
+ string token;
+ if (Win32Registry.TryGetAuthToken(host, out token))
+ {
+ AuthTokensListBox.Items.Add(string.Format("{0} | {1}", host, token));
+ }
+ }
+ }
+
+ private void AuthTokensListBox_SelectedIndexChanged(object sender, EventArgs e)
+ {
+ DeleteAuthTokenButton.Enabled = AuthTokensListBox.SelectedItem != null;
+ }
+
+ private void NewAuthTokenButton_Click(object sender, EventArgs e)
+ {
+ // Inspired by https://stackoverflow.com/a/17546909/2422988
+ Form newAuthTokenPopup = new Form()
+ {
+ FormBorderStyle = FormBorderStyle.FixedToolWindow,
+ StartPosition = FormStartPosition.CenterParent,
+ ClientSize = new Size(300, 100),
+ Text = "Add Authentication Token"
+ };
+ Label hostLabel = new Label()
+ {
+ AutoSize = true,
+ Location = new Point(3, 6),
+ Size = new Size(271, 13),
+ Text = "Host:"
+ };
+ TextBox hostTextBox = new TextBox()
+ {
+ Location = new Point(45, 6),
+ Size = new Size(newAuthTokenPopup.ClientSize.Width - 40 - 10, 23),
+ Text = ""
+ };
+ Label tokenLabel = new Label()
+ {
+ AutoSize = true,
+ Location = new Point(3, 35),
+ Size = new Size(271, 13),
+ Text = "Token:"
+ };
+ TextBox tokenTextBox = new TextBox()
+ {
+ Location = new Point(45, 35),
+ Size = new Size(newAuthTokenPopup.ClientSize.Width - 40 - 10, 23),
+ Text = ""
+ };
+ Button acceptButton = new Button()
+ {
+ DialogResult = DialogResult.OK,
+ Name = "okButton",
+ Size = new Size(75, 23),
+ Text = "&Accept",
+ Location = new Point((newAuthTokenPopup.ClientSize.Width - 80 - 80) / 2, 64)
+ };
+ acceptButton.Click += (origin, evt) =>
+ {
+ newAuthTokenPopup.DialogResult = validNewAuthToken(hostTextBox.Text, tokenTextBox.Text)
+ ? DialogResult.OK
+ : DialogResult.None;
+ };
+ Button cancelButton = new Button()
+ {
+ DialogResult = DialogResult.Cancel,
+ Name = "cancelButton",
+ Size = new Size(75, 23),
+ Text = "&Cancel",
+ Location = new Point(acceptButton.Location.X + acceptButton.Size.Width + 5, 64)
+ };
+
+ newAuthTokenPopup.Controls.Add(hostLabel);
+ newAuthTokenPopup.Controls.Add(hostTextBox);
+ newAuthTokenPopup.Controls.Add(tokenLabel);
+ newAuthTokenPopup.Controls.Add(tokenTextBox);
+ newAuthTokenPopup.Controls.Add(acceptButton);
+ newAuthTokenPopup.Controls.Add(cancelButton);
+ newAuthTokenPopup.AcceptButton = acceptButton;
+ newAuthTokenPopup.CancelButton = cancelButton;
+
+ switch (newAuthTokenPopup.ShowDialog(this))
+ {
+ case DialogResult.Abort:
+ case DialogResult.Cancel:
+ case DialogResult.Ignore:
+ case DialogResult.No:
+ // User cancelled out, so do nothing
+ break;
+
+ case DialogResult.OK:
+ case DialogResult.Yes:
+ Win32Registry.SetAuthToken(hostTextBox.Text, tokenTextBox.Text);
+ RefreshAuthTokensListBox();
+ break;
+ }
+ }
+
+ private static bool validNewAuthToken(string host, string token)
+ {
+ if (host.Length <= 0)
+ {
+ GUI.user.RaiseError("Host field is required.");
+ return false;
+ }
+ if (token.Length <= 0)
+ {
+ GUI.user.RaiseError("Token field is required.");
+ return false;
+ }
+ if (Uri.CheckHostName(host) == UriHostNameType.Unknown)
+ {
+ GUI.user.RaiseError("{0} is not a valid host name.", host);
+ return false;
+ }
+ string oldToken;
+ if (Win32Registry.TryGetAuthToken(host, out oldToken))
+ {
+ GUI.user.RaiseError("{0} already has an authentication token.", host);
+ return false;
+ }
+
+ return true;
+ }
+
+ private void DeleteAuthTokenButton_Click(object sender, EventArgs e)
+ {
+ if (AuthTokensListBox.SelectedItem != null)
+ {
+ string item = AuthTokensListBox.SelectedItem as string;
+ string host = item?.Split('|')[0].Trim();
+
+ Win32Registry.SetAuthToken(host, null);
+ RefreshAuthTokensListBox();
+ DeleteRepoButton.Enabled = false;
+ }
+ }
+
private void CheckForUpdatesButton_Click(object sender, EventArgs e)
{
try