diff --git a/README.md b/README.md index 02f0d13..dbc9fb1 100644 --- a/README.md +++ b/README.md @@ -21,6 +21,8 @@ Discussions on this topic can be found here: https://github.com/jacobsalmela/tcc - `tccutil.py` can be installed without any additional software. - it has an easy to use syntax +- it supports both system wide and user scope TCC manipulation +- it wraps the native `/usr/bin/tccutil` tool - there are other solutions out there, but there were some things I did not like about them: + [Privacy Manager Services](https://github.com/univ-of-utah-marriott-library-apple/privacy_services_manager) has other dependencies that need to be installed (it has also gone over five years without any updates) @@ -58,34 +60,36 @@ Clone this repo and manually copy `tccutil.py` to `/usr/local/bin` or run from a **This utility needs super-user priveleges for most operations.** It is important that you either run this as root or use `sudo`, otherwise it won't work and you will end up with “permission denied” errors. + ``` -usage: tccutil.py [-h] [--service SERVICE] [--list] [--insert INSERT] [-v] - [-r REMOVE] [-e ENABLE] [-d DISABLE] [--version] +usage: tccutil.py [-h] [--service SERVICE] [--list] [--digest] [--insert INSERT] [-v] + [-r REMOVE] [-e ENABLE] [-d DISABLE] [--user [USER]] [--version] [ACTION] Modify Accesibility Preferences positional arguments: - ACTION This option is only used to perform a reset. + ACTION This option is only used to perform a reset, using "/usr/bin/tccutil". See + `man tccutil` for additional syntax optional arguments: -h, --help show this help message and exit --service SERVICE, -s SERVICE Set TCC service - --list, -l List all entries in the accessibility database. + --list, -l List all entries in the accessibility database + --digest Print the digest hash of the accessibility database --insert INSERT, -i INSERT - Adds the given bundle ID or path to the accessibility - database. - -v, --verbose Outputs additional info for some commands. + Adds the given bundle ID or path to the accessibility database + -v, --verbose Outputs additional info for some commands -r REMOVE, --remove REMOVE - Removes a given Bundle ID or Path from the - Accessibility Database. + Removes a given Bundle ID or Path from the Accessibility Database -e ENABLE, --enable ENABLE - Enables Accessibility Access for the given Bundle ID - or Path. + Enables Accessibility Access for the given Bundle ID or Path -d DISABLE, --disable DISABLE - Disables Accessibility Access for the given Bundle ID - or Path. + Disables Accessibility Access for the given Bundle ID or Path + --user [USER], -u [USER] + Modify accessibility database for a given user (defaults to current, + if no additional parameter is provided) --version Show the version of this script ``` @@ -98,12 +102,24 @@ List existing Entries in the Accessibility Database sudo tccutil.py --list ``` +List existing Entries in the Accessibility Database specific to the current user + +```bash +sudo tccutil.py --list -u +``` + Add `/usr/bin/osascript` to the Accessibility Database (using UNIX-Style Option) ```bash sudo tccutil.py -i /usr/bin/osascript ```` +Add `/usr/bin/osascript` to the Accessibility Database specific to user 'myuser' (using UNIX-Style Option) + +```bash +sudo tccutil.py -i /usr/bin/osascript -u myuser +```` + Add *Script Editor* to the Accessibility Database (using Long Option) ```bash @@ -128,6 +144,12 @@ Disable `/usr/bin/osascript` (must already exist in the Database) sudo tccutil.py -d /usr/bin/osascript ``` +Reset system wide accessibility database + +```bash +sudo tccutil.py reset ALL +``` + ## Contributing Many people have contributed already, so feel free to make a PR and we'll get it merged in. diff --git a/tccutil.py b/tccutil.py index 7143151..5c8357b 100755 --- a/tccutil.py +++ b/tccutil.py @@ -4,6 +4,7 @@ # tccutil.py, Utility to modify the macOS Accessibility Database (TCC.db) # # Copyright (C) 2020, @jacobsalmela +# Copyright (C) 2023, @tnarik # # This is free software; you can redistribute it and/or modify it under the # terms of the GNU General Public License, either Version 2 or any later @@ -15,6 +16,7 @@ import sqlite3 import sys import os +import pwd import hashlib from platform import mac_ver from packaging.version import Version as version @@ -23,12 +25,12 @@ util_name = os.path.basename(sys.argv[0]) # Utility Version -util_version = '1.2.13' +util_version = '1.4.0' # Current OS X version osx_version = version(mac_ver()[0]) # mac_ver() returns 10.16 for Big Sur instead 11.+ -# Database Path +# Database Path (System by default) tcc_db = '/Library/Application Support/com.apple.TCC/TCC.db' # Set "sudo" to True if called with Admin-Privileges. @@ -38,8 +40,7 @@ verbose = False # TCC Service -service = "kTCCServiceAccessibility" - +default_service = "kTCCServiceAccessibility" parser = argparse.ArgumentParser(description='Modify Accesibility Preferences') parser.add_argument( @@ -47,40 +48,49 @@ metavar='ACTION', type=str, nargs='?', - help='This option is only used to perform a reset.', + help='This option is only used to perform a reset, using "/usr/bin/tccutil".\ + See `man tccutil` for additional syntax', ) parser.add_argument( '--service', '-s', - default=service, + default=default_service, help="Set TCC service" ) + parser.add_argument( '--list', '-l', action='store_true', - help="List all entries in the accessibility database." + help="List all entries in the accessibility database" ) parser.add_argument( '--digest', action='store_true', - help="Print the digest hash of the accessibility database." + help="Print the digest hash of the accessibility database" ) parser.add_argument( '--insert', '-i', action='append', default=[], - help="Adds the given bundle ID or path to the accessibility database.", + help="Adds the given bundle ID or path to the accessibility database", ) parser.add_argument( "-v", "--verbose", action='store_true', - help="Outputs additional info for some commands.", + help="Outputs additional info for some commands", ) parser.add_argument( "-r", "--remove", action='append', default=[], - help="Removes a given Bundle ID or Path from the Accessibility Database.", + help="Removes a given Bundle ID or Path from the Accessibility Database", ) parser.add_argument( "-e", "--enable", action='append', default=[], - help="Enables Accessibility Access for the given Bundle ID or Path.", + help="Enables Accessibility Access for the given Bundle ID or Path", ) parser.add_argument( "-d", "--disable", action='append', default=[], - help="Disables Accessibility Access for the given Bundle ID or Path." + help="Disables Accessibility Access for the given Bundle ID or Path" +) +parser.add_argument( + '--user', '-u', + nargs="?", + const='', + default=None, + help="Modify accessibility database for a given user (defaults to current, if no additional parameter is provided)", ) parser.add_argument( '--version', action='store_true', @@ -89,15 +99,15 @@ def display_version(): """Print the version of this utility.""" - print("%s %s" % (util_name, util_version)) + print(f"{util_name} {util_version}") sys.exit(0) def sudo_required(): """Check if user has root priveleges to access the database.""" if not sudo: - print("Error:") - print(" When accessing the Accessibility Database, %s needs to be run with admin-privileges.\n" % (util_name)) + print("Error:", file=sys.stderr) + print(f" When accessing the Accessibility Database, {util_name} needs to be run with admin-privileges.\n", file=sys.stderr) display_help(1) @@ -147,13 +157,17 @@ def open_database(digest=False): accessTableDigest in ["ecc443615f", "80a4bb6912"]) or # Big Sur and later (osx_version >= version('10.16') and - accessTableDigest in ["3d1c2a0e97", "cef70648de"])): - print("TCC Database structure is unknown (%s)" % accessTableDigest) + accessTableDigest in ["3d1c2a0e97", "cef70648de"]) or + # Sonoma + (osx_version >= version('14.0') and + accessTableDigest in ["34abf99d20"]) + ): + print(f"TCC Database structure is unknown ({accessTableDigest})", file=sys.stderr) sys.exit(1) verbose_output("Database opened.\n") except TypeError: - print("Error opening Database. You probably need to disable SIP for this to work.") + print("Error opening Database. You probably need to disable SIP for this to work.", file=sys.stderr) sys.exit(1) @@ -162,7 +176,7 @@ def display_help(error_code=None): parser.print_help() if error_code is not None: sys.exit(error_code) - print("%s %s" % (util_name, util_version)) + print(f"{util_name} {util_version}") sys.exit(0) @@ -178,7 +192,7 @@ def close_database(): except: verbose_output("Database closed.") except: - print("Error closing Database.") + print("Error closing Database.", file=sys.stderr) sys.exit(1) except: pass @@ -203,7 +217,7 @@ def verbose_output(*args): def list_clients(): """List items in the database.""" open_database() - c.execute("SELECT client from access WHERE service is '%s'" % service) + c.execute(f"SELECT client from access WHERE service is '{service}'") verbose_output("Fetching Entries from Database...\n") for row in c.fetchall(): # print each entry in the Accessibility pane. @@ -213,18 +227,18 @@ def list_clients(): def cli_util_or_bundle_id(client): """Check if the item is a path or a bundle ID.""" - global client_type # If the app starts with a slash, it is a command line utility. # Setting the client_type to 1 will make the item visible in the # GUI so you can manually click the checkbox. if client[0] == '/': client_type = 1 - verbose_output("Detected \"%s\" as Command Line Utility." % (client)) + verbose_output(f'Detected "{client}" as Command Line Utility.') # Otherwise, the app will be a bundle ID, which starts # with a com., net., or org., etc. else: client_type = 0 - verbose_output("Detected \"%s\" as Bundle ID." % (client)) + verbose_output(f'Detected "{client}" as Bundle ID.') + return client_type def insert_client(client): @@ -232,68 +246,70 @@ def insert_client(client): open_database() # Check if it is a command line utility or a bundle ID # as the default value to enable it is different. - cli_util_or_bundle_id(client) - verbose_output("Inserting \"%s\" into Database..." % (client)) - # Big Sur and later + client_type = cli_util_or_bundle_id(client) + verbose_output(f'Inserting "{client}" into Database...') + # Sonoma if osx_version >= version('10.16'): try: - c.execute("INSERT or REPLACE INTO access VALUES('%s','%s',%s,2,4,1,NULL,NULL,0,'UNUSED',NULL,0,0)" - % (service, client, client_type)) + c.execute(f"INSERT or REPLACE INTO access VALUES('{service}','{client}',{client_type},2,4,1,NULL,NULL,0,'UNUSED',NULL,0, NULL, NULL, NULL,'UNUSED', NULL)") except sqlite3.OperationalError: - print("Attempting to write a readonly database. You probably need to disable SIP.") + print("Attempting to write a readonly database. You probably need to disable SIP.", file=sys.stderr) + # Big Sur and later + elif osx_version >= version('10.16'): + try: + c.execute(f"INSERT or REPLACE INTO access VALUES('{service}','{client}',{client_type},2,4,1,NULL,NULL,0,'UNUSED',NULL,0,0)") + except sqlite3.OperationalError: + print("Attempting to write a readonly database. You probably need to disable SIP.", file=sys.stderr) # Mojave through Big Sur elif osx_version >= version('10.14'): - c.execute("INSERT or REPLACE INTO access VALUES('%s','%s',%s,1,1,NULL,NULL,NULL,'UNUSED',NULL,0,0)" - % (service, client, client_type)) + c.execute(f"INSERT or REPLACE INTO access VALUES('{service}','{client}',{client_type},1,1,NULL,NULL,NULL,'UNUSED',NULL,0,0)") # El Capitan through Mojave elif osx_version >= version('10.11'): - c.execute("INSERT or REPLACE INTO access VALUES('%s','%s',%s,1,1,NULL,NULL)" - % (service, client, client_type)) + c.execute(f"INSERT or REPLACE INTO access VALUES('{service}','{client}',{client_type},1,1,NULL,NULL)") # Yosemite or lower else: - c.execute("INSERT or REPLACE INTO access VALUES('%s','%s',%s,1,1,NULL)" - % (service, client, client_type)) + c.execute(f"INSERT or REPLACE INTO access VALUES('{service}','{client}',{client_type},1,1,NULL)") commit_changes() def delete_client(client): """Remove a client from the database.""" open_database() - verbose_output("Removing \"%s\" from Database..." % (client)) + verbose_output(f'Removing "{client}" from Database...') try: - c.execute("DELETE from access where client IS '%s' AND service IS '%s'" % (client, service)) + c.execute(f"DELETE from access where client IS '{client}' AND service IS '{service}'") except sqlite3.OperationalError: - print("Attempting to write a readonly database. You probably need to disable SIP.") + print("Attempting to write a readonly database. You probably need to disable SIP.", file=sys.stderr) commit_changes() def enable(client): """Add a client from the database.""" open_database() - verbose_output("Enabling %s..." % (client,)) + verbose_output(f'Enabling {client}...') # Setting typically appears in System Preferences # right away (without closing the window). # Set to 1 to enable the client. enable_mode_name = 'auth_value' if osx_version >= version('10.16') else 'allowed' try: - c.execute("UPDATE access SET %s='1' WHERE client='%s' AND service IS '%s'" % (enable_mode_name, client, service)) + c.execute(f"UPDATE access SET {enable_mode_name}='1' WHERE client='{client}' AND service IS '{service}'") except sqlite3.OperationalError: - print("Attempting to write a readonly database. You probably need to disable SIP.") + print("Attempting to write a readonly database. You probably need to disable SIP.", file=sys.stderr) commit_changes() def disable(client): """Disable a client in the database.""" open_database() - verbose_output("Disabling %s..." % (client,)) + verbose_output(f"Disabling {client}...") # Setting typically appears in System Preferences # right away (without closing the window). # Set to 0 to disable the client. enable_mode_name = 'auth_value' if osx_version >= version('10.16') else 'allowed' try: - c.execute("UPDATE access SET %s='0' WHERE client='%s' AND service IS '%s'" % (enable_mode_name, client, service)) + c.execute(f"UPDATE access SET {enable_mode_name}='0' WHERE client='{client}' AND service IS '{service}'") except sqlite3.OperationalError: - print("Attempting to write a readonly database. You probably need to disable SIP.") + print("Attempting to write a readonly database. You probably need to disable SIP.", file=sys.stderr) commit_changes() @@ -305,19 +321,27 @@ def main(): print(" No arguments.\n") display_help(2) - args = parser.parse_args() + args, rest = parser.parse_known_args() if args.version: display_version() return + global tcc_db + if args.user != None: + try: + if (len(args.user) > 0): pwd.getpwnam(args.user) + tcc_db = os.path.abspath(os.path.expanduser(f'~{args.user}/{tcc_db}')) + except KeyError: + print(f'User "{args.user}" does not exist. Do you mean to use "{args.user}" as ACTION?', file=sys.stderr) + sys.exit(1) + if args.action: if args.action == 'reset': - exit_status = os.system("tccutil \ -{}".format(' '.join(sys.argv[1:]))) + exit_status = os.system(f'/usr/bin/tccutil -v {" ".join(rest)}') sys.exit(exit_status / 256) else: - print("Error\n Unrecognized command {}".format(args.action)) + print(f'Error\n Unrecognized command "{args.action}"', file=sys.stderr) global service service = args.service