| #!/usr/bin/env python |
| # |
| # Copyright (C) 2019 The Android Open Source Project |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| |
| |
| import argparse |
| import collections |
| import json |
| import sys |
| |
| def follow_path(obj, path): |
| cur = obj |
| last_key = None |
| for key in path.split('.'): |
| if last_key: |
| if last_key not in cur: |
| return None,None |
| cur = cur[last_key] |
| last_key = key |
| if last_key not in cur: |
| return None,None |
| return cur, last_key |
| |
| |
| def ensure_path(obj, path): |
| cur = obj |
| last_key = None |
| for key in path.split('.'): |
| if last_key: |
| if last_key not in cur: |
| cur[last_key] = dict() |
| cur = cur[last_key] |
| last_key = key |
| return cur, last_key |
| |
| |
| class SetValue(str): |
| def apply(self, obj, val): |
| cur, key = ensure_path(obj, self) |
| cur[key] = val |
| |
| |
| class Replace(str): |
| def apply(self, obj, val): |
| cur, key = follow_path(obj, self) |
| if cur: |
| cur[key] = val |
| |
| |
| class Remove(str): |
| def apply(self, obj): |
| cur, key = follow_path(obj, self) |
| if cur: |
| del cur[key] |
| |
| |
| class AppendList(str): |
| def apply(self, obj, *args): |
| cur, key = ensure_path(obj, self) |
| if key not in cur: |
| cur[key] = list() |
| if not isinstance(cur[key], list): |
| raise ValueError(self + " should be a array.") |
| cur[key].extend(args) |
| |
| # A JSONDecoder that supports line comments start with // |
| class JSONWithCommentsDecoder(json.JSONDecoder): |
| def __init__(self, **kw): |
| super().__init__(**kw) |
| |
| def decode(self, s: str): |
| s = '\n'.join(l for l in s.split('\n') if not l.lstrip(' ').startswith('//')) |
| return super().decode(s) |
| |
| def main(): |
| parser = argparse.ArgumentParser() |
| parser.add_argument('-o', '--out', |
| help='write result to a file. If omitted, print to stdout', |
| metavar='output', |
| action='store') |
| parser.add_argument('input', nargs='?', help='JSON file') |
| parser.add_argument("-v", "--value", type=SetValue, |
| help='set value of the key specified by path. If path doesn\'t exist, creates new one.', |
| metavar=('path', 'value'), |
| nargs=2, dest='patch', default=[], action='append') |
| parser.add_argument("-s", "--replace", type=Replace, |
| help='replace value of the key specified by path. If path doesn\'t exist, no op.', |
| metavar=('path', 'value'), |
| nargs=2, dest='patch', action='append') |
| parser.add_argument("-r", "--remove", type=Remove, |
| help='remove the key specified by path. If path doesn\'t exist, no op.', |
| metavar='path', |
| nargs=1, dest='patch', action='append') |
| parser.add_argument("-a", "--append_list", type=AppendList, |
| help='append values to the list specified by path. If path doesn\'t exist, creates new list for it.', |
| metavar=('path', 'value'), |
| nargs='+', dest='patch', default=[], action='append') |
| args = parser.parse_args() |
| |
| if args.input: |
| with open(args.input) as f: |
| obj = json.load(f, object_pairs_hook=collections.OrderedDict, cls=JSONWithCommentsDecoder) |
| else: |
| obj = json.load(sys.stdin, object_pairs_hook=collections.OrderedDict, cls=JSONWithCommentsDecoder) |
| |
| for p in args.patch: |
| p[0].apply(obj, *p[1:]) |
| |
| if args.out: |
| with open(args.out, "w") as f: |
| json.dump(obj, f, indent=2, separators=(',', ': ')) |
| f.write('\n') |
| else: |
| print(json.dumps(obj, indent=2, separators=(',', ': '))) |
| |
| |
| if __name__ == '__main__': |
| main() |