# Copyright, 2018, by Samuel G. D. Williams. <http://www.codeotaku.com># # Permission is hereby granted, free of charge, to any person obtaining a copy# of this software and associated documentation files (the "Software"), to deal# in the Software without restriction, including without limitation the rights# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell# copies of the Software, and to permit persons to whom the Software is# furnished to do so, subject to the following conditions:# # The above copyright notice and this permission notice shall be included in# all copies or substantial portions of the Software.# # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN# THE SOFTWARE.require_relative'wrapper'require'msgpack'require'time'require'set'moduleCoveredclassPersist<WrapperDEFAULT_PATH=".covered.db"definitialize(output,path=DEFAULT_PATH)super(output)@path=path@touched=Set.newenddefapply(record)# The file must still exist:returnunlesspath=expand_path(record[:path])returnunlessFile.exist?path# If the file has been modified since... we can't use the coverage.returnunlessmtime=record[:mtime]returnifFile.mtime(path).to_f>record[:mtime]record[:coverage].each_with_indexdo|count,index|@output.mark(path,index,count)ifcountendenddefserialize(coverage){# We want to use relative paths so that moving the repo won't break everything:path: relative_path(coverage.path),coverage: coverage.counts,mtime: File.mtime(coverage.path).to_f,}enddefload!(path=@path)returnunlessFile.exist?(@path)# Load existing coverage information and mark all files:File.open(@path,"r")do|file|file.flock(File::LOCK_SH)make_unpacker(file).each(&self.method(:apply))endenddefsave!(path=@path)# Dump all coverage:File.open(@path,"w")do|file|file.flock(File::LOCK_EX)packer=make_packer(file)self.eachdo|coverage|packer.write(serialize(coverage))endpacker.flushendenddefmark(file,line,count)@touched<<filesuperenddefenablesuperload!enddefdisablesuper# @touched.each do |path|# if @output.accept?(path)# puts "Updated #{path} coverage."# end# endsave!end# def each# super do |coverage|# if @touched.include?(coverage.path)# yield coverage# end# end# enddefmake_packer(io)packer=MessagePack::Packer.new(io)packer.register_type(0x00,Symbol,:to_msgpack_ext)packer.register_type(0x01,Time){|object|object.to_s}returnpackerenddefmake_unpacker(io)unpacker=MessagePack::Unpacker.new(io)unpacker.register_type(0x00,Symbol,:from_msgpack_ext)unpacker.register_type(0x01,Time,:parse)returnunpackerendendend