Slide 1

Slide 1 text

Code Generation beyond APT Ragunath Jawahar

Slide 2

Slide 2 text

An Android developer walks into the bar…

Slide 3

Slide 3 text

An Android developer walks into the bar office…

Slide 4

Slide 4 text

No content

Slide 5

Slide 5 text

Analytics Code MixpanelAPI mixpanel = MixpanelAPI.getInstance(this, token); try { JSONObject payload  = new JSONObject(); payload.put("Gender", "Female"); payload.put("Logged  in", false); payload.put("Source", "Twitter"); mixpanel.track("Signup", payload); } catch (JSONException exception) { Log.e("Frodo", "JsonException", exception); }

Slide 6

Slide 6 text

Analytics Code MixpanelAPI mixpanel = MixpanelAPI.getInstance(this, token); try  { JSONObject payload  =  new  JSONObject(); payload.put("Gender",  "Female"); payload.put("Logged  in",  false); payload.put("Source",  "Twitter"); mixpanel.track("Signup",  payload); }  catch  (JSONException exception)  { Log.e("Frodo",  "JsonException",  exception); }

Slide 7

Slide 7 text

Analytics Code MixpanelAPI mixpanel =  MixpanelAPI.getInstance(this,  token); try  { JSONObject payload  = new JSONObject(); payload.put("Gender", "Female"); payload.put("Logged  in", false); payload.put("Source", "Twitter"); mixpanel.track("Signup",  payload); }  catch  (JSONException exception)  { Log.e("Frodo",  "JsonException",  exception); }

Slide 8

Slide 8 text

Analytics Code MixpanelAPI mixpanel =  MixpanelAPI.getInstance(this,  token); try  { JSONObject payload  =  new  JSONObject(); payload.put("Gender",  "Female"); payload.put("Logged  in",  false); payload.put("Source",  "Twitter"); mixpanel.track("Signup", payload); }  catch  (JSONException exception)  { Log.e("Frodo",  "JsonException",  exception); }

Slide 9

Slide 9 text

GREMLINS

Slide 10

Slide 10 text

No content

Slide 11

Slide 11 text

No content

Slide 12

Slide 12 text

…and only 300 more rows

Slide 13

Slide 13 text

MAY THE “BRUTE” FORCE BE WITH YOU

Slide 14

Slide 14 text

Problems •Slow •Boring •Inaccurate •Error Prone •Hard to Test •Difficult to Change •Multiple Platforms •Multiple Apps

Slide 15

Slide 15 text

Requirements • Accuracy • Accommodate Changes • Don’t Leak Cognitive Resources • Testability • Support Multiple Apps and Platforms

Slide 16

Slide 16 text

AOP

Slide 17

Slide 17 text

AOP

Slide 18

Slide 18 text

No content

Slide 19

Slide 19 text

WORKFLOW – SCREEN – EVENT

Slide 20

Slide 20 text

workflow().screen().event()

Slide 21

Slide 21 text

No content

Slide 22

Slide 22 text

No content

Slide 23

Slide 23 text

Java Poet – Method MethodSpec mainMethod = MethodSpec.methodBuilder("main") .addModifiers(Modifier.PUBLIC, Modifier.STATIC) .addParameter(String[].class, "args") .addStatement( "$T.out.println($S)", System.class, "Hello,  World!" ).build();

Slide 24

Slide 24 text

Java Poet – Method MethodSpec mainMethod =  MethodSpec.methodBuilder("main") .addModifiers(Modifier.PUBLIC,  Modifier.STATIC) .addParameter(String[].class,  "args") .addStatement( "$T.out.println($S)", System.class, "Hello,  World!" ).build();

Slide 25

Slide 25 text

Java Poet – Method MethodSpec mainMethod =  MethodSpec.methodBuilder("main") .addModifiers(Modifier.PUBLIC, Modifier.STATIC) .addParameter(String[].class,  "args") .addStatement( "$T.out.println($S)", System.class, "Hello,  World!" ).build();

Slide 26

Slide 26 text

Java Poet – Method MethodSpec mainMethod =  MethodSpec.methodBuilder("main") .addModifiers(Modifier.PUBLIC,  Modifier.STATIC) .addParameter(String[].class, "args") .addStatement( "$T.out.println($S)", System.class, "Hello,  World!" ).build();

Slide 27

Slide 27 text

Java Poet – Method MethodSpec mainMethod =  MethodSpec.methodBuilder("main") .addModifiers(Modifier.PUBLIC,  Modifier.STATIC) .addParameter(String[].class,  "args") .addStatement( "$T.out.println($S)", System.class, "Hello,  World!" ).build();

Slide 28

Slide 28 text

Java Poet – Method MethodSpec mainMethod =  MethodSpec.methodBuilder("main") .addModifiers(Modifier.PUBLIC,  Modifier.STATIC) .addParameter(String[].class,  "args") .addStatement( "$T.out.println($S)", System.class, "Hello,  World!" ).build();

Slide 29

Slide 29 text

Java Poet – Class TypeSpec greeterClass = TypeSpec.classBuilder("Greeter") .addModifiers(Modifier.PUBLIC) .addMethod(mainMethod) .build();

Slide 30

Slide 30 text

Java Poet – Class TypeSpec greeterClass =  TypeSpec.classBuilder("Greeter") .addModifiers(Modifier.PUBLIC) .addMethod(mainMethod) .build();

Slide 31

Slide 31 text

Java Poet – Class TypeSpec greeterClass =  TypeSpec.classBuilder("Greeter") .addModifiers(Modifier.PUBLIC) .addMethod(mainMethod) .build();

Slide 32

Slide 32 text

Java Poet – Class TypeSpec greeterClass =  TypeSpec.classBuilder("Greeter") .addModifiers(Modifier.PUBLIC) .addMethod(mainMethod) .build();

Slide 33

Slide 33 text

Java Poet – Class TypeSpec greeterClass =  TypeSpec.classBuilder("Greeter") .addModifiers(Modifier.PUBLIC) .addMethod(mainMethod) .build();

Slide 34

Slide 34 text

Java Poet – Compilation Unit JavaFile greeterFile = JavaFile.builder("in.kitecash", greeterClass).build(); greeterFile.writeTo(System.out);

Slide 35

Slide 35 text

Java Poet – Compilation Unit JavaFile greeterFile =  JavaFile.builder("in.kitecash", greeterClass).build(); greeterFile.writeTo(System.out);

Slide 36

Slide 36 text

Java Poet – Compilation Unit JavaFile greeterFile =  JavaFile.builder("in.kitecash", greeterClass).build(); greeterFile.writeTo(System.out);

Slide 37

Slide 37 text

Java Poet – Compilation Unit JavaFile greeterFile =  JavaFile.builder("in.kitecash", greeterClass).build(); greeterFile.writeTo(System.out);

Slide 38

Slide 38 text

Java Poet – Generated Class package in.kitecash; import java.lang.String; import java.lang.System; public class Greeter { public static void main(String[] args) { System.out.println("Hello,  World!"); } }

Slide 39

Slide 39 text

BLUEPRINT

Slide 40

Slide 40 text

ITERATE ON PAPER

Slide 41

Slide 41 text

Components •CSV Parser •Source Code Generator

Slide 42

Slide 42 text

Objective workflow().screen().event()

Slide 43

Slide 43 text

High-Level Design •Top-level Workflow classes •Inner Screen classes •Top-level Analytics class •Top-level Tracker abstract class

Slide 44

Slide 44 text

Class Diagram …  more  workflow  classes

Slide 45

Slide 45 text

Class Diagram …  more  inner  screen  classes

Slide 46

Slide 46 text

workflow().screen().event() •Methods present inside Analytics class •Workflows are generated as top-level classes •Workflows encapsulate Screen inner classes

Slide 47

Slide 47 text

workflow().screen().event() •Methods present inside Workflow classes •Screens are generated as inner classes in Workflows •Screens encapsulate event methods

Slide 48

Slide 48 text

workflow().screen().event() •Methods present inside Screen classes •May accept parameters for collecting additional data

Slide 49

Slide 49 text

Analytics •Contains  Workflow methods  that  provide   instances  of  Workflow classes •Instantiated  by  supplying  a  Tracker implementation

Slide 50

Slide 50 text

Tracker public abstract class Tracker { public  abstract  void track(String event, Map payload); }

Slide 51

Slide 51 text

Tracker public  abstract  class  Tracker  { public  abstract  void track(String event, Map payload); }

Slide 52

Slide 52 text

Tracker public  abstract  class  Tracker  { public  abstract  void  track(String event, Map  payload); }

Slide 53

Slide 53 text

Tracker public  abstract  class  Tracker  { public  abstract  void  track(String  event, Map payload); }

Slide 54

Slide 54 text

ACCURACY

Slide 55

Slide 55 text

WRITE CODE TO GENERATE CODE

Slide 56

Slide 56 text

CHANGES

Slide 57

Slide 57 text

CSV Removals – Red Markers

Slide 58

Slide 58 text

CSV Additions – Yellow Markers

Slide 59

Slide 59 text

PRODUCTIVITY

Slide 60

Slide 60 text

Auto-Completion

Slide 61

Slide 61 text

versus this? MixpanelAPI mixpanel =  MixpanelAPI.getInstance(this,  token); try  { JSONObject payload  =  new  JSONObject(); payload.put("Gender",  "Female"); payload.put("Logged  in",  false); payload.put("Source",  "Twitter"); mixpanel.track("Signup",  payload); }  catch  (JSONException exception)  { Log.e("Frodo",  "JsonException",  exception); }

Slide 62

Slide 62 text

Checklist

Slide 63

Slide 63 text

Checklist

Slide 64

Slide 64 text

Checklist

Slide 65

Slide 65 text

TESTING

Slide 66

Slide 66 text

Gradle & Dagger •Multiple Build Types •Multiple Modules and Components

Slide 67

Slide 67 text

Trackers •Logcat Tracker •Mixpanel Tracker •Dialog Tracker •Verifying Tracker

Slide 68

Slide 68 text

LogcatTracker public class LogcatTracker implements Tracker { private static final String TAG  = "LogcatTracker"; @Override public void track(String trigger, Map payload) { Log.d(TAG, "Received  trigger:  " + trigger  +  ",  " + payload); } }

Slide 69

Slide 69 text

LogcatTracker public  class  LogcatTracker implements  Tracker  { private  static  final  String  TAG  =  "LogcatTracker"; @Override public  void  track(String  trigger,  Map  payload)  { Log.d(TAG, "Received  trigger:  " + trigger  +  ",  " + payload); } }

Slide 70

Slide 70 text

AlertDialogTracker public class AlertDialogTracker implements Tracker { private final Context context; public AlertDialogTracker(Context context) { this.context = context; } @Override public void track(String trigger, Map payload) { StringBuilder messageBuilder = new StringBuilder(); //  Create  a  String  message  from  the  payload... showAlertDialog(trigger, stringBuilder.toString()); } }

Slide 71

Slide 71 text

AlertDialogTracker public  class  AlertDialogTracker implements  Tracker  { private  final  Context  context; public  AlertDialogTracker(Context  context)  { this.context =  context; } @Override public  void  track(String  trigger,  Map  payload)  { StringBuilder messageBuilder = new StringBuilder(); //  Create  a  String  message  from  the  payload... showAlertDialog(trigger, stringBuilder.toString()); } }

Slide 72

Slide 72 text

MixpanelTracker public class MixpanelTracker implements Tracker { private final MixpanelAPI mixpanel; public MixpanelTracker(Context context, String projectToken) { this.mixpanel = MixpanelAPI.getInstance(context, projectToken); } @Override public void track(String trigger, Map payload) { JSONObject properties  = new JSONObject(); //  Populate  the  JSONObject from  payload... mixpanel.track(trigger, properties); } }

Slide 73

Slide 73 text

MixpanelTracker public  class  MixpanelTracker implements  Tracker  { private  final  MixpanelAPI mixpanel; public  MixpanelTracker(Context context, String projectToken)  { this.mixpanel =  MixpanelAPI.getInstance(context,  projectToken); } @Override public  void  track(String  trigger,  Map  payload)  { JSONObject properties  =  new  JSONObject(); //  Populate  the  JSONObject from  payload... mixpanel.track(trigger,  properties); } }

Slide 74

Slide 74 text

MixpanelTracker public  class  MixpanelTracker implements  Tracker  { private  final  MixpanelAPI mixpanel; public  MixpanelTracker(Context  context,  String  projectToken)  { this.mixpanel =  MixpanelAPI.getInstance(context,  projectToken); } @Override public  void  track(String  trigger,  Map  payload)  { JSONObject properties  = new JSONObject(); //  Populate  the  JSONObject from  payload... mixpanel.track(trigger, properties); } }

Slide 75

Slide 75 text

iOS + Swift

Slide 76

Slide 76 text

Difficulties •Swift •No Transpilers •No Code Generation Libraries

Slide 77

Slide 77 text

J2ObjC

Slide 78

Slide 78 text

J2ObjC Java Objective-C Swift

Slide 79

Slide 79 text

J2ObjC Java Objective-C Swift

Slide 80

Slide 80 text

J2ObjC Java Objective-C Swift

Slide 81

Slide 81 text

PEACE

Slide 82

Slide 82 text

PEACE

Slide 83

Slide 83 text

abstract  class(es) NOT interface(s)

Slide 84

Slide 84 text

J2ObjC Nuances •Runtime Library •Adds ~10 MB

Slide 85

Slide 85 text

PROCESS

Slide 86

Slide 86 text

Overview • Export spreadsheet from BI team as CSV • Check exported CSV into analytics project • Run Gradle task to generate source code from CSV • Copy generated source code into Android projects • Send generated source code to iOS team • iOS team includes transpiled source code to iOS projects

Slide 87

Slide 87 text

Gradle Task task  generateAnalyticsSource(type: JavaExec) { classpath files('../sidekick/analytics/bin/grumpy-­‐old-­‐man.jar') main  'in.kitecash.sidekick.analytics.GrumpyOldMan' }

Slide 88

Slide 88 text

Gradle Task task  generateAnalyticsSource(type:  JavaExec)  { classpath files('../sidekick/analytics/bin/grumpy-­‐old-­‐man.jar') main  'in.kitecash.sidekick.analytics.GrumpyOldMan' }

Slide 89

Slide 89 text

Gradle Task task  generateAnalyticsSource(type:  JavaExec)  { classpath files('../sidekick/analytics/bin/grumpy-­‐old-­‐man.jar') main  'in.kitecash.sidekick.analytics.GrumpyOldMan' }

Slide 90

Slide 90 text

TAKEAWAY

Slide 91

Slide 91 text

Code Generation Tips • Iterate through the design several times • Doesn’t have to be DRY • Use a prefix in generated source files • Add “Auto-Generated” comments and Javadoc • Prefer abstract classes over interfaces • Extract platform-specific code, use abstraction

Slide 92

Slide 92 text

2 Moar Cents •Make tools work for you •Communicate with all teams that are involved, this is the key

Slide 93

Slide 93 text

ragunathjawahar GitHub / Twitter / Speakerdeck