diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index a603b679..dc45f0f6 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -13,17 +13,22 @@ jobs: platform: - "x64-mingw32" - "x64-mingw-ucrt" - name: cross-compile-windows + - "x86_64-linux-gnu" + - "x86_64-linux-musl" + - "aarch64-linux-gnu" + - "aarch64-linux-musl" + + name: cross-compile runs-on: ubuntu-22.04 - container: - image: "ghcr.io/rake-compiler/rake-compiler-dock-image:1.7.0-mri-${{ matrix.platform }}" steps: - uses: actions/checkout@v4 - - run: git config --global --add safe.directory /__w/tiny_tds/tiny_tds # shrug + - name: Set up Ruby + uses: ruby/setup-ruby@v1 + with: + ruby-version: "2.7" - - name: Install gems - shell: bash + - name: "Install dependencies" run: bundle install - name: Write used versions into file @@ -34,14 +39,14 @@ jobs: uses: actions/cache@v4 with: path: ports - key: cross-compiled-v3-${{ matrix.platform }}-${{ hashFiles('**/.ports_versions') }} + key: cross-compiled-v1-${{ matrix.platform }}-${{ hashFiles('**/.ports_versions') }} restore-keys: | - cross-compiled-v3-${{ matrix.platform }}-${{ hashFiles('**/.ports_versions') }} - cross-compiled-v3-${{ matrix.platform }}- + cross-compiled-v1-${{ matrix.platform }}-${{ hashFiles('**/.ports_versions') }} + cross-compiled-v1-${{ matrix.platform }}- - name: Build gem shell: bash - run: bundle exec rake gem:for_platform[${{ matrix.platform }}] + run: bundle exec rake gem:native:${{ matrix.platform }} - uses: actions/upload-artifact@v4 with: @@ -89,6 +94,18 @@ jobs: ruby -e "require 'tiny_tds'; puts TinyTds::Gem.root_path" exit $LASTEXITCODE + - name: Test if tsql wrapper works + shell: pwsh + run: | + tsql-ttds -C + exit $LASTEXITCODE + + - name: Test if defncopy wrapper works + shell: pwsh + run: | + defncopy-ttds -v + exit $LASTEXITCODE + test-windows-mingw: needs: - cross-compile @@ -123,21 +140,9 @@ jobs: - name: Install native gem and restore cross-compiled code from it shell: pwsh - run: | - $rubyArchitecture = (ruby -e "puts RbConfig::CONFIG['arch']").Trim() - $gemVersion = (Get-Content VERSION).Trim() - $gemToUnpack = "./tiny_tds-$gemVersion-$rubyArchitecture.gem" - - Write-Host "Looking to unpack $gemToUnpack" - gem unpack --target ./tmp "$gemToUnpack" - - # Restore precompiled code - $source = (Resolve-Path ".\tmp\tiny_tds-$gemVersion-$rubyArchitecture\lib\tiny_tds").Path - $destination = (Resolve-Path ".\lib\tiny_tds").Path - Get-ChildItem $source -Recurse -Exclude "*.rb" | Copy-Item -Destination {Join-Path $destination $_.FullName.Substring($source.length)} - - # Restore ports - Copy-Item -Path ".\tmp\tiny_tds-$gemVersion-$rubyArchitecture\ports" -Destination "." -Recurse + run: "& ./test/bin/restore-from-native-gem.ps1" + env: + RUBY_ARCHITECTURE: "x64-mingw32" - name: Setup MSSQL uses: rails-sqlserver/setup-mssql@v1 @@ -214,6 +219,18 @@ jobs: ruby -e "require 'tiny_tds'; puts TinyTds::Gem.root_path" exit $LASTEXITCODE + - name: Test if tsql wrapper works + shell: pwsh + run: | + tsql-ttds -C + exit $LASTEXITCODE + + - name: Test if defncopy wrapper works + shell: pwsh + run: | + defncopy-ttds -v + exit $LASTEXITCODE + test-windows-ucrt: needs: - cross-compile @@ -250,21 +267,9 @@ jobs: - name: Install native gem and restore cross-compiled code from it shell: pwsh - run: | - $rubyArchitecture = (ruby -e "puts RbConfig::CONFIG['arch']").Trim() - $gemVersion = (Get-Content VERSION).Trim() - $gemToUnpack = "./tiny_tds-$gemVersion-$rubyArchitecture.gem" - - Write-Host "Looking to unpack $gemToUnpack" - gem unpack --target ./tmp "$gemToUnpack" - - # Restore precompiled code - $source = (Resolve-Path ".\tmp\tiny_tds-$gemVersion-$rubyArchitecture\lib\tiny_tds").Path - $destination = (Resolve-Path ".\lib\tiny_tds").Path - Get-ChildItem $source -Recurse -Exclude "*.rb" | Copy-Item -Destination {Join-Path $destination $_.FullName.Substring($source.length)} - - # Restore ports - Copy-Item -Path ".\tmp\tiny_tds-$gemVersion-$rubyArchitecture\ports" -Destination "." -Recurse + run: "& ./test/bin/restore-from-native-gem.ps1" + env: + RUBY_ARCHITECTURE: "x64-mingw-ucrt" - name: Setup MSSQL uses: rails-sqlserver/setup-mssql@v1 @@ -336,44 +341,94 @@ jobs: ruby -e "require 'tiny_tds'; puts TinyTds::Gem.root_path" exit $LASTEXITCODE - compile-native-ports: + - name: Test if tsql wrapper works + shell: pwsh + run: | + tsql-ttds -C + exit $LASTEXITCODE + + - name: Test if defncopy wrapper works + shell: pwsh + run: | + defncopy-ttds -v + exit $LASTEXITCODE + + install-linux: + needs: + - cross-compile + strategy: + fail-fast: false + matrix: + platform: + - "x86_64-linux-gnu" + - "x86_64-linux-musl" + - "aarch64-linux-gnu" + - "aarch64-linux-musl" + + ruby-version: + - "2.7" + - "3.0" + - "3.1" + - "3.2" + - "3.3" + - "3.4" + + include: + - platform: x86_64-linux-musl + docker_tag: "-alpine" + bootstrap: "apk add -U build-base &&" # required to compile bigdecimal on Ruby 2.7 + + - platform: aarch64-linux-gnu + docker_platform: "--platform=linux/arm64" + + - platform: aarch64-linux-musl + docker_platform: "--platform=linux/arm64" + docker_tag: "-alpine" + bootstrap: "apk add -U build-base &&" + + name: install-linux runs-on: ubuntu-22.04 - name: cross-compile-linux steps: - uses: actions/checkout@v4 - - uses: ruby/setup-ruby@v1 - with: - ruby-version: 3.4 - bundler-cache: true - - - name: Write used versions into file - run: bundle exec rake ports:version_file - - - name: Cache ports - uses: actions/cache@v4 + - name: Download precompiled gem + uses: actions/download-artifact@v4 with: - path: ports - key: native-v3-${{ hashFiles('**/.ports_versions') }} - restore-keys: | - native-v3-${{ hashFiles('* */.ports_versions') }} - native-v3- - - - name: Build required libraries - run: | - bundle exec rake ports + name: gem-${{ matrix.platform }} + path: precompiled/gems + + - name: Setup QEMU for docker + uses: docker/setup-qemu-action@v3 + if: ${{ matrix.docker_platform }} != '' + + - run: | + docker run --rm -v $PWD/precompiled:/precompiled -w /precompiled \ + ${{ matrix.docker_platform }} ruby:${{ matrix.ruby-version }}${{ matrix.docker_tag }} \ + sh -c " + gem update --system 3.3.22 && + ${{ matrix.bootstrap }} + gem install --no-document ./gems/tiny_tds-$(cat VERSION)-${{ matrix.platform }}.gem && + ruby -e \"require 'tiny_tds'; puts TinyTds::Gem.root_path\" && + tsql-ttds -C && + defncopy-ttds -v + " test-linux: needs: - - compile-native-ports + - cross-compile name: test-linux strategy: fail-fast: false matrix: + force-encryption: + - false + - true + mssql-version: - 2017 - 2019 - 2022 + ruby-version: - "2.7" - "3.0" @@ -391,20 +446,16 @@ jobs: ruby-version: ${{ matrix.ruby-version }} bundler-cache: true - - name: Write used versions into file - run: | - bundle exec rake ports:version_file - - - name: Cache ports - uses: actions/cache@v4 + - name: Download precompiled gem + uses: actions/download-artifact@v4 with: - path: ports - key: native-v3-${{ hashFiles('**/.ports_versions') }} - fail-on-cache-miss: true + name: gem-x86_64-linux-gnu - - name: Build gem - run: | - bundle exec rake build + - name: Install native gem and restore cross-compiled code from it + shell: pwsh + run: "& ./test/bin/restore-from-native-gem.ps1" + env: + RUBY_ARCHITECTURE: "x86_64-linux-gnu" - name: Setup MSSQL uses: rails-sqlserver/setup-mssql@v1 @@ -412,6 +463,7 @@ jobs: components: sqlcmd,sqlengine version: ${{ matrix.mssql-version }} sa-password: "c0MplicatedP@ssword" + force-encryption: ${{ matrix.force-encryption }} - name: Setup MSSQL database run: | @@ -435,6 +487,52 @@ jobs: paths: "test/reports/TEST-*.xml" if: always() + install-linux-native: + strategy: + fail-fast: false + matrix: + ruby-version: + - "2.7" + - "3.0" + - "3.1" + - "3.2" + - "3.3" + - "3.4" + + name: install-linux-native + runs-on: ubuntu-22.04 + steps: + - uses: actions/checkout@v4 + + - uses: ruby/setup-ruby@v1 + with: + ruby-version: ${{ matrix.ruby-version }} + bundler-cache: true + + - name: Install FreeTDS + shell: bash + run: ./test/bin/install-freetds.sh + + - name: Build gem + shell: bash + run: gem build tiny_tds.gemspec + + - name: Install gem + shell: bash + run: gem install "tiny_tds-$(cat VERSION).gem" + + - name: Test if TinyTDS loads + shell: bash + run: ruby -e "require 'tiny_tds'; puts TinyTds::Gem.root_path" + + - name: Test if tsql wrapper works + shell: bash + run: tsql-ttds -C + + - name: Test if defncopy wrapper works + shell: bash + run: defncopy-ttds -v + install_macos: strategy: fail-fast: false diff --git a/CHANGELOG.md b/CHANGELOG.md index 3128e1ec..ffe8e6b7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,9 @@ +## 3.2.0 + +* Reduce number of files shipped with precompiled Windows gem +* Provide precompiled gem for Linux (GNU + MUSL / 64-bit x86 + ARM) +* Fix wrappers for `tsql` and `defncopy` utility. + ## 3.1.0 * Add Ruby 3.4 to the cross compile list @@ -13,6 +19,7 @@ * Add `bigdecimal` to dependencies ## 2.1.7 + * Add Ruby 3.3 to the cross compile list ## 2.1.6 diff --git a/README.md b/README.md index f0af29f6..435fd09e 100644 --- a/README.md +++ b/README.md @@ -18,16 +18,41 @@ The API is simple and consists of these classes: ## Install -Installing with rubygems should just work. TinyTDS is currently tested on Ruby version 2.7.0 and upward. +tiny_tds is tested with Ruby v2.7 and upwards. +### Windows and Linux (64-bit) + +We precompile tiny_tds with static versions of FreeTDS and supporting libraries. Just run: + +```shell +gem install tiny_tds ``` -$ gem install tiny_tds + +It should find the platform-specific gem. + +You can also avoid getting the platform-specific gem if you want to compile FreeTDS yourself: + +```shell +gem install tiny_tds --platform ruby ``` -If you use Windows, we pre-compile TinyTDS with static versions of FreeTDS and supporting libraries. -If you're using RubyInstaller, the binary gem will require that devkit is installed and in your path to operate properly. +### Mac + +Install FreeTDS via Homebrew: + +```shell +brew install freetds +``` -On all other platforms, we will find these dependencies. It is recommended that you install the latest FreeTDS via your method of choice. For example, here is how to install FreeTDS on Ubuntu. You might also need the `build-essential` and possibly the `libc6-dev` packages. +Then you can install tiny_tds: + +```shell +gem install tiny_tds +``` + +### Everybody else + +`tiny_tds` will find FreeTDS based on your compiler paths. Below you can see an example on how to install FreeTDS on a Debian system. ```shell $ apt-get install wget @@ -37,41 +62,21 @@ $ apt-get install libc6-dev $ wget http://www.freetds.org/files/stable/freetds-1.4.23.tar.gz $ tar -xzf freetds-1.4.23.tar.gz $ cd freetds-1.4.23 -$ ./configure --prefix=/usr/local --with-tdsver=7.4 +$ ./configure --prefix=/usr/local --with-tdsver=7.4 --disable-odbc $ make $ make install ``` -Please read the MiniPortile and/or Windows sections at the end of this file for advanced configuration options past the following: +You can also tell `tiny_tds` where to find your FreeTDS installation. -``` ---with-freetds-dir=DIR - Use the freetds library placed under DIR. +```shell +gem install tiny_tds -- --with-freetds-dir=/opt/freetds ``` - ## Getting Started Optionally, Microsoft has done a great job writing [an article](https://learn.microsoft.com/en-us/sql/connect/ruby/ruby-driver-for-sql-server?view=sql-server-ver16) on how to get started with SQL Server and Ruby using TinyTDS, however, the articles are using outdated versions. - -## FreeTDS Compatibility & Configuration - -TinyTDS is developed against FreeTDs 1.1+. We also test with SQL Server 2017, 2019, 2022 and Azure. Older version of SQL Server or FreeTDS could work, but are not supported. - -> [!IMPORTANT] -> -> Windows users of our pre-compiled native gems need not worry about installing FreeTDS and its dependencies. - -* **Do I need to install FreeTDS?** Yes! Somehow, someway, you are going to need FreeTDS for TinyTDS to compile against. - -* **OK, I am installing FreeTDS, how do I configure it?** Contrary to what most people think, you do not need to specially configure FreeTDS in any way for client libraries like TinyTDS to use it. About the only requirement is that you compile it with libiconv for proper encoding support. FreeTDS must also be compiled with OpenSSL (or the like) to use it with Azure. See the "Using TinyTDS with Azure" section below for more info. - -* **Do I need to configure `--with-tdsver` equal to anything?** Most likely! Technically you should not have to. This is only a default for clients/configs that do not specify what TDS version they want to use. - -* **I want to configure FreeTDS using `--enable-msdblib` and/or `--enable-sybase-compat` so it works for my database. Cool?** It's a waste of time and totally moot! Client libraries like TinyTDS define their own C structure names where they diverge from Sybase to SQL Server. Technically we use the MSDBLIB structures which does not mean we only work with that database vs Sybase. These configs are just a low level default for C libraries that do not define what they want. So I repeat, you do not NEED to use any of these, nor will they hurt anything since we control what C structure names we use internally! - - ## Data Types Our goal is to support every SQL Server data type and covert it to a logical Ruby object. When dates or times are returned, they are instantiated to either `:utc` or `:local` time depending on the query options. Only [datetimeoffset] types are excluded. All strings are associated the to the connection's encoding and all binary data types are associated to Ruby's `ASCII-8BIT/BINARY` encoding. @@ -387,19 +392,6 @@ Please read our [thread_test.rb](https://github.com/rails-sqlserver/tiny_tds/blo This is possible. Since FreeTDS v1.0, utf-16 is enabled by default and supported by tiny_tds. You can toggle it by using `use_utf16` when establishing the connection. -## Compiling Gems for Windows - -For the convenience of Windows users, TinyTDS ships pre-compiled gems for supported versions of Ruby on Windows. In order to generate these gems, [rake-compiler-dock](https://github.com/rake-compiler/rake-compiler-dock) is used. This project provides several [Docker images](https://registry.hub.docker.com/u/larskanis/) with rvm, cross-compilers and a number of different target versions of Ruby. - -Run the following rake task to compile the gems for Windows. This will check the availability of [Docker](https://www.docker.com/) (and boot2docker on Windows or OS-X) and will give some advice for download and installation. When docker is running, it will download the docker image (once-only) and start the build: - -``` -$ rake gem:native -``` - -The compiled gems will exist in `./pkg` directory. - - ## Development & Testing First, clone the repo using the command line or your Git GUI of choice. @@ -446,16 +438,25 @@ $ rake TINYTDS_UNIT_DATASERVER=mydbserver TINYTDS_SCHEMA=sqlserver_2017 $ rake TINYTDS_UNIT_HOST=mydb.host.net TINYTDS_SCHEMA=sqlserver_azure ``` -## Docker Builds +### Compiling Gems for Windows and Linux + +For the convenience, TinyTDS ships pre-compiled gems for supported versions of Ruby on Windows and Linux. In order to generate these gems, [rake-compiler-dock](https://github.com/rake-compiler/rake-compiler-dock) is used. -If you use a [multi stage](https://docs.docker.com/develop/develop-images/multistage-build/) Docker build to assemble your gems in one phase and then copy your app and gems -into another, lighter, container without build tools you will need to make sure you tell the OS how to find dependencies for TinyTDS. +Run the following rake task to compile the gems. This will check the availability of [Docker](https://www.docker.com/) and will give some advice for download and installation. When docker is running, it will download the docker image (once-only) and start the build: + +```shell +bundle exec rake gem:native +``` -After you have built and installed FreeTDS it will normally place library files in `/usr/local/lib`. When TinyTDS builds native extensions, -it [already knows to look here](https://github.com/rails-sqlserver/tiny_tds/blob/master/ext/tiny_tds/extconf.rb#L31) but if you copy your app to a new container that link will be broken. +The compiled gems will exist in `./pkg` directory. + +If you only need a specific gem for one platform and architecture, run this command: + +```shell +bundle exec rake gem:native:x64-mingw-ucrt +``` -Set the LD_LIBRARY_PATH environment variable `export LD_LIBRARY_PATH=/usr/local/lib:${LD_LIBRARY_PATH}` and run `ldconfig`. If you run `ldd tiny_tds.so` you should not see any broken links. Make -sure you also copied in the library dependencies from your build container with a command like `COPY --from=builder /usr/local/lib /usr/local/lib`. +All the supported architectures and platforms are listed in the `Rakefile` in the `CrossLibraries` constant. ## Help & Support @@ -481,4 +482,4 @@ My name is Ken Collins and I currently maintain the SQL Server adapter for Activ ## License -TinyTDS is Copyright (c) 2010-2015 Ken Collins, and Will Bond (Veracross LLC) . It is distributed under the MIT license. Windows binaries contain pre-compiled versions of FreeTDS which is licensed under the GNU LGPL license at +TinyTDS is Copyright (c) 2010-2015 Ken Collins, and Will Bond (Veracross LLC) . It is distributed under the MIT license. Windows and Linux binaries contain pre-compiled versions of FreeTDS and `libconv` which is licensed under the GNU LGPL license at . They also contain OpenSSL, which is licensed under the OpenSSL license at . diff --git a/Rakefile b/Rakefile index 9999bff6..887b9ff2 100644 --- a/Rakefile +++ b/Rakefile @@ -3,23 +3,20 @@ require 'rbconfig' require 'rake' require 'rake/clean' require 'rake/extensiontask' -require_relative './ext/tiny_tds/extconsts' SPEC = Gem::Specification.load(File.expand_path('../tiny_tds.gemspec', __FILE__)) -ruby_cc_ucrt_versions = "3.4.0:3.3.5:3.2.0:3.1.0".freeze -ruby_cc_mingw32_versions = "3.0.0:2.7.0".freeze - -GEM_PLATFORM_HOSTS = { - 'x64-mingw32' => { - host: 'x86_64-w64-mingw32', - ruby_versions: ruby_cc_mingw32_versions - }, - 'x64-mingw-ucrt' => { - host: 'x86_64-w64-mingw32', - ruby_versions: ruby_cc_ucrt_versions - }, -} +CrossLibrary = Struct.new :platform, :openssl_config +CrossLibraries = [ + ['x64-mingw-ucrt', 'mingw64'], + ['x64-mingw32', 'mingw64'], + ['x86_64-linux-gnu', 'linux-x86_64'], + ['x86_64-linux-musl', 'linux-x86_64'], + ['aarch64-linux-gnu', 'linux-aarch64'], + ['aarch64-linux-musl', 'linux-aarch64'], +].map do |platform, openssl_config| + CrossLibrary.new platform, openssl_config +end # Add our project specific files to clean for a rebuild CLEAN.include FileList["{ext,lib}/**/*.{so,#{RbConfig::CONFIG['DLEXT']},o}"], @@ -35,22 +32,35 @@ Dir['tasks/*.rake'].sort.each { |f| load f } Rake::ExtensionTask.new('tiny_tds', SPEC) do |ext| ext.lib_dir = 'lib/tiny_tds' ext.cross_compile = true - ext.cross_platform = GEM_PLATFORM_HOSTS.keys + ext.cross_platform = CrossLibraries.map(&:platform) # Add dependent DLLs to the cross gems ext.cross_compiling do |spec| # The fat binary gem doesn't depend on the freetds package, since it bundles the library. spec.metadata.delete('msys2_mingw_dependencies') - - # We don't need the sources in a fat binary gem - spec.files = spec.files.reject { |f| f =~ %r{^ports\/archives/} } - - # Make sure to include the ports binaries and libraries - spec.files += FileList["ports/#{spec.platform.to_s}/**/**/{bin,lib}/*"].exclude do |f| - File.directory? f + + if spec.platform.to_s =~ /mingw/ + spec.files += [ + "ports/#{spec.platform.to_s}/bin/libsybdb-5.dll", + "ports/#{spec.platform.to_s}/bin/defncopy.exe", + "ports/#{spec.platform.to_s}/bin/tsql.exe" + ] + elsif spec.platform.to_s =~ /linux/ + spec.files += [ + "ports/#{spec.platform.to_s}/lib/libsybdb.so.5", + "ports/#{spec.platform.to_s}/bin/defncopy", + "ports/#{spec.platform.to_s}/bin/tsql" + ] end + end - spec.files += Dir.glob('exe/*') + ext.cross_config_options += CrossLibraries.map do |xlib| + { + xlib.platform => [ + "--with-cross-build=#{xlib.platform}", + "--with-openssl-platform=#{xlib.openssl_config}" + ] + } end end diff --git a/VERSION b/VERSION index fd2a0186..944880fa 100644 --- a/VERSION +++ b/VERSION @@ -1 +1 @@ -3.1.0 +3.2.0 diff --git a/ext/tiny_tds/extconf.rb b/ext/tiny_tds/extconf.rb index d0691792..55a0d65e 100644 --- a/ext/tiny_tds/extconf.rb +++ b/ext/tiny_tds/extconf.rb @@ -1,91 +1,190 @@ -ENV['RC_ARCHS'] = '' if RUBY_PLATFORM =~ /darwin/ - -# :stopdoc: - require 'mkmf' -require 'rbconfig' -require_relative './extconsts' - -# Shamelessly copied from nokogiri -# - -def do_help - print <. subdirectory for fat binary gems + major_minor = RUBY_VERSION[ /^(\d+\.\d+)/ ] or + raise "Oops, can't extract the major/minor version from #{RUBY_VERSION.dump}" + require "tiny_tds/#{major_minor}/tiny_tds" rescue LoadError require 'tiny_tds/tiny_tds' end end -else - # Load dependent shared libraries into the process, so that they are already present, - # when tiny_tds.so is loaded. This ensures, that shared libraries are loaded even when - # the path is different between build and run time (e.g. Heroku). - ports_libs = File.join(TinyTds::Gem.ports_root_path, - "#{RbConfig::CONFIG['host']}/lib/*.so") - Dir[ports_libs].each do |lib| - require 'fiddle' - Fiddle.dlopen(lib) - end - - require 'tiny_tds/tiny_tds' end diff --git a/lib/tiny_tds/bin.rb b/lib/tiny_tds/bin.rb index 832692a3..87cee11a 100644 --- a/lib/tiny_tds/bin.rb +++ b/lib/tiny_tds/bin.rb @@ -81,24 +81,11 @@ def which exe = File.expand_path File.join(path, "#{name}#{ext}"), @root next if exe == @binstub next unless File.executable?(exe) - next unless binary?(exe) + return exe end end nil end - - # Implementation directly copied from ptools. - # https://github.com/djberg96/ptools - # https://opensource.org/licenses/Artistic-2.0 - # - def binary?(file) - bytes = File.stat(file).blksize - return false unless bytes - bytes = 4096 if bytes > 4096 - s = (File.read(file, bytes) || '') - s = s.encode('US-ASCII', undef: :replace).split(//) - ((s.size - s.grep(' '..'~').size) / s.size.to_f) > 0.30 - end end end diff --git a/lib/tiny_tds/gem.rb b/lib/tiny_tds/gem.rb index a52f047d..55be78b4 100644 --- a/lib/tiny_tds/gem.rb +++ b/lib/tiny_tds/gem.rb @@ -12,15 +12,11 @@ def ports_root_path end def ports_bin_paths - Dir.glob(File.join(ports_root_path,ports_host,'**','bin')) + Dir.glob(File.join(ports_root_path, '**', 'bin')) end def ports_lib_paths - Dir.glob(File.join(ports_root_path,ports_host,'**','lib')) - end - - def ports_host - RbConfig::CONFIG["arch"] + Dir.glob(File.join(ports_root_path, '**', 'lib')) end end end diff --git a/tasks/native_gem.rake b/tasks/native_gem.rake index 355c1fb3..18be796d 100644 --- a/tasks/native_gem.rake +++ b/tasks/native_gem.rake @@ -1,23 +1,16 @@ -# encoding: UTF-8 +CrossLibraries.each do |xlib| + platform = xlib.platform + + desc "Build fat binary gem for platform #{platform}" + task "gem:native:#{platform}" do + require "rake_compiler_dock" -desc 'Build the native binary gems using rake-compiler-dock' -task 'gem:native' => ['ports:cross'] do - require 'rake_compiler_dock' + RakeCompilerDock.sh <<-EOT, platform: platform + bundle install && + rake native:#{platform} pkg/#{SPEC.full_name}-#{platform}.gem MAKEOPTS=-j`nproc` RUBY_CC_VERSION=3.4.1:3.3.5:3.2.0:3.1.0:3.0.0:2.7.0 MAKEFLAGS="V=1" + EOT + end - # make sure to install our bundle - sh "bundle package --all" # Avoid repeated downloads of gems by using gem files from the host. - - GEM_PLATFORM_HOSTS.each do |plat, meta| - RakeCompilerDock.sh "bundle --local && RUBY_CC_VERSION=#{meta[:ruby_versions]} rake native:#{plat} gem", platform: plat - end -end - -# assumes you are in a container provided by Rake compiler -# if not, use the task above -task 'gem:for_platform', [:gem_platform] do |_task, args| - args.with_defaults(gem_platform: RbConfig::CONFIG["arch"]) - - sh "bundle install" - Rake::Task["ports:compile"].invoke(GEM_PLATFORM_HOSTS[args.gem_platform][:host], args.gem_platform) - sh "RUBY_CC_VERSION=#{GEM_PLATFORM_HOSTS[args.gem_platform][:ruby_versions]} rake native:#{args.gem_platform} gem" + desc "Build the native binary gems" + multitask 'gem:native' => "gem:native:#{platform}" end diff --git a/tasks/ports.rake b/tasks/ports.rake index 1ab6bf96..4fdc9498 100644 --- a/tasks/ports.rake +++ b/tasks/ports.rake @@ -1,99 +1,20 @@ -# encoding: UTF-8 -require 'mini_portile2' -require 'fileutils' -require_relative 'ports/libiconv' -require_relative 'ports/openssl' -require_relative 'ports/freetds' require_relative '../ext/tiny_tds/extconsts' namespace :ports do libraries_to_compile = { - openssl: Ports::Openssl.new(OPENSSL_VERSION), - libiconv: Ports::Libiconv.new(ICONV_VERSION), - freetds: Ports::Freetds.new(FREETDS_VERSION) + openssl: OPENSSL_VERSION, + libiconv: ICONV_VERSION, + freetds: FREETDS_VERSION } - directory "ports" - CLEAN.include "ports/*mingw*" - CLEAN.include "ports/*.installed" - - task :openssl, [:host, :gem_platform] do |_task, args| - args.with_defaults(host: RbConfig::CONFIG['host'], gem_platform: RbConfig::CONFIG["arch"]) - - libraries_to_compile[:openssl].files = [OPENSSL_SOURCE_URI] - libraries_to_compile[:openssl].host = args.host - libraries_to_compile[:openssl].gem_platform = args.gem_platform - - libraries_to_compile[:openssl].cook - libraries_to_compile[:openssl].activate - end - - task :libiconv, [:host, :gem_platform] do |_task, args| - args.with_defaults(host: RbConfig::CONFIG['host'], gem_platform: RbConfig::CONFIG["arch"]) - - libraries_to_compile[:libiconv].files = [ICONV_SOURCE_URI] - libraries_to_compile[:libiconv].host = args.host - libraries_to_compile[:libiconv].gem_platform = args.gem_platform - libraries_to_compile[:libiconv].cook - libraries_to_compile[:libiconv].activate - end - - task :freetds, [:host, :gem_platform] do |_task, args| - args.with_defaults(host: RbConfig::CONFIG['host'], gem_platform: RbConfig::CONFIG["arch"]) - - libraries_to_compile[:freetds].files = [FREETDS_SOURCE_URI] - libraries_to_compile[:freetds].host = args.host - libraries_to_compile[:freetds].gem_platform = args.gem_platform - - if libraries_to_compile[:openssl] - # freetds doesn't have an option that will provide an rpath - # so we do it manually - ENV['OPENSSL_CFLAGS'] = "-Wl,-rpath -Wl,#{libraries_to_compile[:openssl].path}/lib64" - # Add the pkgconfig file with MSYS2'ish path, to prefer our ports build - # over MSYS2 system OpenSSL. - ENV['PKG_CONFIG_PATH'] = "#{libraries_to_compile[:openssl].path.gsub(/^(\w):/i) { "/" + $1.downcase }}/lib64/pkgconfig:#{ENV['PKG_CONFIG_PATH']}" - libraries_to_compile[:freetds].configure_options << "--with-openssl=#{libraries_to_compile[:openssl].path}" - end - - if libraries_to_compile[:libiconv] - libraries_to_compile[:freetds].configure_options << "--with-libiconv-prefix=#{libraries_to_compile[:libiconv].path}" - end - - libraries_to_compile[:freetds].cook - libraries_to_compile[:freetds].activate - end - - task :compile, [:host, :gem_platform] do |_task, args| - args.with_defaults(host: RbConfig::CONFIG['host'], gem_platform: RbConfig::CONFIG["arch"]) - - puts "Compiling ports for #{args.host} (Ruby platform #{args.gem_platform}) ..." - - libraries_to_compile.keys.each do |lib| - Rake::Task["ports:#{lib}"].invoke(args.host, args.gem_platform) - end - end - - desc 'Build the ports windows binaries via rake-compiler-dock' - task 'cross' do - require 'rake_compiler_dock' - - # build the ports for all our cross compile hosts - GEM_PLATFORM_HOSTS.each do |gem_platform, meta| - # make sure to install our bundle - build = ['bundle'] - build << "RUBY_CC_VERSION=#{meta[:ruby_versions]} rake ports:compile[#{meta[:host]},#{gem_platform}] MAKE='make -j`nproc`'" - RakeCompilerDock.sh build.join(' && '), platform: gem_platform - end - end - desc "Notes the actual versions for the compiled ports into a file" task "version_file", [:gem_platform] do |_task, args| args.with_defaults(gem_platform: RbConfig::CONFIG["arch"]) ports_version = {} - libraries_to_compile.each do |library, library_recipe| - ports_version[library] = library_recipe.version + libraries_to_compile.each do |library, version| + ports_version[library] = version end ports_version[:platform] = args.gem_platform @@ -103,6 +24,3 @@ namespace :ports do end end end - -desc 'Build ports and activate libraries for the current architecture.' -task :ports => ['ports:compile'] diff --git a/tasks/ports/freetds.rb b/tasks/ports/freetds.rb deleted file mode 100644 index d0f4166c..00000000 --- a/tasks/ports/freetds.rb +++ /dev/null @@ -1,32 +0,0 @@ -require_relative './recipe' - -module Ports - class Freetds < Recipe - def initialize(version) - super('freetds', version) - - set_patches - end - - private - - def configure_defaults - opts = super - - opts << '--with-pic' - opts << '--disable-odbc' - opts << '--with-tdsver=7.3' - - if windows? - opts << '--sysconfdir=C:/Sites' - opts << '--enable-sspi' - end - - opts - end - - def set_patches - self.patch_files.concat get_patches(name, version) - end - end -end diff --git a/tasks/ports/libiconv.rb b/tasks/ports/libiconv.rb deleted file mode 100644 index 52e0f184..00000000 --- a/tasks/ports/libiconv.rb +++ /dev/null @@ -1,26 +0,0 @@ -require_relative './recipe' - -module Ports - class Libiconv < Recipe - def initialize(version) - super('libiconv', version) - - set_patches - end - - private - - def configure_defaults - [ - "--host=#{@host}", - '--disable-static', - '--enable-shared', - 'CFLAGS=-fPIC -O2' - ] - end - - def set_patches - self.patch_files.concat get_patches(name, version) - end - end -end diff --git a/tasks/ports/openssl.rb b/tasks/ports/openssl.rb deleted file mode 100644 index 17da4e0a..00000000 --- a/tasks/ports/openssl.rb +++ /dev/null @@ -1,62 +0,0 @@ -require_relative './recipe' - -module Ports - class Openssl < Recipe - def initialize(version) - super('openssl', version) - - set_patches - end - - def configure - return if configured? - - md5_file = File.join(tmp_path, 'configure.md5') - digest = Digest::MD5.hexdigest(computed_options.to_s) - File.open(md5_file, "w") { |f| f.write digest } - - # Windows doesn't recognize the shebang so always explicitly use sh - execute('configure', "sh -c \"./Configure #{computed_options.join(' ')}\"") - end - - def install - unless installed? - execute('install', %Q(#{make_cmd} install_sw install_ssldirs)) - end - end - - private - - def configure_defaults - opts = [ - 'shared', - target_arch, - "--openssldir=#{path}", - ] - - if cross_build? - opts << "--cross-compile-prefix=#{host}-" - end - - opts - end - - def target_arch - if windows? - arch = '' - arch = '64' if host=~ /x86_64/ - - "mingw#{arch}" - else - arch = 'x32' - arch = 'x86_64' if host=~ /x86_64/ - - "linux-#{arch}" - end - end - - def set_patches - self.patch_files.concat get_patches(name, version) - end - end -end diff --git a/tasks/ports/recipe.rb b/tasks/ports/recipe.rb deleted file mode 100644 index b3d4dd04..00000000 --- a/tasks/ports/recipe.rb +++ /dev/null @@ -1,64 +0,0 @@ -# encoding: UTF-8 -require 'mini_portile2' -require 'fileutils' -require 'rbconfig' - -module Ports - class Recipe < MiniPortile - attr_writer :gem_platform - - def cook - checkpoint = "ports/checkpoints/#{name}-#{version}-#{gem_platform}.installed" - - unless File.exist? checkpoint - super - FileUtils.mkdir_p("ports/checkpoints") - FileUtils.touch checkpoint - end - end - - private - - attr_reader :gem_platform - - def port_path - "#{@target}/#{gem_platform}/#{@name}/#{@version}" - end - - def tmp_path - "tmp/#{gem_platform}/ports/#{@name}/#{@version}" - end - - def configure_defaults - [ - "--host=#{@host}", - '--disable-static', - '--enable-shared' - ] - end - - def windows? - host =~ /mswin|mingw32/ - end - - def system_host - RbConfig::CONFIG['host'] - end - - def cross_build? - host != system_host - end - - def get_patches(libname, version) - patches = [] - - patch_path = File.expand_path( - File.join('..','..','..','patches',libname,version), - __FILE__ - ) - - patches.concat(Dir[File.join(patch_path, '*.patch')].sort) - patches.concat(Dir[File.join(patch_path, '*.diff')].sort) - end - end -end diff --git a/test/bin/install-freetds.sh b/test/bin/install-freetds.sh index d4ba3955..8c1f0eac 100755 --- a/test/bin/install-freetds.sh +++ b/test/bin/install-freetds.sh @@ -10,11 +10,9 @@ fi wget http://www.freetds.org/files/stable/freetds-$FREETDS_VERSION.tar.gz tar -xzf freetds-$FREETDS_VERSION.tar.gz cd freetds-$FREETDS_VERSION -./configure --prefix=/opt/local \ - --with-openssl=/opt/local \ - --with-tdsver=7.3 +./configure make -make install +sudo make install cd .. rm -rf freetds-$FREETDS_VERSION rm freetds-$FREETDS_VERSION.tar.gz diff --git a/test/bin/restore-from-native-gem.ps1 b/test/bin/restore-from-native-gem.ps1 new file mode 100644 index 00000000..daeb0287 --- /dev/null +++ b/test/bin/restore-from-native-gem.ps1 @@ -0,0 +1,10 @@ +$gemVersion = (Get-Content VERSION).Trim() +$gemToUnpack = "./tiny_tds-$gemVersion-$env:RUBY_ARCHITECTURE.gem" + +Write-Host "Looking to unpack $gemToUnpack" +gem unpack --target ./tmp "$gemToUnpack" + +# Restore precompiled code +$source = (Resolve-Path ".\tmp\tiny_tds-$gemVersion-$env:RUBY_ARCHITECTURE\lib\tiny_tds").Path +$destination = (Resolve-Path ".\lib\tiny_tds").Path +Get-ChildItem $source -Recurse -Exclude "*.rb" | Copy-Item -Destination {Join-Path $destination $_.FullName.Substring($source.length)} diff --git a/test/gem_test.rb b/test/gem_test.rb index 0412a88a..eea45623 100644 --- a/test/gem_test.rb +++ b/test/gem_test.rb @@ -98,79 +98,6 @@ class GemTest < Minitest::Spec end end end - - describe '#ports_lib_paths' do - let(:ports_lib_paths) { TinyTds::Gem.ports_lib_paths } - - describe 'when the ports directories exist' do - let(:fake_lib_paths) do - ports_host_root = File.join(gem_root, 'ports', 'fake-host-with-dirs') - [ - File.join('a','lib'), - File.join('a','inner','lib'), - File.join('b','lib') - ].map do |p| - File.join(ports_host_root, p) - end - end - - before do - RbConfig::CONFIG['arch'] = 'fake-host-with-dirs' - fake_lib_paths.each do |path| - FileUtils.mkdir_p(path) - end - end - - after do - FileUtils.remove_entry_secure( - File.join(gem_root, 'ports', 'fake-host-with-dirs'), true - ) - end - - it 'should return all the lib directories' do - _(ports_lib_paths.sort).must_equal fake_lib_paths.sort - end - - it 'should return all the lib directories regardless of cwd' do - Dir.chdir '/' - _(ports_lib_paths.sort).must_equal fake_lib_paths.sort - end - end - - describe 'when the ports directories are missing' do - before do - RbConfig::CONFIG['arch'] = 'fake-host-without-dirs' - end - - - it 'should return no directories' do - _(ports_lib_paths).must_be_empty - end - - it 'should return no directories regardless of cwd' do - Dir.chdir '/' - _(ports_lib_paths).must_be_empty - end - end - end - - describe '#ports_host' do - { - 'x64-mingw-ucrt' => 'x64-mingw-ucrt', - 'x64-mingw32' => 'x64-mingw32', - 'x86_64-linux' => 'x86_64-linux', - }.each do |host,expected| - describe "on a #{host} architecture" do - before do - RbConfig::CONFIG['arch'] = host - end - - it "should return a #{expected} ports host" do - _(TinyTds::Gem.ports_host).must_equal expected - end - end - end - end end end diff --git a/tiny_tds.gemspec b/tiny_tds.gemspec index e50f3bbe..67f5e903 100644 --- a/tiny_tds.gemspec +++ b/tiny_tds.gemspec @@ -21,7 +21,7 @@ Gem::Specification.new do |s| s.required_ruby_version = '>= 2.7.0' s.metadata['msys2_mingw_dependencies'] = 'freetds' s.add_dependency 'bigdecimal', '~> 3' - s.add_development_dependency 'mini_portile2', '~> 2.5.0' + s.add_development_dependency 'mini_portile2', '~> 2.8.0' s.add_development_dependency 'rake', '~> 13.0.0' s.add_development_dependency 'rake-compiler', '~> 1.2' s.add_development_dependency 'rake-compiler-dock', '~> 1.7.0'