#!/usr/bin/env python

import json
import os
import re
import subprocess
import sys
import argparse


class CrashLogPatcher:

    SYMBOL_REGEX = re.compile(r'^([0-9a-fA-F]+) T _(.*)$')
    UUID_REGEX = re.compile(r'UUID: ([-0-9a-fA-F]+) \(([^\(]+)\) .*')

    def __init__(self, data, binary, offsets, json):
        self.data = data
        self.binary = binary
        self.offsets = offsets
        self.json = json

    def patch_executable(self):
        self.data = self.data.replace("@EXEC@", self.binary)
        self.data = self.data.replace("@NAME@", os.path.basename(self.binary))

    def patch_uuid(self):
        output = subprocess.check_output(['dwarfdump', '--uuid', self.binary]).decode("utf-8")
        m = self.UUID_REGEX.match(output)
        if m:
            self.data = self.data.replace("@UUID@", m.group(1))

    def patch_addresses(self):
        if not self.offsets:
            return
        output = subprocess.check_output(['nm', self.binary]).decode("utf-8")
        for line in output.splitlines():
            m = self.SYMBOL_REGEX.match(line)
            if m:
                address = m.group(1)
                symbol = m.group(2)
                if symbol in self.offsets:
                    patch_addr = int(m.group(1), 16) + int(
                        self.offsets[symbol])
                    if self.json:
                        patch_addr = patch_addr - 0x100000000
                        representation = int
                    else:
                        representation = hex
                    self.data = self.data.replace(
                        "@{}@".format(symbol), str(representation(patch_addr)))


if __name__ == '__main__':
    parser = argparse.ArgumentParser(description='Crashlog Patcher')
    parser.add_argument('--binary', required=True)
    parser.add_argument('--crashlog', required=True)
    parser.add_argument('--offsets', required=True)
    parser.add_argument('--json', default=False, action='store_true')
    args = parser.parse_args()

    offsets = json.loads(args.offsets)

    with open(args.crashlog, 'r') as file:
        data = file.read()

    p = CrashLogPatcher(data, args.binary, offsets, args.json)
    p.patch_executable()
    p.patch_uuid()
    p.patch_addresses()

    with open(args.crashlog, 'w') as file:
        file.write(p.data)
