123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413 |
- require 'tmpdir'
- require 'open-uri'
- require_relative 'build/TxClient'
- require_relative 'build/CsprojResxWriter'
- ISCC = ENV['ISCC'] || 'C:\Program Files (x86)\Inno Setup 6\ISCC.exe'
- SZIP = ENV['SZIP'] || File.join(__dir__, 'build', '7za.exe')
- SIGNTOOL = ENV['SIGNTOOL'] || 'C:\Program Files (x86)\Microsoft SDKs\Windows\v7.1A\Bin\signtool.exe'
- VSWHERE = 'build/vswhere.exe'
- CONFIG = ENV['CONFIG'] || 'Release'
- MSBUILD_LOGGER = ENV['MSBUILD_LOGGER']
- SRC_DIR = 'src/SyncTrayzor'
- INSTALLER_DIR = 'installer'
- DEPLOY_DIR = 'deploy'
- SLN = 'src/SyncTrayzor.sln'
- CHOCOLATEY_NUSPEC = 'chocolatey/synctrayzor.nuspec'
- CHECKSUM_UTIL_CSPROJ = 'src/ChecksumUtil/ChecksumUtil.csproj'
- CHECKSUM_UTIL_EXE = 'bin/ChecksumUtil/Release/ChecksumUtil.exe'
- SYNCTHING_RELEASES_CERT = 'security/syncthing_releases_cert.asc'
- CHECKSUM_FILE_PRIV_KEY = 'security/private_key.asc'
- PFX = ENV['PFX'] || File.join(INSTALLER_DIR, 'SyncTrayzorCA.pfx')
- PORTABLE_SYNCTHING_VERSION = 'latest'
- class ArchDirConfig
- attr_reader :arch
- attr_reader :bin_dir
- attr_reader :installer_dir
- attr_reader :installer_output
- attr_reader :installer_iss
- attr_reader :portable_output_dir
- attr_reader :portable_output_file
- attr_reader :syncthing_binaries
- def initialize(arch, github_arch)
- @arch = arch
- @github_arch = github_arch
- @bin_dir = "bin/#{@arch}/#{CONFIG}"
- @installer_dir = File.join(INSTALLER_DIR, @arch)
- @installer_output = File.join(@installer_dir, "SyncTrayzorSetup-#{@arch}.exe")
- @installer_iss = File.join(@installer_dir, "installer-#{@arch}.iss")
- @portable_output_dir = "SyncTrayzorPortable-#{@arch}"
- @portable_output_file = File.join(DEPLOY_DIR, "SyncTrayzorPortable-#{@arch}.zip")
- @syncthing_binaries = { 'latest' => 'syncthing.exe' }
- end
- def sha1sum_download_uri(version)
- "https://github.com/syncthing/syncthing/releases/download/v#{version}/sha1sum.txt.asc"
- end
- def download_uri(version)
- "https://github.com/syncthing/syncthing/releases/download/v#{version}/syncthing-windows-#{@github_arch}-v#{version}.zip"
- end
- end
- SYNCTHING_VERSIONS_TO_UPDATE = ['latest']
- ARCH_CONFIG = [ArchDirConfig.new('x64', 'amd64'), ArchDirConfig.new('x86', '386')]
- ASSEMBLY_INFOS = FileList['**/AssemblyInfo.cs']
- def build(sln, platform, rebuild = true)
- if ENV['MSBUILD']
- msbuild = ENV['MSBUILD']
- else
- path = `#{VSWHERE} -requires Microsoft.Component.MSBuild -format value -property installationPath`.chomp
- msbuild = File.join(path, 'MSBuild', 'Current', 'Bin', 'MSBuild.exe')
- end
- puts "MSBuild is at #{msbuild}"
- tasks = rebuild ? 'Clean;Rebuild' : 'Build'
- cmd = "\"#{msbuild}\" \"#{sln}\" -t:#{tasks} -p:Configuration=#{CONFIG};Platform=#{platform}"
- if MSBUILD_LOGGER
- cmd << " -logger:\"#{MSBUILD_LOGGER}\" /verbosity:minimal"
- else
- cmd << " -verbosity:quiet"
- end
-
- puts cmd
- sh cmd
- end
- namespace :build do
- ARCH_CONFIG.each do |arch_config|
- desc "Build the project (#{arch_config.arch})"
- task arch_config.arch do
- build(SLN, arch_config.arch)
- end
- end
- end
- desc 'Build both 64-bit and 32-bit binaries'
- task :build => ARCH_CONFIG.map{ |x| :"build:#{x.arch}" }
- task :"build-checksum-util" do
- build(CHECKSUM_UTIL_CSPROJ, 'AnyCPU', false)
- end
- namespace :installer do
- ARCH_CONFIG.each do |arch_config|
- desc "Create the installer (#{arch_config.arch})"
- task arch_config.arch do
- unless File.exist?(ISCC)
- warn "Please install Inno Setup"
- exit 1
- end
- rm arch_config.installer_output if File.exist?(arch_config.installer_output)
- sh %Q{"#{ISCC}" #{arch_config.installer_iss}}
- mkdir_p DEPLOY_DIR
- mv arch_config.installer_output, DEPLOY_DIR
- end
- end
- end
- desc 'Build both 64-bit and 32-bit installers'
- task :installer => ARCH_CONFIG.map{ |x| :"installer:#{x.arch}" }
- namespace :"sign-installer" do
- ARCH_CONFIG.each do |arch_config|
- desc "Sign the installer (#{arch_config.arch}). Specify PASSWORD if required"
- task arch_config.arch do
- unless File.exist?(SIGNTOOL)
- warn "You must install the Windows SDK"
- exit 1
- end
- unless File.exist?(PFX)
- warn "#{PFX} must exist"
- exit 1
- end
- args = "sign /f #{PFX} /t http://timestamp.verisign.com/scripts/timstamp.dll"
- args << " /p #{ENV['PASSWORD']}" if ENV['PASSWORD']
- args << " /v #{File.join(DEPLOY_DIR, File.basename(arch_config.installer_output))}"
- # Don't want to print out the pasword!
- puts "Invoking signtool"
- system %Q{"#{SIGNTOOL}" #{args}}
- end
- end
- end
- desc 'Sign both 64-bit and 32-bit installers. Specify PASSWORD if required'
- task :"sign-installer" => ARCH_CONFIG.map{ |x| :"sign-installer:#{x.arch}" }
- def cp_to_portable(output_dir, src, output_filename = nil)
- dest = File.join(output_dir, output_filename || src)
- raise "Cannot find #{src}" unless File.exist?(src)
- # It could be an empty directory - so ignore it
- # We'll create it as and when if there are any files in it
- if File.file?(src)
- mkdir_p File.dirname(dest) unless File.exist?(File.dirname(dest))
- cp src, dest
- end
- end
- namespace :portable do
- ARCH_CONFIG.each do |arch_config|
- desc "Create the portable package (#{arch_config.arch})"
- task arch_config.arch do
- mkdir_p File.dirname(arch_config.portable_output_file)
- rm arch_config.portable_output_file if File.exist?(arch_config.portable_output_file)
- Dir.mktmpdir do |tmp|
- portable_dir = File.join(tmp, arch_config.portable_output_dir)
- Dir.chdir(arch_config.bin_dir) do
- files = FileList['**/*'].exclude(
- '*.xml', '*.vshost.*', '*.log', '*.Installer.config', '*/FluentValidation.resources.dll',
- '*/System.Windows.Interactivity.resources.dll', 'syncthing.exe', 'data/*', 'logs',
- 'd3dcompiler_47.dll', 'libEGL.dll', 'libGLESv2.dll', 'swiftshader/libEGL.dll', 'swiftshader/libGLESv2.dll',)
- files.each do |file|
- cp_to_portable(portable_dir, file)
- end
- end
- Dir.chdir(File.join('bin', 'PortableInstaller', CONFIG)) do
- cp_to_portable(portable_dir, 'PortableInstaller.exe')
- end
- cp File.join(SRC_DIR, 'Icons', 'default.ico'), arch_config.portable_output_dir
- FileList['*.md', '*.txt'].each do |file|
- cp_to_portable(portable_dir, file)
- end
-
- Dir.chdir(File.join(arch_config.installer_dir, 'ucrt')) do
- FileList['*.dll'].each do |file|
- cp_to_portable(portable_dir, file)
- end
- end
- Dir.chdir(File.join(arch_config.installer_dir, 'vc++')) do
- FileList['*.dll'].each do |file|
- cp_to_portable(portable_dir, file)
- end
- end
- Dir.chdir(arch_config.installer_dir) do
- cp_to_portable(portable_dir, arch_config.syncthing_binaries[PORTABLE_SYNCTHING_VERSION], 'syncthing.exe')
- end
- sh %Q{"#{SZIP}" a -tzip -mx=7 -y #{arch_config.portable_output_file} #{portable_dir}}
- end
- end
- end
- end
- def create_checksums(checksum_file, password, algorithm, files)
- rm checksum_file if File.exist?(checksum_file)
- args = %Q{create "#{checksum_file}" #{algorithm} "#{CHECKSUM_FILE_PRIV_KEY}" "#{password}" } + files.map{ |x| "\"#{x}\"" }.join(' ')
- # Don't want to print out the pasword!
- puts "Invoking #{CHECKSUM_UTIL_EXE} for #{checksum_file}"
- system %Q{"#{CHECKSUM_UTIL_EXE}" #{args}}
- end
- desc 'Create checksums files'
- task :"create-checksums" => [:"build-checksum-util"] do
- password = ENV['PASSWORD'] || '""'
- checksum_file = File.join(DEPLOY_DIR, 'sha1sum.txt.asc')
- files = Dir["#{DEPLOY_DIR}/*.{zip,exe}"]
- create_checksums(File.join(DEPLOY_DIR, 'sha1sum.txt.asc'), password, 'sha1', files)
- create_checksums(File.join(DEPLOY_DIR, 'sha512sum.txt.asc'), password, 'sha512', files)
- end
- desc 'Clean portable and installer, all architectures'
- task :clean do
- rm_rf DEPLOY_DIR if File.exist?(DEPLOY_DIR)
- end
- namespace :package do
- ARCH_CONFIG.each do |arch_config|
- desc "Build installer and portable (#{arch_config.arch})"
- task arch_config.arch =>
- [
- :"update-syncthing:#{arch_config.arch}",
- :"build:#{arch_config.arch}",
- :"installer:#{arch_config.arch}",
- :"sign-installer:#{arch_config.arch}",
- :"portable:#{arch_config.arch}"
- ]
- end
- end
- desc 'Build installer and portable for all architectures'
- task :package => [:clean, *ARCH_CONFIG.map{ |x| :"package:#{x.arch}" }, :"create-checksums"]
- desc 'Build chocolatey package'
- task :chocolatey do
- chocolatey_dir = File.dirname(CHOCOLATEY_NUSPEC)
- cp Dir[File.join(DEPLOY_DIR, 'SyncTrayzorSetup-*.exe')], File.join(chocolatey_dir, 'tools')
- cp 'LICENSE.txt', File.join(chocolatey_dir, 'tools')
- Dir.chdir(chocolatey_dir) do
- sh "choco pack"
- end
- mv Dir[File.join(chocolatey_dir, 'synctrayzor.*.nupkg')], DEPLOY_DIR
- end
- desc "Bump version number"
- task :version, [:version] do |t, args|
- parts = args[:version].split('.')
- parts4 = parts.dup
- parts4 << '0' if parts4.length == 3
- version4 = parts4.join('.')
- ASSEMBLY_INFOS.each do |info|
- content = IO.read(info)
- content[/^\[assembly: AssemblyVersion\(\"(.+?)\"\)\]/, 1] = version4
- content[/^\[assembly: AssemblyFileVersion\(\"(.+?)\"\)\]/, 1] = version4
- File.open(info, 'w'){ |f| f.write(content) }
- end
- choco_content = IO.read(CHOCOLATEY_NUSPEC)
- choco_content[/<version>(.+?)<\/version>/, 1] = args[:version]
- File.open(CHOCOLATEY_NUSPEC, 'w'){ |f| f.write(choco_content) }
- end
- desc 'Create both 64-bit and 32-bit portable packages'
- task :portable => ARCH_CONFIG.map{ |x| :"portable:#{x.arch}" }
- namespace :syncthing do
- namespace :download do
- ARCH_CONFIG.each do |arch_config|
- desc "Download syncthing (#{arch_config.arch})"
- task arch_config.arch, [:version] => [:"build-checksum-util"] do |t, args|
- Dir.mktmpdir do |tmp|
- download_file = File.join(tmp, File.basename(arch_config.download_uri(args[:version])))
- File.open(download_file, 'wb') do |outfile|
- open(arch_config.download_uri(args[:version])) do |infile|
- outfile.write(infile.read)
- end
- end
- File.open(File.join(tmp, 'sha1sum.txt.asc.'), 'w') do |outfile|
- open(arch_config.sha1sum_download_uri(args[:version])) do |infile|
- outfile.write(infile.read)
- end
- end
- sh CHECKSUM_UTIL_EXE, 'verify', File.join(tmp, 'sha1sum.txt.asc'), 'sha1', SYNCTHING_RELEASES_CERT, download_file
- Dir.chdir(tmp) do
- sh %Q{"#{SZIP}" e -y #{File.basename(download_file)}}
- end
- cp File.join(tmp, 'syncthing.exe'), File.join(arch_config.installer_dir, 'syncthing.exe')
- end
- end
- end
- end
- desc 'Download syncthing for all architectures'
- task :download, [:version] => ARCH_CONFIG.map{ |x| :"syncthing:download:#{x.arch}" }
- namespace :update do
- ARCH_CONFIG.each do |arch_config|
- desc "Update syncthing binaries (#{arch_config.arch}"
- task arch_config.arch do
- arch_config.syncthing_binaries.values_at(*SYNCTHING_VERSIONS_TO_UPDATE).each do |bin|
- path = File.join(arch_config.installer_dir, bin)
- raise "Could not find #{path}" unless File.exist?(path)
- Dir.mktmpdir do |tmp|
- sh path, '-upgrade', "-home=#{tmp}" do; end
- end
- old_bin = "#{path}.old"
- rm old_bin if File.exist?(old_bin)
- end
- end
- end
- end
- desc 'Update syncthing binaries, all architectures'
- task :update => ARCH_CONFIG.map{ |x| :"syncthing:update:#{x.arch}" }
- end
- def create_tx_client
- raise "TX_PASSWORD not specified" if ENV['TX_PASSWORD'].nil? || ENV['TX_PASSWORD'].empty?
- TxClient.new('synctrayzor', 'strings', 'canton7', ENV['TX_PASSWORD'])
- end
- def create_csproj_resx_writer
- csproj_resx_writer = CsprojResxWriter.new('src/SyncTrayzor/SyncTrayzor.csproj', 'Properties')
- csproj_resx_writer.language_exceptions['es_ES'] = 'es'
- csproj_resx_writer
- end
- namespace :tx do
- desc "Remove all translations from csproj"
- task :clean do
- create_csproj_resx_writer().remove_all_resx
- puts "Cleaned translations"
- end
- desc "Fetch all translations"
- task :pull => [:"tx:clean"] do
- tx_client = create_tx_client()
- csproj_resx_writer = create_csproj_resx_writer()
- tx_client.list_translations().each do |language|
- next if language == 'en'
- puts "Fetching #{language}..."
- tx_client.download_translation(language, csproj_resx_writer.absolute_resx_path_for_language(language))
- csproj_resx_writer.add_resx_to_csproj(language)
- end
- end
- desc "Push source translations"
- task :push do
- tx_client = create_tx_client()
- csproj_resx_writer = create_csproj_resx_writer()
- source_resx = csproj_resx_writer.read_and_sort_source_resx
- response = tx_client.upload_source(source_resx)
- puts "Added: #{response['strings_added']}. Updated: #{response['strings_updated']}. Deleted: #{response['strings_delete']}."
- end
- end
- namespace :icons do
- desc "Create a tray icon (pass 256x256 source)"
- task :"tray-icon", [:source] do |t,args|
- source = args[:source]
- # See https://wiki.lazarus.freepascal.org/Windows_Icon
- sizes = [
- 16, 32, # Normal 96 DPI
- 20, 40, # 120 DPI (125%)
- 24, 48, # 144 DPI (150%)
- 32, 64, # 192 DPI (200%)
- 36, 72, # 225%
- 40, 80, # 250%
- 44, 88, # 275%
- 48, 96, # 300%
- ].uniq.sort
- raise "Need a source image" unless source
- Dir.chdir(File.join('src', 'SyncTrayzor', 'Icons')) do
- sizes.each do |size|
- sh 'magick', 'convert', source, '-resize', "#{size}x#{size}", source.pathmap("%n-#{size}%x")
- end
- sh 'magick', 'convert', *sizes.map{ |x| source.pathmap("%n-#{x}%x") }, source.pathmap('%n.ico')
- rm sizes.map{ |x| source.pathmap("%n-#{x}%x") }
- end
- end
- end
|