class W3DHub class Cache LOG_TAG = "W3DHub::Cache".freeze def self.path(uri) ext = File.basename(uri).split(".").last "#{CACHE_PATH}/#{Digest::SHA2.hexdigest(uri)}.#{ext}" end # Fetch a generic uri def self.fetch(uri:, force_fetch: false, async: true) path = path(uri) if !force_fetch && File.exist?(path) path elsif async BackgroundWorker.job( -> { Async::HTTP::Internet.instance.get(uri, W3DHub::Api::DEFAULT_HEADERS) }, ->(response) { response.save(path, "wb") if response.success? } ) else response = Async::HTTP::Internet.instance.get(uri, W3DHub::Api::DEFAULT_HEADERS) response.save(path, "wb") if response.success? end end def self.create_directories(path, is_directory = false) target_directory = is_directory ? path : File.dirname(path) FileUtils.mkdir_p(target_directory) unless Dir.exist?(target_directory) end def self.package_path(category, subcategory, name, version) package_cache_dir = Store.settings[:package_cache_dir] if version.to_s.length.zero? "#{package_cache_dir}/#{category}/#{subcategory}/#{name}.package" else "#{package_cache_dir}/#{category}/#{subcategory}/#{version}/#{name}.package" end end def self.install_path(application, channel) game_data = Store.settings[:games]&.dig(:"#{application.id}_#{channel.id}") return game_data[:install_directory] if game_data && game_data[:install_directory] "#{Store.settings[:app_install_dir]}/#{application.category}/#{application.id}/#{channel.id}" end # Download a W3D Hub package # TODO: More work needed to make this work reliably def self._async_fetch_package(package, block) path = package_path(package.category, package.subcategory, package.name, package.version) headers = Api::FORM_ENCODED_HEADERS start_from_bytes = package.custom_partially_valid_at_bytes logger.info(LOG_TAG) { " Start from bytes: #{start_from_bytes} of #{package.size}" } create_directories(path) file = File.open(path, start_from_bytes.positive? ? "r+b" : "wb") if start_from_bytes.positive? headers = Api::FORM_ENCODED_HEADERS + [["Range", "bytes=#{start_from_bytes}-"]] file.pos = start_from_bytes end body = "data=#{JSON.dump({ category: package.category, subcategory: package.subcategory, name: package.name, version: package.version })}" response = Api.post("#{Api::ENDPOINT}/apis/launcher/1/get-package", headers, body) total_bytes = package.size remaining_bytes = total_bytes - start_from_bytes response.each do |chunk| file.write(chunk) remaining_bytes -= chunk.size block.call(chunk, remaining_bytes, total_bytes) end response.success? ensure file&.close end # Download a W3D Hub package def self.fetch_package(package, block) path = package_path(package.category, package.subcategory, package.name, package.version) headers = { "Content-Type": "application/x-www-form-urlencoded", "User-Agent": Api::USER_AGENT } headers["Authorization"] = "Bearer #{Store.account.access_token}" if Store.account start_from_bytes = package.custom_partially_valid_at_bytes logger.info(LOG_TAG) { " Start from bytes: #{start_from_bytes} of #{package.size}" } create_directories(path) file = File.open(path, start_from_bytes.positive? ? "r+b" : "wb") if start_from_bytes.positive? headers["Range"] = "bytes=#{start_from_bytes}-" file.pos = start_from_bytes end streamer = lambda do |chunk, remaining_bytes, total_bytes| file.write(chunk) block.call(chunk, remaining_bytes, total_bytes) end # Create a new connection due to some weirdness somewhere in Excon response = Excon.post( "#{Api::ENDPOINT}/apis/launcher/1/get-package", tcp_nodelay: true, headers: headers, body: "data=#{JSON.dump({ category: package.category, subcategory: package.subcategory, name: package.name, version: package.version })}", chunk_size: 50_000, response_block: streamer ) response.status == 200 || response.status == 206 ensure file&.close end def self.acquire_net_lock(key) Store["net_locks"] ||= {} if Store["net_locks"][key] false else Store["net_locks"][key] = true end end def self.release_net_lock(key) Store["net_locks"] ||= {} if Store["net_locks"][key] Store["net_locks"].delete(key) else warn "!!! net_lock not found for #{key.inspect}" end end def self.net_lock?(key) Store["net_locks"] ||= {} Store["net_locks"][key] end end end