189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project/*
289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Copyright 2014, The Android Open Source Project
389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Licensed under the Apache License, Version 2.0 (the "License");
589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** you may not use this file except in compliance with the License.
689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** You may obtain a copy of the License at
789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**     http://www.apache.org/licenses/LICENSE-2.0
989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
1089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Unless required by applicable law or agreed to in writing, software
1189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** distributed under the License is distributed on an "AS IS" BASIS,
1289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** See the License for the specific language governing permissions and
1489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** limitations under the License.
1589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project*/
1689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
1789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectpackage com.android.commands.appops;
1889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
1989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport android.app.ActivityManager;
2089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport android.app.ActivityThread;
2189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport android.app.AppOpsManager;
2289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport android.content.Context;
2389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport android.content.pm.IPackageManager;
2489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport android.os.ServiceManager;
2589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport android.os.UserHandle;
2690100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten
2789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport android.util.TimeUtils;
2889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport com.android.internal.app.IAppOpsService;
2989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport com.android.internal.os.BaseCommand;
3089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
3189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport java.io.PrintStream;
3289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectimport java.util.List;
332db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huber
342db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huber/**
352db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huber * This class is a command line utility for manipulating AppOps permissions.
362db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huber */
3789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectpublic class AppOpsCommand extends BaseCommand {
381173118eace0e9e347cb007f0da817cee87579edGlenn Kasten
39247d9ebdbe7f165644f308745eee061fcbb77771Mark Salyzyn    public static void main(String[] args) {
401173118eace0e9e347cb007f0da817cee87579edGlenn Kasten        new AppOpsCommand().run(args);
4189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
4289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
4389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    @Override
4489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    public void onShowUsage(PrintStream out) {
4589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        out.println("usage: appops set [--user <USER_ID>] <PACKAGE> <OP> <MODE>\n"
4689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                + "       appops get [--user <USER_ID>] <PACKAGE> [<OP>]\n"
4789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                + "       appops reset [--user <USER_ID>] [<PACKAGE>]\n"
4889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                + "  <PACKAGE> an Android package name.\n"
4989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                + "  <OP>      an AppOps operation.\n"
5089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                + "  <MODE>    one of allow, ignore, deny, or default\n"
5189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                + "  <USER_ID> the user id under which the package is installed. If --user is not\n"
5289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                + "            specified, the current user is assumed.\n");
5389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
54247d9ebdbe7f165644f308745eee061fcbb77771Mark Salyzyn
55d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    private static final String COMMAND_SET = "set";
56d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    private static final String COMMAND_GET = "get";
57247d9ebdbe7f165644f308745eee061fcbb77771Mark Salyzyn    private static final String COMMAND_RESET = "reset";
584f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang
594f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang    @Override
60247d9ebdbe7f165644f308745eee061fcbb77771Mark Salyzyn    public void onRun() throws Exception {
614f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang        String command = nextArgRequired();
624f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang        switch (command) {
634f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang            case COMMAND_SET:
6489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                runSet();
6589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                break;
6689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
6789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            case COMMAND_GET:
6889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                runGet();
6989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                break;
7089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
7189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            case COMMAND_RESET:
7289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                runReset();
7389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                break;
7489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
7589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            default:
7689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                System.err.println("Error: Unknown command: '" + command + "'.");
7789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                break;
7889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
7989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
80fff6d715a8db0daf08a50634f242c40268de3d49Glenn Kasten
8189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    private static final String ARGUMENT_USER = "--user";
8289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
8389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // Modes
8489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    private static final String MODE_ALLOW = "allow";
8589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    private static final String MODE_DENY = "deny";
8690100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten    private static final String MODE_IGNORE = "ignore";
8790100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten    private static final String MODE_DEFAULT = "default";
8890100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten
8990100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten    private int strOpToOp(String op) {
9090100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        try {
9190100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten            return AppOpsManager.strOpToOp(op);
9290100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        } catch (IllegalArgumentException e) {
9390100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        }
9490100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        try {
9590100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten            return Integer.parseInt(op);
9690100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        } catch (NumberFormatException e) {
9790100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        }
9890100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        try {
9990100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten            return AppOpsManager.strDebugOpToOp(op);
10090100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        } catch (IllegalArgumentException e) {
10190100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten            System.err.println("Error: " + e.getMessage());
10290100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten            return -1;
10390100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        }
10490100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten    }
10590100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten
10690100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten    private void runSet() throws Exception {
10790100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        String packageName = null;
10890100b5573f95e8404c6e2917520e090fe8b49fdGlenn Kasten        String op = null;
10989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        String mode = null;
11089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        int userId = UserHandle.USER_CURRENT;
11189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        for (String argument; (argument = nextArg()) != null;) {
11289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            if (ARGUMENT_USER.equals(argument)) {
11389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                userId = Integer.parseInt(nextArgRequired());
114            } else {
115                if (packageName == null) {
116                    packageName = argument;
117                } else if (op == null) {
118                    op = argument;
119                } else if (mode == null) {
120                    mode = argument;
121                } else {
122                    System.err.println("Error: Unsupported argument: " + argument);
123                    return;
124                }
125            }
126        }
127
128        if (packageName == null) {
129            System.err.println("Error: Package name not specified.");
130            return;
131        } else if (op == null) {
132            System.err.println("Error: Operation not specified.");
133            return;
134        } else if (mode == null) {
135            System.err.println("Error: Mode not specified.");
136            return;
137        }
138
139        final int opInt = strOpToOp(op);
140        if (opInt < 0) {
141            return;
142        }
143        final int modeInt;
144        switch (mode) {
145            case MODE_ALLOW:
146                modeInt = AppOpsManager.MODE_ALLOWED;
147                break;
148            case MODE_DENY:
149                modeInt = AppOpsManager.MODE_ERRORED;
150                break;
151            case MODE_IGNORE:
152                modeInt = AppOpsManager.MODE_IGNORED;
153                break;
154            case MODE_DEFAULT:
155                modeInt = AppOpsManager.MODE_DEFAULT;
156                break;
157            default:
158                System.err.println("Error: Mode " + mode + " is not valid,");
159                return;
160        }
161
162        // Parsing complete, let's execute the command.
163
164        if (userId == UserHandle.USER_CURRENT) {
165            userId = ActivityManager.getCurrentUser();
166        }
167
168        final IPackageManager pm = ActivityThread.getPackageManager();
169        final IAppOpsService appOpsService = IAppOpsService.Stub.asInterface(
170                ServiceManager.getService(Context.APP_OPS_SERVICE));
171        final int uid = pm.getPackageUid(packageName, userId);
172        if (uid < 0) {
173            System.err.println("Error: No UID for " + packageName + " in user " + userId);
174            return;
175        }
176        appOpsService.setMode(opInt, uid, packageName, modeInt);
177    }
178
179    private void runGet() throws Exception {
180        String packageName = null;
181        String op = null;
182        int userId = UserHandle.USER_CURRENT;
183        for (String argument; (argument = nextArg()) != null;) {
184            if (ARGUMENT_USER.equals(argument)) {
185                userId = Integer.parseInt(nextArgRequired());
186            } else {
187                if (packageName == null) {
188                    packageName = argument;
189                } else if (op == null) {
190                    op = argument;
191                } else {
192                    System.err.println("Error: Unsupported argument: " + argument);
193                    return;
194                }
195            }
196        }
197
198        if (packageName == null) {
199            System.err.println("Error: Package name not specified.");
200            return;
201        }
202
203        final int opInt = op != null ? strOpToOp(op) : 0;
204
205        // Parsing complete, let's execute the command.
206
207        if (userId == UserHandle.USER_CURRENT) {
208            userId = ActivityManager.getCurrentUser();
209        }
210
211        final IPackageManager pm = ActivityThread.getPackageManager();
212        final IAppOpsService appOpsService = IAppOpsService.Stub.asInterface(
213                ServiceManager.getService(Context.APP_OPS_SERVICE));
214        final int uid = pm.getPackageUid(packageName, userId);
215        if (uid < 0) {
216            System.err.println("Error: No UID for " + packageName + " in user " + userId);
217            return;
218        }
219        List<AppOpsManager.PackageOps> ops = appOpsService.getOpsForPackage(uid, packageName,
220                op != null ? new int[] {opInt} : null);
221        if (ops == null || ops.size() <= 0) {
222            System.out.println("No operations.");
223            return;
224        }
225        final long now = System.currentTimeMillis();
226        for (int i=0; i<ops.size(); i++) {
227            List<AppOpsManager.OpEntry> entries = ops.get(i).getOps();
228            for (int j=0; j<entries.size(); j++) {
229                AppOpsManager.OpEntry ent = entries.get(j);
230                System.out.print(AppOpsManager.opToName(ent.getOp()));
231                System.out.print(": ");
232                switch (ent.getMode()) {
233                    case AppOpsManager.MODE_ALLOWED:
234                        System.out.print("allow");
235                        break;
236                    case AppOpsManager.MODE_IGNORED:
237                        System.out.print("ignore");
238                        break;
239                    case AppOpsManager.MODE_ERRORED:
240                        System.out.print("deny");
241                        break;
242                    case AppOpsManager.MODE_DEFAULT:
243                        System.out.print("default");
244                        break;
245                    default:
246                        System.out.print("mode=");
247                        System.out.print(ent.getMode());
248                        break;
249                }
250                if (ent.getTime() != 0) {
251                    System.out.print("; time=");
252                    StringBuilder sb = new StringBuilder();
253                    TimeUtils.formatDuration(now - ent.getTime(), sb);
254                    System.out.print(sb);
255                    System.out.print(" ago");
256                }
257                if (ent.getRejectTime() != 0) {
258                    System.out.print("; rejectTime=");
259                    StringBuilder sb = new StringBuilder();
260                    TimeUtils.formatDuration(now - ent.getRejectTime(), sb);
261                    System.out.print(sb);
262                    System.out.print(" ago");
263                }
264                if (ent.getDuration() == -1) {
265                    System.out.print(" (running)");
266                } else if (ent.getDuration() != 0) {
267                    System.out.print("; duration=");
268                    StringBuilder sb = new StringBuilder();
269                    TimeUtils.formatDuration(ent.getDuration(), sb);
270                    System.out.print(sb);
271                }
272                System.out.println();
273            }
274        }
275    }
276
277    private void runReset() throws Exception {
278        String packageName = null;
279        int userId = UserHandle.USER_CURRENT;
280        for (String argument; (argument = nextArg()) != null;) {
281            if (ARGUMENT_USER.equals(argument)) {
282                String userStr = nextArgRequired();
283                if ("all".equals(userStr)) {
284                    userId = UserHandle.USER_ALL;
285                } else if ("current".equals(userStr)) {
286                    userId = UserHandle.USER_CURRENT;
287                } else if ("owner".equals(userStr)) {
288                    userId = UserHandle.USER_OWNER;
289                } else {
290                    userId = Integer.parseInt(nextArgRequired());
291                }
292            } else {
293                if (packageName == null) {
294                    packageName = argument;
295                } else {
296                    System.err.println("Error: Unsupported argument: " + argument);
297                    return;
298                }
299            }
300        }
301
302        // Parsing complete, let's execute the command.
303
304        if (userId == UserHandle.USER_CURRENT) {
305            userId = ActivityManager.getCurrentUser();
306        }
307
308        final IAppOpsService appOpsService = IAppOpsService.Stub.asInterface(
309                ServiceManager.getService(Context.APP_OPS_SERVICE));
310        appOpsService.resetAllModes(userId, packageName);
311        System.out.print("Reset all modes for: ");
312        if (userId == UserHandle.USER_ALL) {
313            System.out.print("all users");
314        } else {
315            System.out.print("user "); System.out.print(userId);
316        }
317        System.out.print(", ");
318        if (packageName == null) {
319            System.out.println("all packages");
320        } else {
321            System.out.print("package "); System.out.println(packageName);
322        }
323    }
324}
325