diff options
author | Kenneth Skovhede <kenneth@hexad.dk> | 2015-12-01 15:24:37 +0300 |
---|---|---|
committer | Kenneth Skovhede <kenneth@hexad.dk> | 2015-12-01 15:24:37 +0300 |
commit | 8f23061496d51200bcc1466f9bcdb3fb816fdf78 (patch) | |
tree | f430818cb44c47bf2c3be8a1074bc8daa77af230 /BuildTools | |
parent | be3fe49a7535b3bc6f70e51ce8e3283229856145 (diff) |
Implemented option to output gpg signature key as part of the build process
Diffstat (limited to 'BuildTools')
-rw-r--r-- | BuildTools/AutoUpdateBuilder/Program.cs | 349 |
1 files changed, 207 insertions, 142 deletions
diff --git a/BuildTools/AutoUpdateBuilder/Program.cs b/BuildTools/AutoUpdateBuilder/Program.cs index 2ddf77ec1..d2df70e61 100644 --- a/BuildTools/AutoUpdateBuilder/Program.cs +++ b/BuildTools/AutoUpdateBuilder/Program.cs @@ -1,143 +1,208 @@ -using System;
-using System.Collections.Generic;
-
-namespace AutoUpdateBuilder
-{
- public class Program
- {
- public static int Main(string[] _args)
- {
- var args = new List<string>(_args);
- var opts = Duplicati.Library.Utility.CommandLineParser.ExtractOptions(args);
-
- string inputfolder;
- string outputfolder;
- string keyfile;
- string manifestfile;
- string keyfilepassword;
-
- opts.TryGetValue("input", out inputfolder);
- opts.TryGetValue("output", out outputfolder);
- opts.TryGetValue("keyfile", out keyfile);
- opts.TryGetValue("manifest", out manifestfile);
- opts.TryGetValue("keyfile-password", out keyfilepassword);
-
- var usedoptions = new string[] { "input", "output", "keyfile", "manifest", "keyfile-password" };
-
- if (string.IsNullOrWhiteSpace(inputfolder))
- {
- Console.WriteLine("Missing input folder");
- return 4;
- }
-
- if (string.IsNullOrWhiteSpace(outputfolder))
- {
- Console.WriteLine("Missing output folder");
- return 4;
- }
-
- if (string.IsNullOrWhiteSpace(keyfile))
- {
- Console.WriteLine("Missing keyfile");
- return 4;
- }
-
- if (!System.IO.Directory.Exists(inputfolder))
- {
- Console.WriteLine("Input folder not found");
- return 4;
- }
-
- if (string.IsNullOrWhiteSpace(keyfilepassword))
- {
- Console.WriteLine("Enter keyfile passphrase: ");
- keyfilepassword = Console.ReadLine().Trim();
- }
-
- if (!System.IO.File.Exists(keyfile))
- {
- Console.WriteLine("Keyfile not found, creating new");
- var newkey = System.Security.Cryptography.RSACryptoServiceProvider.Create().ToXmlString(true);
- using (var enc = new Duplicati.Library.Encryption.AESEncryption(keyfilepassword, new Dictionary<string, string>()))
- using (var fs = System.IO.File.OpenWrite(keyfile))
- using (var ms = new System.IO.MemoryStream(System.Text.Encoding.UTF8.GetBytes(newkey)))
- enc.Encrypt(ms, fs);
- }
-
- if (!System.IO.Directory.Exists(outputfolder))
- System.IO.Directory.CreateDirectory(outputfolder);
-
- var privkey = (System.Security.Cryptography.RSACryptoServiceProvider)System.Security.Cryptography.RSACryptoServiceProvider.Create();
-
- using(var enc = new Duplicati.Library.Encryption.AESEncryption(keyfilepassword, new Dictionary<string, string>()))
- using(var ms = new System.IO.MemoryStream())
- using(var fs = System.IO.File.OpenRead(keyfile))
- {
- enc.Decrypt(fs, ms);
- ms.Position = 0;
-
- using(var sr = new System.IO.StreamReader(ms))
- privkey.FromXmlString(sr.ReadToEnd());
- }
-
- if (Duplicati.Library.AutoUpdater.AutoUpdateSettings.SignKey == null || privkey.ToXmlString(false) != Duplicati.Library.AutoUpdater.AutoUpdateSettings.SignKey.ToXmlString(false))
- {
- Console.WriteLine("The public key in the project is not the same as the public key from the file");
- Console.WriteLine("Try setting the key to: ");
- Console.WriteLine(privkey.ToXmlString(false));
- return 5;
- }
-
- Duplicati.Library.AutoUpdater.UpdateInfo updateInfo;
-
- using (var fs = System.IO.File.OpenRead(manifestfile))
- using (var sr = new System.IO.StreamReader(fs))
- using (var jr = new Newtonsoft.Json.JsonTextReader(sr))
- updateInfo = new Newtonsoft.Json.JsonSerializer().Deserialize<Duplicati.Library.AutoUpdater.UpdateInfo>(jr);
-
-
- var isopts = new Dictionary<string, string>(opts, StringComparer.InvariantCultureIgnoreCase);
- foreach (var usedopt in usedoptions)
- isopts.Remove(usedopt);
-
- foreach (var k in updateInfo.GetType().GetFields())
- if (isopts.ContainsKey(k.Name))
- {
- try
- {
- //Console.WriteLine("Setting {0} to {1}", k.Name, isopts[k.Name]);
- if (k.FieldType == typeof(string[]))
- k.SetValue(updateInfo, isopts[k.Name].Split(new char[] { ';' }, StringSplitOptions.RemoveEmptyEntries));
- else if (k.FieldType == typeof(Version))
- k.SetValue(updateInfo, new Version(isopts[k.Name]));
- else if (k.FieldType == typeof(int))
- k.SetValue(updateInfo, int.Parse(isopts[k.Name]));
- else if (k.FieldType == typeof(long))
- k.SetValue(updateInfo, long.Parse(isopts[k.Name]));
- else
- k.SetValue(updateInfo, isopts[k.Name]);
- }
- catch (Exception ex)
- {
- Console.WriteLine("Failed setting {0} to {1}: {2}", k.Name, isopts[k.Name], ex.Message);
- }
-
- isopts.Remove(k.Name);
- }
-
- foreach(var opt in isopts)
- Console.WriteLine("Warning! unused option: {0} = {1}", opt.Key, opt.Value);
-
- using (var tf = new Duplicati.Library.Utility.TempFile())
- {
- using (var fs = System.IO.File.OpenWrite(tf))
- using (var tw = new System.IO.StreamWriter(fs))
- new Newtonsoft.Json.JsonSerializer().Serialize(tw, updateInfo);
-
- Duplicati.Library.AutoUpdater.UpdaterManager.CreateUpdatePackage(privkey, inputfolder, outputfolder, tf);
- }
-
- return 0;
- }
- }
+using System; +using System.Collections.Generic; + +namespace AutoUpdateBuilder +{ + public class Program + { + public static int Main(string[] _args) + { + var args = new List<string>(_args); + var opts = Duplicati.Library.Utility.CommandLineParser.ExtractOptions(args); + + string inputfolder; + string outputfolder; + string keyfile; + string manifestfile; + string keyfilepassword; + string gpgkeyfile; + string gpgpath; + + opts.TryGetValue("input", out inputfolder); + opts.TryGetValue("output", out outputfolder); + opts.TryGetValue("keyfile", out keyfile); + opts.TryGetValue("manifest", out manifestfile); + opts.TryGetValue("keyfile-password", out keyfilepassword); + opts.TryGetValue("gpgkeyfile", out gpgkeyfile); + opts.TryGetValue("gpgpath", out gpgpath); + + var usedoptions = new string[] { "input", "output", "keyfile", "manifest", "keyfile-password", "gpgkeyfile", "gpgpath" }; + + if (string.IsNullOrWhiteSpace(inputfolder)) + { + Console.WriteLine("Missing input folder"); + return 4; + } + + if (string.IsNullOrWhiteSpace(outputfolder)) + { + Console.WriteLine("Missing output folder"); + return 4; + } + + if (string.IsNullOrWhiteSpace(keyfile)) + { + Console.WriteLine("Missing keyfile"); + return 4; + } + + if (!System.IO.Directory.Exists(inputfolder)) + { + Console.WriteLine("Input folder not found"); + return 4; + } + + if (string.IsNullOrWhiteSpace(keyfilepassword)) + { + Console.WriteLine("Enter keyfile passphrase: "); + keyfilepassword = Console.ReadLine().Trim(); + } + + if (!System.IO.File.Exists(keyfile)) + { + Console.WriteLine("Keyfile not found, creating new"); + var newkey = System.Security.Cryptography.RSACryptoServiceProvider.Create().ToXmlString(true); + using (var enc = new Duplicati.Library.Encryption.AESEncryption(keyfilepassword, new Dictionary<string, string>())) + using (var fs = System.IO.File.OpenWrite(keyfile)) + using (var ms = new System.IO.MemoryStream(System.Text.Encoding.UTF8.GetBytes(newkey))) + enc.Encrypt(ms, fs); + } + + if (!System.IO.Directory.Exists(outputfolder)) + System.IO.Directory.CreateDirectory(outputfolder); + + var privkey = (System.Security.Cryptography.RSACryptoServiceProvider)System.Security.Cryptography.RSACryptoServiceProvider.Create(); + + using(var enc = new Duplicati.Library.Encryption.AESEncryption(keyfilepassword, new Dictionary<string, string>())) + using(var ms = new System.IO.MemoryStream()) + using(var fs = System.IO.File.OpenRead(keyfile)) + { + enc.Decrypt(fs, ms); + ms.Position = 0; + + using(var sr = new System.IO.StreamReader(ms)) + privkey.FromXmlString(sr.ReadToEnd()); + } + + if (Duplicati.Library.AutoUpdater.AutoUpdateSettings.SignKey == null || privkey.ToXmlString(false) != Duplicati.Library.AutoUpdater.AutoUpdateSettings.SignKey.ToXmlString(false)) + { + Console.WriteLine("The public key in the project is not the same as the public key from the file"); + Console.WriteLine("Try setting the key to: "); + Console.WriteLine(privkey.ToXmlString(false)); + return 5; + } + + + string gpgkeyid = null; + string gpgkeypassphrase = null; + + if (string.IsNullOrWhiteSpace(gpgkeyfile)) + { + Console.WriteLine("No gpgfile, skipping GPG signature files"); + } + else if (!System.IO.File.Exists(gpgkeyfile)) + { + Console.WriteLine("Missing gpgfile"); + return 6; + } + else + { + using(var enc = new Duplicati.Library.Encryption.AESEncryption(keyfilepassword, new Dictionary<string, string>())) + using(var ms = new System.IO.MemoryStream()) + using(var fs = System.IO.File.OpenRead(gpgkeyfile)) + { + enc.Decrypt(fs, ms); + ms.Position = 0; + + // No real format, just two lines + using (var sr = new System.IO.StreamReader(ms)) + { + var lines = sr.ReadToEnd().Split(new string[] { Environment.NewLine }, StringSplitOptions.RemoveEmptyEntries); + gpgkeyid = lines[0]; + gpgkeypassphrase = lines[1]; + } + } + } + + + Duplicati.Library.AutoUpdater.UpdateInfo updateInfo; + + using (var fs = System.IO.File.OpenRead(manifestfile)) + using (var sr = new System.IO.StreamReader(fs)) + using (var jr = new Newtonsoft.Json.JsonTextReader(sr)) + updateInfo = new Newtonsoft.Json.JsonSerializer().Deserialize<Duplicati.Library.AutoUpdater.UpdateInfo>(jr); + + + var isopts = new Dictionary<string, string>(opts, StringComparer.InvariantCultureIgnoreCase); + foreach (var usedopt in usedoptions) + isopts.Remove(usedopt); + + foreach (var k in updateInfo.GetType().GetFields()) + if (isopts.ContainsKey(k.Name)) + { + try + { + //Console.WriteLine("Setting {0} to {1}", k.Name, isopts[k.Name]); + if (k.FieldType == typeof(string[])) + k.SetValue(updateInfo, isopts[k.Name].Split(new char[] { ';' }, StringSplitOptions.RemoveEmptyEntries)); + else if (k.FieldType == typeof(Version)) + k.SetValue(updateInfo, new Version(isopts[k.Name])); + else if (k.FieldType == typeof(int)) + k.SetValue(updateInfo, int.Parse(isopts[k.Name])); + else if (k.FieldType == typeof(long)) + k.SetValue(updateInfo, long.Parse(isopts[k.Name])); + else + k.SetValue(updateInfo, isopts[k.Name]); + } + catch (Exception ex) + { + Console.WriteLine("Failed setting {0} to {1}: {2}", k.Name, isopts[k.Name], ex.Message); + } + + isopts.Remove(k.Name); + } + + foreach(var opt in isopts) + Console.WriteLine("Warning! unused option: {0} = {1}", opt.Key, opt.Value); + + using (var tf = new Duplicati.Library.Utility.TempFile()) + { + using (var fs = System.IO.File.OpenWrite(tf)) + using (var tw = new System.IO.StreamWriter(fs)) + new Newtonsoft.Json.JsonSerializer().Serialize(tw, updateInfo); + + Duplicati.Library.AutoUpdater.UpdaterManager.CreateUpdatePackage(privkey, inputfolder, outputfolder, tf); + } + + if (gpgkeyid != null) + { + gpgpath = gpgpath ?? "gpg"; + var srcfile = System.IO.Path.Combine(outputfolder, "package.zip"); + + var proc = System.Diagnostics.Process.Start(new System.Diagnostics.ProcessStartInfo() { + FileName = gpgpath, + Arguments = string.Format("--passphrase-fd 0 --batch --yes --default-key={1} --output \"{0}.sig\" --detach-sig \"{0}\"", srcfile, gpgkeyid), + RedirectStandardInput = true, + UseShellExecute = false + }); + + proc.StandardInput.WriteLine(gpgkeypassphrase); + proc.WaitForExit(); + + proc = System.Diagnostics.Process.Start(new System.Diagnostics.ProcessStartInfo() { + FileName = gpgpath, + Arguments = string.Format("--passphrase-fd 0 --batch --yes --default-key={1} --armor --output \"{0}.sig.asc\" --detach-sig \"{0}\"", srcfile, gpgkeyid), + RedirectStandardInput = true, + UseShellExecute = false + }); + + proc.StandardInput.WriteLine(gpgkeypassphrase); + proc.WaitForExit(); + + } + + + return 0; + } + } }
\ No newline at end of file |