1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351
|
#!/usr/bin/env ruby
#
# Copyright
#
# Copyright (C) 2014-2025 Jari Aalto <jari.aalto@cante.net>
# Copyright (C) 2025 Eman Resu <elevenaka11@gmail.com>
# Copyright (C) 2007-2014 Peter Hutterer <peter.hutterer@who-t.net>
# Copyright (C) 2007-2014 Benjamin Close <Benjamin.Close@clearchain.com>
#
# License
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
# Description
#
# Splitpatch is a utility to split a patch up into
# multiple patch files. If the --hunks option is provided on the
# command line, each hunk is saved on its own patch file.
PROGRAM = "splitpatch"
VERSION = 2025.0210
LICENSE = "GPL-2.0-or-later" # See official acronyms: https://spdx.org/licenses/
HOMEPAGE = "https://github.com/jaalto/splitpatch"
class Splitter
def initialize(file, encode)
@encode = encode
@filename = file
@fullname = false
end
def fullname(opt)
@fullname = opt
end
def validFile?
return File.exist?(@filename) && File.readable?(@filename)
end
def createFile(filename)
if File.exist?(filename)
puts "File #{filename} already exists. Renaming patch."
appendix = 0
zero = appendix.to_s.rjust(3, '0')
while File.exist?("#{filename}.#{zero}")
appendix += 1
zero = appendix.to_s.rjust(3, '0')
end
filename << ".#{zero}"
end
return open(filename, "w")
end
def getFilenameByHeader(header)
filename = getFilename(header[0])
if (@fullname && filename == 'dev-null') ||
(! @fullname && filename == 'null')
filename = getFilename(header[1])
end
filename
end
def getFilename(line)
tokens = line.split(" ")
tokens = tokens[1].split(":")
tokens = tokens[0].split("/")
if @fullname
tokens.reject!(&:empty?)
return tokens.join('-')
else
return tokens[-1]
end
end
# Specifically for git patches
def getGitFilename(line)
folder = line.split.last
if @fullname
filename = folder.split("/").drop(1).join("-")
else
filename = folder.split("/").last
end
return filename
end
# Split the patchfile by files
def splitByFile
legacy = false
git = false
outfile = nil
stream = open(@filename, 'rb')
until (stream.eof?)
line = stream.readline
# We need to create a new file
if (line =~ /^Index: .*/) == 0
# Patch includes Index lines
# Drop into "legacy mode"
legacy = true
if (outfile)
outfile.close_write
end
filename = getFilename(line)
filename << ".patch"
outfile = createFile(filename)
outfile.write(line)
elsif (line =~ /^diff --git .*/) == 0 and not legacy
git = true
if (outfile)
outfile.close_write
end
outfile = nil
filename = getGitFilename(line)
filename << ".patch"
outfile = createFile(filename)
outfile.write(line)
# This line will show up in a git patch, but it shouldn't mark the start of the patch
elsif (line =~ /--- .*/) == 0 and not legacy and not git
if (outfile)
outfile.close_write
end
# Find filename
# Next line is header too
header = [ line, stream.readline ]
filename = getFilenameByHeader(header)
filename << ".patch"
outfile = createFile(filename)
outfile.write(header.join(''))
else
if outfile
outfile.write(line)
end
end
end
end
def splitByHunk
legacy = false
git = false
in_git_header = false
outfile = nil
stream = open(@filename, 'rb')
filename = ""
counter = 0
header = []
until (stream.eof?)
line = stream.readline.encode("UTF-8", @encode)
# We need to create a new file
if (line =~ /^Index: .*/) == 0
# Patch includes Index lines
# Drop into "legacy mode"
legacy = true
filename = getFilename(line)
header << line
# Remaining 3 lines of header
for i in 0..2
line = stream.readline
header << line
end
counter = 0
# Start of a new git patch
elsif (line =~ /^diff --git .*/) == 0
# If we hit this, the previous git patch had no hunk, so the file never started
if (in_git_header)
hunklessfilename = "#{filename}.000.patch"
outfile = createFile(hunklessfilename)
outfile.write(header.join('')) # Header ended up storing the entire patch
end
git = true
header = [ line ]
filename = getGitFilename(line)
counter = 0
# Future lines will be within a header until we reach a hunk, new patch, or EOF
in_git_header = true
elsif (line =~ /--- .*/) == 0 and not legacy and not git
#find filename
# next line is header too
header = [ line, stream.readline ]
filename = getFilenameByHeader(header)
counter = 0
elsif (line =~ /@@ .* @@/) == 0
in_git_header = false
if (outfile)
outfile.close_write
end
zero = counter.to_s.rjust(3, '0')
hunkfilename = "#{filename}.#{zero}.patch"
outfile = createFile(hunkfilename)
counter += 1
outfile.write(header.join(''))
outfile.write(line)
# We haven't found a hunk, new patch, or EOF yet
elsif (in_git_header)
header << line
else
if outfile
outfile.write(line)
end
end
end
# Last patch in file had no hunk, hit EOF before we could write everything in header
if (in_git_header)
hunklessfilename = "#{filename}.000.patch"
outfile = createFile(hunklessfilename)
outfile.write(header.join(''))
end
end
end
def help
puts <<EOF
SYNOPSIS
#{PROGRAM} [options] FILE.patch
OPTIONS
-e=ENCODING, --encode=ENCODING
Read file and save patch hunks using ENCODING. Default is 'UTF-8'.
-f, --fullname
Use full name upon saving patch hunks
-h, --help
Show short help. This page.
-H, --hunk
Split by hunks.
-V, --version
Display version, licence and homepage.
DESCRIPTION
Split the patch up into files or hunks
Divide a patch or diff file into pieces. The split can made by file
or by hunk basis. This makes it possible to separate changes that
might not be desirable or assemble the patch into a more coherent set
of changes. See e.g. combinediff(1) from patchutils package.
Note: only patches in unified format are recognized.
AUTHORS
Peter Hutterer (orig. Author) <peter.hutterer@who-t.net>
Benjamin Close (orig. Author) <Benjamin.Close@clearchain.com>
Jari Aalto (Maintainer) <jari.aalto@cante.net>"
Homepage: #{HOMEPAGE}
EOF
end
def version
puts "#{VERSION} #{LICENSE} #{HOMEPAGE}"
end
def parsedOptions
if ARGV.length < 1
puts "ERROR: missing argument. See --help."
exit 1
end
opts = {
encode: "UTF-8"
}
ARGV.each do |opt|
case opt
when /^-e=(.+?)$/, /^--encode=(.+?)$/
opts[:encode] = $~[1]
when /^-f$/, /--fullname/
opts[:fullname] = true
when /^-h$/, /--help/
opts[:help] = true
when /^-H$/, /--hunks?/
opts[:hunk] = true
when /^-V$/, /--version/
opts[:version] = true
when /^-/
puts "ERROR: Unknown option: #{opt}. See --help."
exit 1
else
opts[:file] = opt
end
end
return opts
end
def main
opts = parsedOptions
if opts[:help]
help
exit
end
if opts[:version]
version
exit
end
s = Splitter.new(opts[:file], opts[:encode])
s.fullname(true) if opts[:fullname]
if !s.validFile?
puts "File does not exist or is not readable: #{opts[:file]}"
end
if opts[:hunk]
s.splitByHunk
else
s.splitByFile
end
end
main
# End of file
|