# frozen_string_literal: true # A project file is a file within a project that contains license information # Currently extended by LicenseFile, PackageManagerFile, and ReadmeFile # # Sublcasses should implement the possible_matchers method module Licensee module ProjectFiles class ProjectFile extend Forwardable def_delegator :@data, :[] attr_reader :content include Licensee::HashHelper HASH_METHODS = %i[ filename content content_hash content_normalized matcher matched_license attribution ].freeze ENCODING = Encoding::UTF_8 ENCODING_OPTIONS = { invalid: :replace, undef: :replace, replace: '' }.freeze # Create a new Licensee::ProjectFile with content and metadata # # content - file content # metadata - can be either the string filename, or a hash containing # metadata about the file content. If a hash is given, the # filename should be given using the :name key. See individual # project types for additional available metadata # # Returns a new Licensee::ProjectFile def initialize(content, metadata = {}) @content = content.dup @content.force_encoding(ENCODING) @content.encode!(ENCODING, **ENCODING_OPTIONS) unless @content.valid_encoding? @content.encode!(ENCODING, universal_newline: true) metadata = { name: metadata } if metadata.is_a? String @data = metadata || {} end # TODO: In the next major release, filename should be the basename # and path should be either the absolute path or the relative path to # the project root, but maintaining the alias for backward compatability def filename @data[:name] end alias path filename def directory @data[:dir] || '.' end alias dir directory def path_relative_to_root File.join(directory, filename) end alias relative_path path_relative_to_root def possible_matchers raise 'Not implemented' end def matcher @matcher ||= possible_matchers.map { |m| m.new(self) }.find(&:match) end # Returns the percent confident with the match def confidence matcher&.confidence end def license matcher&.match end alias match license def matched_license license&.spdx_id end # Is this file a COPYRIGHT file with only a copyright statement? # If so, it can be excluded from determining if a project has >1 license def copyright? return false unless is_a?(LicenseFile) return false unless matcher.is_a?(Matchers::Copyright) filename =~ /\Acopyright(?:#{LicenseFile::OTHER_EXT_REGEX})?\z/io end def content_hash nil end def content_normalized nil end def attribution nil end end end end