123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- class Package < ApplicationModel
- @@root = Rails.root.to_s # rubocop:disable Style/ClassVars
- =begin
- verify if package is installed correctly
- package = Package.find(123)
- issues = package.verify
- returns:
- # if no issue exists
- nil
- # list of issues
- {
- 'path/to/file' => 'missing',
- 'path/to/file' => 'changed',
- }
- =end
- def verify
- # get package
- json_file = self.class._get_bin(name, version)
- package = JSON.parse(json_file)
- # verify installed files
- issues = {}
- package['files'].each do |file|
- if !File.exist?(file['location'])
- logger.error "File #{file['location']} is missing"
- issues[file['location']] = 'missing'
- next
- end
- content_package = Base64.decode64(file['content'])
- content_fs = self.class._read_file(file['location'])
- next if content_package == content_fs
- logger.error "File #{file['location']} is different"
- issues[file['location']] = 'changed'
- end
- return nil if issues.blank?
- issues
- end
- =begin
- install all packages located under auto_install/*.zpm
- Package.auto_install
- =end
- def self.auto_install
- path = "#{@@root}/auto_install/"
- return if !File.exist?(path)
- data = []
- Dir.foreach(path) do |entry|
- if entry.include?('.zpm') && entry !~ %r{^\.}
- data.push entry
- end
- end
- data.each do |file|
- install(file: "#{path}/#{file}")
- end
- data
- end
- =begin
- remove all linked files in application
- note: will not take down package migrations, use Package.unlink instead
- Package.unlink_all
- =end
- def self.unlink_all
- # link files
- Dir.glob("#{@@root}/**/*") do |entry|
- if File.symlink?(entry)
- logger.info "unlink: #{entry}"
- File.delete(entry)
- end
- backup_file = "#{entry}.link_backup"
- if File.exist?(backup_file)
- logger.info "Restore backup file of #{backup_file} -> #{entry}."
- File.rename(backup_file, entry)
- end
- end
- end
- # check if zpm is a package source repo
- def self._package_base_dir?(package_base_dir)
- package = false
- Dir.glob("#{package_base_dir}/*.szpm") do |entry|
- package = entry.sub(%r{^.*/(.+?)\.szpm$}, '\1')
- end
- if package == false
- raise "Can't link package, '#{package_base_dir}' is no package source directory!"
- end
- logger.debug { package.inspect }
- package
- end
- =begin
- execute migration down + unlink files
- Package.unlink('/path/to/src/extension')
- =end
- def self.unlink(package_base_dir)
- # check if zpm is a package source repo
- package = _package_base_dir?(package_base_dir)
- # migration down
- Package::Migration.migrate(package, 'reverse')
- # link files
- Dir.glob("#{package_base_dir}/**/*") do |entry|
- entry = entry.sub('//', '/')
- file = entry
- file = file.sub(%r{#{package_base_dir}}, '')
- dest = "#{@@root}/#{file}"
- if File.symlink?(dest.to_s)
- logger.info "Unlink file: #{dest}"
- File.delete(dest.to_s)
- end
- backup_file = "#{dest}.link_backup"
- if File.exist?(backup_file)
- logger.info "Restore backup file of #{backup_file} -> #{dest}."
- File.rename(backup_file, dest.to_s)
- end
- end
- end
- =begin
- link files
- Package.link('/path/to/src/extension')
- Migrations will not be executed because the the codebase was modified
- in the current process and is therefore inconsistent. This must be done
- subsequently in a separate step.
- =end
- def self.link(package_base_dir)
- # link files
- Dir.glob("#{package_base_dir}/**/*") do |entry|
- entry = entry.sub('//', '/')
- file = entry
- file = file.sub(%r{#{package_base_dir}}, '')
- file = file.sub(%r{^/}, '')
- # ignore files
- if file.start_with?('README')
- logger.info "NOTICE: Ignore #{file}"
- next
- end
- # get new file destination
- dest = "#{@@root}/#{file}"
- if File.directory?(entry.to_s) && !File.exist?(dest.to_s)
- logger.info "Create dir: #{dest}"
- FileUtils.mkdir_p(dest.to_s)
- end
- if File.file?(entry.to_s) && (File.file?(dest.to_s) && !File.symlink?(dest.to_s))
- backup_file = "#{dest}.link_backup"
- if File.exist?(backup_file)
- raise "Can't link #{entry} -> #{dest}, destination and .link_backup already exists!"
- end
- logger.info "Create backup file of #{dest} -> #{backup_file}."
- File.rename(dest.to_s, backup_file)
- end
- if File.file?(entry)
- if File.symlink?(dest.to_s)
- File.delete(dest.to_s)
- end
- logger.info "Link file: #{entry} -> #{dest}"
- File.symlink(entry.to_s, dest.to_s)
- end
- end
- end
- =begin
- install zpm package
- package = Package.install(file: '/path/to/package.zpm')
- or
- package = Package.install(string: zpm_as_string)
- returns
- package # record of newly created package
- Migrations will not be executed because the the codebase was modified
- in the current process and is therefore inconsistent. This must be done
- subsequently in a separate step.
- =end
- def self.install(data)
- if data[:file]
- json = _read_file(data[:file], true)
- package = JSON.parse(json)
- elsif data[:string]
- package = JSON.parse(data[:string])
- end
- # package meta data
- meta = {
- name: package['name'],
- version: package['version'],
- vendor: package['vendor'],
- url: package['url'],
- state: 'uninstalled',
- created_by_id: 1,
- updated_by_id: 1,
- }
- # verify if package can get installed
- package_db = Package.find_by(name: meta[:name])
- if package_db
- if !data[:reinstall]
- if Gem::Version.new(package_db.version) == Gem::Version.new(meta[:version])
- raise "Package '#{meta[:name]}-#{meta[:version]}' already installed!"
- end
- if Gem::Version.new(package_db.version) > Gem::Version.new(meta[:version])
- raise "Newer version (#{package_db.version}) of package '#{meta[:name]}-#{meta[:version]}' already installed!"
- end
- end
- # uninstall files of old package
- uninstall(
- name: package_db.name,
- version: package_db.version,
- migration_not_down: true,
- reinstall: data[:reinstall],
- )
- end
- Transaction.execute do
- # store package
- if !data[:reinstall]
- package_db = Package.create(meta)
- Store.create!(
- object: 'Package',
- o_id: package_db.id,
- data: package.to_json,
- filename: "#{meta[:name]}-#{meta[:version]}.zpm",
- preferences: {},
- created_by_id: UserInfo.current_user_id || 1,
- )
- end
- # write files
- package['files'].each do |file|
- if !allowed_file_path?(file['location'])
- raise "Can't create file, because of not allowed file location: #{file['location']}!"
- end
- ensure_no_duplicate_files!(package_db.name, file['location'])
- permission = file['permission'] || '644'
- content = Base64.decode64(file['content'])
- _write_file(file['location'], permission, content)
- end
- # update package state
- package_db.reload
- package_db.state = 'installed'
- package_db.save
- end
- package_db
- end
- def self.ensure_no_duplicate_files!(name, location)
- all_files.each do |check_package, check_files|
- next if check_package == name
- next if check_files.exclude?(location)
- raise "Can't create file, because file '#{location}' is already provided by package '#{check_package}'!"
- end
- end
- def self.all_files
- Auth::RequestCache.fetch_value('Package/all_files') do
- Package.all.each_with_object({}) do |package, result|
- json_file = Package._get_bin(package.name, package.version)
- package_json = JSON.parse(json_file)
- result[package.name] = package_json['files'].pluck('location')
- end
- end
- end
- def self.app_frontend_files?
- Auth::RequestCache.fetch_value('Package/app_frontend_files') do
- Package.all_files.values.flatten.any? { |f| f.starts_with?('app/frontend') }
- end
- end
- def self.gem_files?
- Dir['Gemfile.local.*'].present?
- end
- def self.app_package_installation?
- File.exist?('/usr/bin/zammad')
- end
- =begin
- reinstall package
- package = Package.reinstall(package_name)
- returns
- package # record of newly created package
- =end
- def self.reinstall(package_name)
- package = Package.find_by(name: package_name)
- if !package
- raise "No such package '#{package_name}'"
- end
- file = _get_bin(package.name, package.version)
- install(string: file, reinstall: true)
- package
- end
- =begin
- uninstall package
- package = Package.uninstall(name: 'package', version: '0.1.1')
- or
- package = Package.uninstall(string: zpm_as_string)
- returns
- package # record of newly created package
- =end
- def self.uninstall(data)
- if data[:string]
- package = JSON.parse(data[:string])
- else
- json_file = _get_bin(data[:name], data[:version])
- package = JSON.parse(json_file)
- end
- # down migrations
- if !data[:migration_not_down]
- Package::Migration.migrate(package['name'], 'reverse')
- end
- record = Package.find_by(
- name: package['name'],
- version: package['version'],
- )
- if record.state == 'installed'
- package['files'].each do |file|
- permission = file['permission'] || '644'
- content = Base64.decode64(file['content'])
- _delete_file(file['location'], permission, content)
- end
- end
- # delete package
- if data[:reinstall]
- record.update(state: 'uninstalled')
- else
- record.destroy
- end
- record
- end
- =begin
- execute all pending package migrations at once
- Package.migration_execute
- =end
- def self.migration_execute
- Package.all.each do |package|
- json_file = Package._get_bin(package.name, package.version)
- package = JSON.parse(json_file)
- Package::Migration.migrate(package['name'])
- end
- # sync package po files
- Translation.sync
- end
- def self._get_bin(name, version)
- package = Package.find_by(
- name: name,
- version: version,
- )
- if !package
- raise "No such package '#{name}' version '#{version}'"
- end
- list = Store.list(
- object: 'Package',
- o_id: package.id,
- )
- # find file
- if !list || !list.first
- raise "No such file in storage list #{name} #{version}"
- end
- if !list.first.content
- raise "No such file in storage #{name} #{version}"
- end
- list.first.content
- end
- def self._read_file(file, fullpath = false)
- location = case fullpath
- when false
- "#{@@root}/#{file}"
- when true
- file
- else
- "#{fullpath}/#{file}"
- end
- File.binread(location)
- end
- def self._write_file(file, permission, data)
- location = "#{@@root}/#{file}"
- # rename existing file if not already the same file
- if File.exist?(location)
- backup_location = "#{location}.save"
- content_fs = _read_file(file)
- if content_fs == data && File.exist?(backup_location)
- logger.debug { "NOTICE: file '#{location}' already exists, skip install" }
- return true
- end
- logger.info "NOTICE: backup old file '#{location}' to #{backup_location}"
- File.rename(location, backup_location)
- end
- # check if directories need to be created
- directories = location.split '/'
- (0..(directories.length - 2)).each do |position|
- tmp_path = ''
- (1..position).each do |count|
- tmp_path = "#{tmp_path}/#{directories[count]}"
- end
- next if tmp_path == ''
- next if File.exist?(tmp_path)
- Dir.mkdir(tmp_path, 0o755)
- end
- # install file
- logger.info "NOTICE: install '#{location}' (#{permission})"
- file = File.new(location, 'wb')
- file.write(data)
- file.close
- File.chmod(permission.to_s.to_i(8), location)
- true
- end
- def self._delete_file(file, _permission, _data)
- location = "#{@@root}/#{file}"
- # install file
- logger.info "NOTICE: uninstall '#{location}'"
- FileUtils.rm_rf(location)
- # rename existing file
- backup_location = "#{location}.save"
- if File.exist?(backup_location)
- logger.info "NOTICE: restore old file '#{backup_location}' to #{location}"
- File.rename(backup_location, location)
- end
- true
- end
- def self.allowed_file_path?(file)
- file.exclude?('..') && file.exclude?('%2e%2e')
- end
- private_class_method :allowed_file_path?
- end
|