A patch parsing library.
Project description
What The Patch!?
What The Patch!? is a library for both parsing and applying patch files.
Features
Parsing of almost all diff formats (except forwarded ed):
normal (default, –normal)
copied context (-c, –context)
unified context (-u, –unified)
ed script (-e, –ed)
rcs ed script (-n, –rcs)
Parsing of several SCM patches:
CVS
SVN
Git
Installation
To install What The Patch!?, simply:
$ pip install whatthepatch
Usage
Let us say we have a patch file containing some changes, aptly named ‘somechanges.patch’:
--- lao 2012-12-26 23:16:54.000000000 -0600
+++ tzu 2012-12-26 23:16:50.000000000 -0600
@@ -1,7 +1,6 @@
-The Way that can be told of is not the eternal Way;
-The name that can be named is not the eternal name.
The Nameless is the origin of Heaven and Earth;
-The Named is the mother of all things.
+The named is the mother of all things.
+
Therefore let there always be non-being,
so we may see their subtlety,
And let there always be being,
@@ -9,3 +8,6 @@
The two are the same,
But after they are produced,
they have different names.
+They both may be called deep and profound.
+Deeper and more profound,
+The door of all subtleties!
Parsing
Here is how we would use What The Patch!? in Python to get the changeset for each diff in the patch:
>>> import whatthepatch
>>> import pprint
>>> with open('tests/casefiles/diff-unified.diff') as f:
... text = f.read()
...
>>> for diff in whatthepatch.parse_patch(text):
... print(diff) # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
...
diff(header=header(index_path=None,
old_path='lao',
old_version='2013-01-05 16:56:19.000000000 -0600',
new_path='tzu',
new_version='2013-01-05 16:56:35.000000000 -0600'),
changes=[Change(old=1, new=None, hunk=1, line='The Way that can be told of is not the eternal Way;'),
Change(old=2, new=None, hunk=1, line='The name that can be named is not the eternal name.'),
Change(old=3, new=1, hunk=1, line='The Nameless is the origin of Heaven and Earth;'),
Change(old=4, new=None, hunk=1, line='The Named is the mother of all things.'),
Change(old=None, new=2, hunk=1, line='The named is the mother of all things.'),
Change(old=None, new=3, hunk=1, line=''), Change(old=5, new=4, hunk=1, line='Therefore let there always be non-being,'),
Change(old=6, new=5, hunk=1, line=' so we may see their subtlety,'),
Change(old=7, new=6, hunk=1, line='And let there always be being,'),
Change(old=9, new=8, hunk=2, line='The two are the same,'),
Change(old=10, new=9, hunk=2, line='But after they are produced,'),
Change(old=11, new=10, hunk=2, line=' they have different names.'),
Change(old=None, new=11, hunk=2, line='They both may be called deep and profound.'),
Change(old=None, new=12, hunk=2, line='Deeper and more profound,'),
Change(old=None, new=13, hunk=2, line='The door of all subtleties!')],
text='...')
The changes are listed as they are in the patch, but instead of the +/- syntax of the patch, we get a tuple of two numbers and the text of the line. What these numbers indicate are as follows:
( old=1, new=None, ... ) indicates line 1 of the file lao was removed.
( old=None, new=2, ... ) indicates line 2 of the file tzu was inserted.
( old=5, new=4, ... ) indicates that line 5 of lao and line 4 of tzu are equal.
Please note that not all patch formats provide the actual lines modified, so some results will have the text portion of the tuple set to None.
Applying
To apply a diff to some lines of text, first read the patch and parse it.
>>> import whatthepatch
>>> with open('tests/casefiles/diff-default.diff') as f:
... text = f.read()
...
>>> with open('tests/casefiles/lao') as f:
... lao = f.read()
...
>>> diff = [x for x in whatthepatch.parse_patch(text)]
>>> diff = diff[0]
>>> tzu = whatthepatch.apply_diff(diff, lao)
>>> tzu # doctest: +NORMALIZE_WHITESPACE
['The Nameless is the origin of Heaven and Earth;',
'The named is the mother of all things.',
'',
'Therefore let there always be non-being,',
' so we may see their subtlety,',
'And let there always be being,',
' so we may see their outcome.',
'The two are the same,',
'But after they are produced,',
' they have different names.',
'They both may be called deep and profound.',
'Deeper and more profound,',
'The door of all subtleties!']
Contribute
Fork this repository
Create a new branch to work on
Commit your tests and/or changes
Push and create a pull request here!
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Hashes for whatthepatch-graingert-0.0.5.tar.gz
Algorithm | Hash digest | |
---|---|---|
SHA256 | a7643be700dbb127c6bf506fc99e8123c447c394454c538e23624a35e10c370f |
|
MD5 | 8d57ab0b95fb3b355bf41d16a19d7398 |
|
BLAKE2b-256 | da7bfc1c89abc346104941ae58dd5c3739bf2df27ffa0384e0c3b37e5e726f2d |
Hashes for whatthepatch_graingert-0.0.5-py2.py3-none-any.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | c3d11302583add7becd2b4e8fb126b59064d42abceea9ead79a0fffb4afcbf8a |
|
MD5 | 713a2750aadb41a235209acd3cffafa1 |
|
BLAKE2b-256 | ab2490fd206598f06181b6cdbfdb67dcc0d9327c210334f74ca06f968475f4b9 |