-
Notifications
You must be signed in to change notification settings - Fork 366
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
1 changed file
with
108 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,2 +1,109 @@ | ||
# libsu | ||
An Android library as reusable code to call `su` properly. It also comes with many handy features. | ||
|
||
[data:image/s3,"s3://crabby-images/e2841/e28418b2cad6881b002fb9daf40d984b5415b614" alt=""](https://jitpack.io/#topjohnwu/libsu) | ||
|
||
An Android library that provides APIs to a Unix (root) shell. | ||
|
||
Some poorly coded applications requests a new shell (call `su`, or worse `su -c <commands>`) for every single command, which is very inefficient. This library makes sharing a single, globally shared shell session in Android applications super easy: developers don't have to worry about concurrency issues, and it comes with a rich selection of both synchronous and asynchronous APIs to create a powerful root app. | ||
|
||
One complex Android application using `libsu` for all root related operations is [Magisk Manager](https://github.com/topjohnwu/MagiskManager). | ||
|
||
## Download | ||
```java | ||
repositories { | ||
maven { url 'https://jitpack.io' } | ||
} | ||
dependencies { | ||
implementation 'com.github.topjohnwu:libsu:1.0.0' | ||
} | ||
``` | ||
|
||
## Simple Tutorial | ||
|
||
### Setup | ||
Subclass `Shell.ContainerApp` and use it as your `Application`: | ||
|
||
```java | ||
public class ExampleApp extends Shell.ContainerApp { | ||
public ExampleApp() { | ||
// Set some flags here if you need | ||
Shell.setFlags(Shell.FLAG_REDIRECT_STDERR); | ||
Shell.verboseLogging(BuildConfig.DEBUG); | ||
} | ||
} | ||
``` | ||
|
||
Specify the custom Application in `AndroidManifest.xml` | ||
|
||
```xml | ||
<?xml version="1.0" encoding="utf-8"?> | ||
<manifest xmlns:android="http://schemas.android.com/apk/res/android" | ||
...> | ||
<application | ||
android:name=".ExampleApp" | ||
...> | ||
... | ||
</application> | ||
</manifest> | ||
``` | ||
|
||
### Synchronous Shell Operations | ||
|
||
High level synchronous APIs are under `Shell.Sync`. Think twice before calling these methods in the main thread, as they could cause the app to freeze and end up with ANR errors. | ||
|
||
```java | ||
/* Simple root shell commands, get results immediately */ | ||
List<String> result = Shell.Sync.su("find /dev/block -iname 'boot'"); | ||
|
||
/* Do something with the result */ | ||
|
||
/* Execute scripts from raw resources */ | ||
result = Shell.Sync.loadScript(getResources().openRawResource(R.raw.script))); | ||
``` | ||
|
||
### Asynchronous Shell Operations | ||
|
||
High level asynchronous APIs are under `Shell.Async`. These methods will return immediately and will not get the output synchronously. Use callbacks to receive the results, or update UI asynchronously. | ||
|
||
```java | ||
/* Run commands and don't care the result */ | ||
Shell.Async.su("setenforce 0", "setprop test.prop test"); | ||
|
||
/* Get results after commands are done with a callback */ | ||
Shell.Async.su(new Shell.Async.Callback() { | ||
@Override | ||
public void onTaskResult(List<String> out, List<String> err) { | ||
/* Do something with the result */ | ||
} | ||
}, "cat /proc/mounts"); | ||
|
||
/* Use a CallbackList to receive a callback every time a new line is outputted */ | ||
|
||
List<String> callbackList = new CallbackList<String>() { | ||
@Override | ||
public void onAddElement(String s) { | ||
/* Do something with the new line */ | ||
} | ||
}; | ||
|
||
// Pass the callback list to receive shell outputs | ||
Shell.Async.su(callbackList, "for i in 1 2 3 4 5; do echo $i; sleep 1; done"); | ||
``` | ||
|
||
### Advanced | ||
Initialize the shell with custom `Shell.Initializer`, similar to what `.bashrc` will do. | ||
|
||
```java | ||
Shell.setInitializer(new Shell.Initializer() { | ||
@Override | ||
public void onRootShellInit(@NonNull Shell shell) { | ||
shell.run(null, null, "export PATH=" + BUSYBOX_PATH + ":$PATH"); | ||
} | ||
}); | ||
``` | ||
|
||
## Documentation | ||
|
||
This repo also comes with an example app (`:example`), check the code and play/experiment with it. | ||
|
||
More detailed full documentation is in the project's [JavaDoc Page](https://topjohnwu.github.io/libsu). |