The 2016 Android Developer Toolbox [TUNISIA]

75
The 2016 Android Developer Toolbox #droidconTN

Transcript of The 2016 Android Developer Toolbox [TUNISIA]

The 2016 Android Developer Toolbox

#droidconTN

@Nilhcem

Gau$er Mechling

Aspiring So+ware Cra+sman

#droidconTN

happnFind the people you've crossed paths with.

#droidconTN

"Use the right tools for the right job"

#droidconTN

Gradle(with the Android Plugin for Gradle)

#droidconTN

Build Variants

#droidconTN

U+2020h"ps://github.com/JakeWharton/u2020

#droidconTN

Debug screenExample from Google iosched

#droidconTN

Internal se*ngs app

Similar to u2020 Sample app

• Display build / device informa6on

• Change endpoint (restart process with JakeWharton/ProcessPhoenix)

• Show logs (pedrovgs/Lynx)

• Allow easy bug report capturing (maFprecious/telescope)

• Enable/Disable Takt / Stetho / Scalpel / Madge...

#droidconTN

Droidcon Tunisia applica/on

h"ps://github.com/Nilhcem/droidcontn-2016

#droidconTN

MeasuringTes$ng

Analyzing

#droidconTN

Android Studio(Android Monitor tab)

#droidconTN

#droidconTN

Detect Memory Leaks

#droidconTN

#droidconTN

Leak Canaryh"ps://github.com/square/leakcanary

#droidconTN

Frame Rateh"ps://github.com/wasabeef/Takt

compile 'jp.wasabeef:takt:1.0.2'

public class MyApplication extends Application { @Override public void onCreate() { super.onCreate(); Takt.stock(this).play(); }}

#droidconTN

Hugoh"ps://github.com/JakeWharton/hugo

Prefix classes/methods with:

@DebugLog

Result:

DroidconApp V ⇢ onCreate()

V ⇢ initGraph()

V ⇠ initGraph [13ms]

V ⇢ initLogger()

V ⇠ initLogger [1ms]

V ⇠ onCreate [73ms]

#droidconTN

Pidcath"ps://github.com/JakeWharton/pidcat

$ pidcat com.nilhcem.droidcontn

#droidconTN

AndroidDevMetricsh"ps://github.com/frogermcs/dagger2metrics

apply plugin: 'com.frogermcs.dagger2metrics'

public class MyApplication extends Application { @Override public void onCreate() { super.onCreate(); Dagger2Metrics.enableCapturing(this); }}

#droidconTN

#QualityMa*ers

#droidconTN

Code sta)c analysis tools

#droidconTN

Code sta)c analysis tools• Lint

• PMD

• Checkstyle

• Findbugs

• Facebook's infer

#droidconTN

SonarQube

#droidconTN

SonarQube Dockerfile

FROM java:8MAINTAINER Nilhcem

RUN DEBIAN_FRONTEND=noninteractive apt updateRUN DEBIAN_FRONTEND=noninteractive apt install -y wget unzipRUN wget -q https://sonarsource.bintray.com/Distribution/sonarqube/sonarqube-5.3.zipRUN unzip -qq sonarqube-5.3.zip -d /opt/RUN rm sonarqube-5.3.zip

EXPOSE 9000EXPOSE 9092

CMD ["/opt/sonarqube-5.3/bin/linux-x86-64/sonar.sh", "console"]

Then launch:

docker build -t nilhcem/sonarqube .docker run -p 9000:9000 -p 9092:9092 -d nilhcem/sonarqube

#droidconTN

SonarQube Gradle Configura1on

apply plugin: 'org.sonarqube'

sonarqube { properties { def appProject = project(':app') def appProjectDir = appProject.projectDir def appProjectBuildDir = appProject.buildDir

property 'sonar.projectKey', 'my-app' property 'sonar.projectName', 'Android app' property 'sonar.projectVersion', appProject.android.defaultConfig.versionName

property 'sonar.host.url', 'http://docker:9000' property 'sonar.jdbc.url', 'jdbc:h2:tcp://docker:9092/sonar' property 'sonar.jdbc.username', 'sonar' property 'sonar.jdbc.password', 'sonar'

property 'sonar.sources', 'src' property 'sonar.sourceEncoding', 'UTF-8' property 'sonar.java.binaries', 'build' property 'sonar.exclusions', '**/*Test.java'

property 'sonar.core.codeCoveragePlugin', 'jacoco' property 'sonar.junit.reportsPath', "$appProjectBuildDir/test-results/prodDebug" property 'sonar.jacoco.reportPath', "$appProjectBuildDir/jacoco/testProdDebugUnitTest.exec" }}

./gradlew :app:assembleProdDebug :app:testProdDebug :app:sonarqube

#droidconTN

#droidconTN

#droidconTN

Measuring

Tes$ngAnalyzing

#droidconTN

Mock Server

#droidconTN

NodeJS + Express/* Setup */var fs = require('fs');var express = require('express');var app = express();app.set('port', process.env.PORT || 8080);var port = app.get('port');

/* Speakers list */app.get('/speakers', function(req, res) { res.type('application/json; charset=utf8'); res.status(200).send(fs.readFileSync('data/speakers.json', 'utf8'));});

/* Other web services */// ...

/* Starting the server */app.listen(port, function () { console.log('Express server listening on port ' + port);});

#droidconTN

NodeJS + Express: Slow service

sleep(2000);

// Ugly, but does the job :)function sleep(durationMillis) { var now = new Date().getTime(); while(new Date().getTime() < now + durationMillis) { // do nothing }}

#droidconTN

Example

h"ps://github.com/Nilhcem/droidcontn-2016/tree/master/mockserver

$ npm install$ node server.js

Then, go tohttp://localhost:8990/

#droidconTN

Hosts Editor

h"ps://play.google.com/store/apps/details?id=com.nilhcem.hostseditor

#droidconTN

HTTP Debugging

#droidconTN

HTTP Debugging• mitmproxy

• Fiddler

• Charles proxy

#droidconTN

Charles Proxy

• Simulate a laggy/unstable connec3on

• Repeat queries

• Check the responses

• Add some breakpoints to

• Cancel an HTTP(s) call

• Edit a request

• Edit a response

#droidconTN

Cancel a request

#droidconTN

Edit a response

#droidconTN

MeasuringTes$ng

Analyzing#droidconTN

Developer Op*ons

#droidconTN

#droidconTN

#droidconTN

hierarchyviewerh"ps://developer.android.com/tools/performance/hierarchy-viewer/setup.html#hvproto-variable

h"p://developer.android.com/tools/debugging/debugging-ui.html

$ export ANDROID_HVPROTO=ddm$ exec ${ANDROID_HOME}/tools/monitor

#droidconTN

uiautomatorviewer

#droidconTN

Anima&ons

Developer op)ons -> Anima)on scale

#droidconTN

Anima&ons

Make a screencast:

$ adb shell screenrecord /sdcard/demo.mp4$ adb pull /sdcard/demo.mp4

VLC :

Press the keyboard E key to see frames one by one

#droidconTN

apktool + dex2jar + JD-GUIh"p://ibotpeaches.github.io/Apktoolh"ps://github.com/pxb1988/dex2jar

h"p://jd.benow.ca

#droidconTN

jadxh"ps://github.com/skylot/jadx

#droidconTN

Stetho

#droidconTN

Stetho - UI

#droidconTN

Stetho - Network

#droidconTN

Stetho - Resources

#droidconTN

Stetho - Dumpapp

#droidconTN

Stetho - Dumpapp

#droidconTN

Stetho - Dumpapp

#droidconTN

Some more dumpapp examples

#droidconTN

$ dumpapp accessToken invalidate

$ dumpapp accessToken showfce1235425dcdeadbeef8cafebabe42

$ dumpapp clipboard getHello

$ dumpapp clipboard set "Text to copy"$ dumpapp onTrimMemory$ dumpapp openIntent scheme://open/speaker/3

$ dumpapp gcmTokenuHyMKnEQ:APA91bEHZ6afFLQQMzKgSDjp5y_0397usitPqj_Bp02

$ dumpapp geolocDataLocation[fused 22.5430883,114.1043205 acc=21 et=+21m5s492ms]

$ dumpapp runningServicescom.example.LocationService

#droidconTN

Stetho - Custom pluginh"p://code.tutsplus.com/tutorials/debugging-android-apps-with-facebooks-stetho--cms-24205

class AppDumperPlugin implements DumperPlugin { @Override public String getName() { return "my_plugin_name"; }

@Override public void dump(DumperContext dumpContext) throws DumpException { PrintStream writer = dumperContext.getStdout(); String commandName = (args.isEmpty()) ? "" : args.remove(0);

if (commandName.equals("test")) { out.println("Hello, World!"); } }}

#droidconTN

Stetho - Custom plugin

#droidconTN

Stetho - Dumpapp (example)(ActivityProvider in a dependency graph in debug)

@Singletonpublic class ActivityProvider implements Application.ActivityLifecycleCallbacks {

private Activity currentActivity;

@Inject public ActivityProvider(Application app) { app.registerActivityLifecycleCallbacks(this); }

public Activity getCurrentActivity() { return currentActivity; }

@Override public void onActivityResumed(Activity activity) { currentActivity = activity; }

@Override public void onActivityPaused(Activity activity) { currentActivity = null; }}

#droidconTN

Stetho - Dumpapp (example)

AppDumperPlugin.java

private void displayCurrentSessionData(PrintStream writer) { Activity activity = activityProvider.getCurrentActivity(); if (activity instanceof SessionDetailsActivity) { try { // Use reflection to access private "session" field Field field = SessionDetailsActivity.class.getDeclaredField("session"); field.setAccessible(true); Session session = (Session) field.get(activity); writer.println(new GsonBuilder().setPrettyPrinting().create().toJson(session)); } catch (Exception e) { writer.println(e.getMessage()); } }}

#droidconTN

Stetho console + Rhino

#droidconTN

Measuring

Tes$ng

Analysing

Lastly

#droidconTN

ADB + Shell# Open a deep linking intentadb shell am start -a android.intent.action.VIEW -d "scheme://app/deep/linking"

# List running servicesadb shell dumpsys activity services

# Get the path of an install applicationadb shell pm path app.package.name

# Take a screenshotadb shell screencap -p | perl -pe '\''s/\x0D\x0A/\x0A/g'\'' > screen.png

# Paste text from your computer clipboard to your android devicepbpaste | sed "s/%/%%/g" | sed "s/ /\%\s/g" | xargs adb shell input text

#droidconTN

Postmanh"ps://www.getpostman.com/

#droidconTN

Lockito

#droidconTN

Fill RAMOn-device low-memory tes2ng for Android

h"ps://play.google.com/store/apps/details?id=com.tspoon.androidtoolbelt

#droidconTN

ViewInspectorh"ps://github.com/xfumihiro/ViewInspector

#droidconTN

Methodscounth"p://www.methodscount.com/

h"ps://github.com/mihaip/dex-method-counts

#droidconTN

Vysorh"p://www.vysor.io/

#droidconTN

As a conclusion...

#droidconTN

Choose according to your needs and tastes

#droidconTN

Master your toolsto build be*er apps

❤h"ps://twi"er.com/Nilhcem

h"ps://plus.google.com/+Gau2erMechling#droidconTN