SlideShare a Scribd company logo
1
Polyglot Plugin Development
How to write plugins in a language other than Java




Stefan Saasen
Confluence Team Lead, Atlassian

                                                     2
Audience
• Written an Atlassian plugin in Java
• Interested in other programming languages
• Eager to see what is possible with Atlassian plugins




                                                         3
Show of hands
• Who has actually written a Atlassian plugin?
• In any other language than Java?




                                                 4
5
Rhino
        6
7
https://bitbucket.org/ssaasen/atlassian-jruby-example-plugin

https://bitbucket.org/ssaasen/atlassian-scala-example-plugin




                                                               8
Polyglot Plugins
• Why?
• How?
 • Example 1: Scala
 • Example 2: JRuby

• Go and do it!


                      9
Why not?
           10
Why ?
        11
What to tell the boss?
• Leverage existing knowledge in the team
• Libraries/applications already written in language X
• Time to market/devspeed
• Explore technologies in a sandboxed environment
• Grass looks always greener on the other side


                                                         12
What to tell the boss?
• Leverage existing knowledge in the team
• Libraries/applications already written in language X
• Time to market/devspeed
• Explore technologies in a sandboxed environment
• Grass looks always greener on the other side


                                                         13
14
“No time to write in Java”

                  “Less bugs”
“Succinctness”
                                15
16
Polyglot Plugins
• Why?
• How?
 • Example 1: Scala
 • Example 2: JRuby

• Go and do it!


                      17
• Hybrid language: Object oriented & functional
• Concise, statically typed (with type inference)
• Higher order/first class functions, pattern matching
• Powerful type system
• Rich standard library


                                                        18
19
20
21
22
How?


       23
Overview

                                          Scala




                                                  scalac
                Scala Runtime Library

Java Bytecode
   *.class                                Java

                 Java Standard Library




                                                  javac
                                    JVM


                                                           24
25
26
Build process
• Start with Maven to get
  the AMPS goodness
• Use the maven-scala-
  plugin
• Add SBT if necessary



                            27
Build process f.
 • Compile time is a concern!
 • FSC
 • Useful Maven goals:
    mvn scala:cc
    mvn scala:console


                                28
Bundle runtime
                 29
Size might be a
  problem :(
                  30
Deployment options
     Option          Scope       Package        Size

                                mvn package
     Bundle          compile     -Pbundle
                                               8.5 MB    Simple

Bundle & Shrink                 mvn package              Deploy
                     compile                   680 KB
 w/ProGuard                     -Pproduction            externally

                                                         Deploy
Install separately   provided   mvn package    50 KB
                                                        internally
                                                                     31
Simply install using the
   plugin manager!

                           32
Development checklist
• Dependencies: Scala libraries are jar files and often
  available from Maven repositories
• Compile time: mvn scala:cc for continuous compilation
• Workflow: scala:cc & atlas-cli pi or
  mvn package && atlas-install-plugin
• Profit: mvn scala:console to get the Scala REPL

                                                          33
Scala: Working
with Java APIs
                 34
Scala gotchas (the 80%)
• Collections
• Method naming conventions
• Annotations
• Null vs. Option




                              35
Collections
 • Immutable vs Mutable, serial vs. parallel
 • Rich API: Prefer Scala collections
 • Convert between Java and Scala collections when
   calling Java methods
 • Solution: JavaConverters (explicit, preferred) vs.
   JavaConversions (implicit)
 • Watch out for serialization: Bandana/XStream
                                                        36
37
38
39
Method naming conventions
• Libraries rely on bean style naming conventions
  e.g. XWork, Velocity
• Scala: uniform access principle
• Solution: Use @BeanProperty and
  @BooleanBeanProperty



                                                    40
41
42
43
(Meta)-Annotations
• How to use @JavaAnnotation on a getter method
  when defining fields?
• Scala meta annotations @beanGetter, @beanGetter,
  @field, @getter, @setter




                                                     44
45
46
47
Option vs. null
 • In Scala Option represents an optional value.
 • Don’t let the billion-dollar mistake leak into your
   Scala code!
 • Solution: wrap nullable method calls in an Option




                                                         48
49
50
51
52
Show me the
   code!
              53
Polyglot Plugin Programming
Polyglot Plugin Programming
Polyglot Plugin Programming
57
58
59
60
Example Scala plugin
• Servlet & XWork action
• Confluence macro
• REST via Jersey/Jackson
• Build setup
• Use as a starting point
                    https://bitbucket.org/ssaasen/atlassian-scala-example-plugin

                                                                                   61
Let’s recap
 • Replace Java entirely with Scala or mix and match
 • Runtime library: 3 deployment options
 • Fits into the AMPS development workflow
 • Scala Gotchas: watch out for common problems
 • “Scala is a great fit to write Atlassian plugins in”


                                                         62
Agenda
• Why?
• How?
 • Example 1: Scala
 • Example 2: JRuby

• Go and do it!


                      63
(J)Ruby
• Dynamic, object-oriented language
• Everything is an object (no “primitives”)
• Meta-programming and open, executable classes
  allow for easy creation of DSLs
• JRuby: Ruby runtime written in Java



                                                  64
65
How?


       66
Overview
                    Ruby
Ruby files *.rb                                  Your Ruby
                  Standard          RubyGems




                                                              jrubyc
                                                   Code
                   Library

                                  Ruby Runtime

Java Bytecode
   *.class                                             Java

                     Java Standard Library




                                                              javac
                                        JVM




                                                                       67
68
69
70
Plugins can’t be
written in Ruby

                   71
But we can still use
 Ruby in a plugin!

                       72
Recipe for using JRuby
• Embed the Ruby Runtime
• Use Java to execute Ruby scripts and classes
• Make components of the host application available
• Use the script execution result in Java




                                                      73
Embedding Strategies
• javax.scripting - JSR 223
• Apache Bean Scripting Framework (BSF)
• JRuby Embed - lower level JRuby API




                                          74
Embedding Strategies
• javax.scripting - JSR 223
  Only configurable via system properties :(
• Apache Bean scripting framework (BSF)
  No point in adding a separate API
• JRuby Embed - lower level JRuby API



                                              75
76
Building the plugin
• Package Ruby files
• Download and package Ruby gems
• Make Ruby gems available on the Ruby LOAD_PATH
• Bundle everything into a plugin jar file
• A lot of Maven XML :)
• Caveat: JRuby runtime is even larger!
                                                   77
Show me the
   code!
              78
79
80
1 minute


           81
82
83
84
85
86
87
88
89
Example JRuby plugin
• How to use Rubygem based
  libraries in a plugin
• How to access host
  components in Ruby
• How to use the execution result
• Sinatra in Confluence ;-)
                     https://bitbucket.org/ssaasen/atlassian-jruby-example-plugin
                                                                                    90
Let’s recap
• The plugin can’t be written in Ruby!
• Use the Ruby runtime
• Development workflow bonus: dynamic reloading!
• The overhead of adding the runtime and providing
  Java glue code makes this only feasible for high
  value use cases :(

                                                     91
Conclusion
• Scala is a great alternative when
  developing an Atlassian plugin
• JRuby is powerful, makes
  developing in a dynamic language
  possible but requires more work and
  custom Java adapter code
• Java interoperability needs to be well
  understood in both cases
                                           92
TAKE-AWAYS




“   The power of the Atlassian plugin framework allows
    you to build your plugins the way you want and

                                     ”
    Java doesn’t have to be the limit.




     #atlascamp


                                                         93
Thank you!




             94
• “Happy Foods” by swanksalot
  http://www.flickr.com/photos/swanksalot/5021262869/
• “Construction” by Daniel Morris
  http://www.flickr.com/photos/danielmorris/275438405/
• “70 80 90” by roujo
  http://www.flickr.com/photos/tekmagika/437989361



                                                        95

More Related Content

PDF
JRuby in the enterprise
PDF
Why JRuby?
KEY
Ruby Plugins for Jenkins
PPTX
Spring 3.1 to 3.2 in a Nutshell - Spring I/O 2012
PDF
JRoR Deploying Rails on JRuby
PDF
Migrate to JRuby
PPT
Ruby on rails
PDF
Java Edge.2009.Grails.Web.Dev.Made.Easy
JRuby in the enterprise
Why JRuby?
Ruby Plugins for Jenkins
Spring 3.1 to 3.2 in a Nutshell - Spring I/O 2012
JRoR Deploying Rails on JRuby
Migrate to JRuby
Ruby on rails
Java Edge.2009.Grails.Web.Dev.Made.Easy

What's hot (20)

PDF
Testing Ember Apps
PDF
An introduction and future of Ruby coverage library
PDF
A Taste of Pharo 7.0
PDF
Camel Desing Patterns Learned Through Blood, Sweat, and Tears
PPTX
Capybara and cucumber with DSL using ruby
PDF
Bitter Java, Sweeten with JRuby
PDF
GraalVM - MadridJUG 2019-10-22
PDF
Cloud Native Camel Design Patterns
KEY
Euruko 2012 - JRuby
PDF
JVM Memory Management Details
PDF
Event Driven Architecture with Apache Camel
PPTX
Java performance tuning
PPTX
Performance of Microservice Frameworks on different JVMs
PDF
GraalVM - OpenSlava 2019-10-18
KEY
High performance network programming on the jvm oscon 2012
PDF
Maven: from Scratch to Production (.pdf)
PDF
ToulouseJUG-Maven 3.x, will it lives up to its promises
PDF
BordeauxJUG-Maven 3.x, will it lives up to its promises
PPT
MAVEN
PDF
From java-to-ruby-book-summary
Testing Ember Apps
An introduction and future of Ruby coverage library
A Taste of Pharo 7.0
Camel Desing Patterns Learned Through Blood, Sweat, and Tears
Capybara and cucumber with DSL using ruby
Bitter Java, Sweeten with JRuby
GraalVM - MadridJUG 2019-10-22
Cloud Native Camel Design Patterns
Euruko 2012 - JRuby
JVM Memory Management Details
Event Driven Architecture with Apache Camel
Java performance tuning
Performance of Microservice Frameworks on different JVMs
GraalVM - OpenSlava 2019-10-18
High performance network programming on the jvm oscon 2012
Maven: from Scratch to Production (.pdf)
ToulouseJUG-Maven 3.x, will it lives up to its promises
BordeauxJUG-Maven 3.x, will it lives up to its promises
MAVEN
From java-to-ruby-book-summary
Ad

Viewers also liked (6)

PDF
Atlassian unite pix
PDF
Unite keynote all - mike's edit (don't edit!)
PPTX
Ravendb@swedenprogressive
KEY
How HipChat Powers the HipChat Team - Atlassian Summit 2012
PDF
Atlassian Overview
PDF
Guerilla marketing your service desk
Atlassian unite pix
Unite keynote all - mike's edit (don't edit!)
Ravendb@swedenprogressive
How HipChat Powers the HipChat Team - Atlassian Summit 2012
Atlassian Overview
Guerilla marketing your service desk
Ad

Similar to Polyglot Plugin Programming (20)

PDF
Enterprise OSGi at eBay
PDF
Venkat Subramaniam Blending Java With Dynamic Languages
KEY
Polyglot Grails
ODP
How to start using Scala
PDF
Absorbing Scala Into Java Ecosystem
PDF
What is new and cool j2se & java
PDF
Real World Technologies
PDF
Java Future S Ritter
PDF
Calling All Modularity Solutions: A Comparative Study from eBay
PDF
Ola Bini Evolving The Java Platform
PDF
Powering the Next Generation Services with Java Platform - Spark IT 2010
PDF
Calling all modularity solutions
PDF
Java Core | Java 8 and OSGi Modularisation | Tim Ellison & Neil Bartlett
PDF
Jax london 2011
KEY
Scala Introduction
PPTX
Scala adoption by enterprises
KEY
Java to Scala: Why & How
PPTX
Lessons Learned: Scala and its Ecosystem
PDF
JRuby - Enterprise 2.0
PDF
Java ScriptingJava Scripting: One VM, Many Languages
Enterprise OSGi at eBay
Venkat Subramaniam Blending Java With Dynamic Languages
Polyglot Grails
How to start using Scala
Absorbing Scala Into Java Ecosystem
What is new and cool j2se & java
Real World Technologies
Java Future S Ritter
Calling All Modularity Solutions: A Comparative Study from eBay
Ola Bini Evolving The Java Platform
Powering the Next Generation Services with Java Platform - Spark IT 2010
Calling all modularity solutions
Java Core | Java 8 and OSGi Modularisation | Tim Ellison & Neil Bartlett
Jax london 2011
Scala Introduction
Scala adoption by enterprises
Java to Scala: Why & How
Lessons Learned: Scala and its Ecosystem
JRuby - Enterprise 2.0
Java ScriptingJava Scripting: One VM, Many Languages

More from Atlassian (20)

PPTX
International Women's Day 2020
PDF
10 emerging trends that will unbreak your workplace in 2020
PDF
Forge App Showcase
PDF
Let's Build an Editor Macro with Forge UI
PDF
Meet the Forge Runtime
PDF
Forge UI: A New Way to Customize the Atlassian User Experience
PDF
Take Action with Forge Triggers
PDF
Observability and Troubleshooting in Forge
PDF
Trusted by Default: The Forge Security & Privacy Model
PDF
Designing Forge UI: A Story of Designing an App UI System
PDF
Forge: Under the Hood
PDF
Access to User Activities - Activity Platform APIs
PDF
Design Your Next App with the Atlassian Vendor Sketch Plugin
PDF
Tear Up Your Roadmap and Get Out of the Building
PDF
Nailing Measurement: a Framework for Measuring Metrics that Matter
PDF
Building Apps With Color Blind Users in Mind
PDF
Creating Inclusive Experiences: Balancing Personality and Accessibility in UX...
PDF
Beyond Diversity: A Guide to Building Balanced Teams
PDF
The Road(map) to Las Vegas - The Story of an Emerging Self-Managed Team
PDF
Building Apps With Enterprise in Mind
International Women's Day 2020
10 emerging trends that will unbreak your workplace in 2020
Forge App Showcase
Let's Build an Editor Macro with Forge UI
Meet the Forge Runtime
Forge UI: A New Way to Customize the Atlassian User Experience
Take Action with Forge Triggers
Observability and Troubleshooting in Forge
Trusted by Default: The Forge Security & Privacy Model
Designing Forge UI: A Story of Designing an App UI System
Forge: Under the Hood
Access to User Activities - Activity Platform APIs
Design Your Next App with the Atlassian Vendor Sketch Plugin
Tear Up Your Roadmap and Get Out of the Building
Nailing Measurement: a Framework for Measuring Metrics that Matter
Building Apps With Color Blind Users in Mind
Creating Inclusive Experiences: Balancing Personality and Accessibility in UX...
Beyond Diversity: A Guide to Building Balanced Teams
The Road(map) to Las Vegas - The Story of an Emerging Self-Managed Team
Building Apps With Enterprise in Mind

Recently uploaded (20)

PPT
Teaching material agriculture food technology
PDF
Spectral efficient network and resource selection model in 5G networks
PPTX
VMware vSphere Foundation How to Sell Presentation-Ver1.4-2-14-2024.pptx
PPTX
KOM of Painting work and Equipment Insulation REV00 update 25-dec.pptx
PDF
Electronic commerce courselecture one. Pdf
PDF
Network Security Unit 5.pdf for BCA BBA.
PPTX
Programs and apps: productivity, graphics, security and other tools
PDF
Building Integrated photovoltaic BIPV_UPV.pdf
PDF
The Rise and Fall of 3GPP – Time for a Sabbatical?
PDF
Review of recent advances in non-invasive hemoglobin estimation
PDF
Advanced methodologies resolving dimensionality complications for autism neur...
PDF
Mobile App Security Testing_ A Comprehensive Guide.pdf
PPTX
MYSQL Presentation for SQL database connectivity
PDF
Build a system with the filesystem maintained by OSTree @ COSCUP 2025
PDF
Optimiser vos workloads AI/ML sur Amazon EC2 et AWS Graviton
PPTX
Big Data Technologies - Introduction.pptx
PDF
Blue Purple Modern Animated Computer Science Presentation.pdf.pdf
PPTX
20250228 LYD VKU AI Blended-Learning.pptx
PDF
Architecting across the Boundaries of two Complex Domains - Healthcare & Tech...
PDF
7 ChatGPT Prompts to Help You Define Your Ideal Customer Profile.pdf
Teaching material agriculture food technology
Spectral efficient network and resource selection model in 5G networks
VMware vSphere Foundation How to Sell Presentation-Ver1.4-2-14-2024.pptx
KOM of Painting work and Equipment Insulation REV00 update 25-dec.pptx
Electronic commerce courselecture one. Pdf
Network Security Unit 5.pdf for BCA BBA.
Programs and apps: productivity, graphics, security and other tools
Building Integrated photovoltaic BIPV_UPV.pdf
The Rise and Fall of 3GPP – Time for a Sabbatical?
Review of recent advances in non-invasive hemoglobin estimation
Advanced methodologies resolving dimensionality complications for autism neur...
Mobile App Security Testing_ A Comprehensive Guide.pdf
MYSQL Presentation for SQL database connectivity
Build a system with the filesystem maintained by OSTree @ COSCUP 2025
Optimiser vos workloads AI/ML sur Amazon EC2 et AWS Graviton
Big Data Technologies - Introduction.pptx
Blue Purple Modern Animated Computer Science Presentation.pdf.pdf
20250228 LYD VKU AI Blended-Learning.pptx
Architecting across the Boundaries of two Complex Domains - Healthcare & Tech...
7 ChatGPT Prompts to Help You Define Your Ideal Customer Profile.pdf

Polyglot Plugin Programming

  • 1. 1
  • 2. Polyglot Plugin Development How to write plugins in a language other than Java Stefan Saasen Confluence Team Lead, Atlassian 2
  • 3. Audience • Written an Atlassian plugin in Java • Interested in other programming languages • Eager to see what is possible with Atlassian plugins 3
  • 4. Show of hands • Who has actually written a Atlassian plugin? • In any other language than Java? 4
  • 5. 5
  • 6. Rhino 6
  • 7. 7
  • 9. Polyglot Plugins • Why? • How? • Example 1: Scala • Example 2: JRuby • Go and do it! 9
  • 10. Why not? 10
  • 11. Why ? 11
  • 12. What to tell the boss? • Leverage existing knowledge in the team • Libraries/applications already written in language X • Time to market/devspeed • Explore technologies in a sandboxed environment • Grass looks always greener on the other side 12
  • 13. What to tell the boss? • Leverage existing knowledge in the team • Libraries/applications already written in language X • Time to market/devspeed • Explore technologies in a sandboxed environment • Grass looks always greener on the other side 13
  • 14. 14
  • 15. “No time to write in Java” “Less bugs” “Succinctness” 15
  • 16. 16
  • 17. Polyglot Plugins • Why? • How? • Example 1: Scala • Example 2: JRuby • Go and do it! 17
  • 18. • Hybrid language: Object oriented & functional • Concise, statically typed (with type inference) • Higher order/first class functions, pattern matching • Powerful type system • Rich standard library 18
  • 19. 19
  • 20. 20
  • 21. 21
  • 22. 22
  • 23. How? 23
  • 24. Overview Scala scalac Scala Runtime Library Java Bytecode *.class Java Java Standard Library javac JVM 24
  • 25. 25
  • 26. 26
  • 27. Build process • Start with Maven to get the AMPS goodness • Use the maven-scala- plugin • Add SBT if necessary 27
  • 28. Build process f. • Compile time is a concern! • FSC • Useful Maven goals: mvn scala:cc mvn scala:console 28
  • 30. Size might be a problem :( 30
  • 31. Deployment options Option Scope Package Size mvn package Bundle compile -Pbundle 8.5 MB Simple Bundle & Shrink mvn package Deploy compile 680 KB w/ProGuard -Pproduction externally Deploy Install separately provided mvn package 50 KB internally 31
  • 32. Simply install using the plugin manager! 32
  • 33. Development checklist • Dependencies: Scala libraries are jar files and often available from Maven repositories • Compile time: mvn scala:cc for continuous compilation • Workflow: scala:cc & atlas-cli pi or mvn package && atlas-install-plugin • Profit: mvn scala:console to get the Scala REPL 33
  • 35. Scala gotchas (the 80%) • Collections • Method naming conventions • Annotations • Null vs. Option 35
  • 36. Collections • Immutable vs Mutable, serial vs. parallel • Rich API: Prefer Scala collections • Convert between Java and Scala collections when calling Java methods • Solution: JavaConverters (explicit, preferred) vs. JavaConversions (implicit) • Watch out for serialization: Bandana/XStream 36
  • 37. 37
  • 38. 38
  • 39. 39
  • 40. Method naming conventions • Libraries rely on bean style naming conventions e.g. XWork, Velocity • Scala: uniform access principle • Solution: Use @BeanProperty and @BooleanBeanProperty 40
  • 41. 41
  • 42. 42
  • 43. 43
  • 44. (Meta)-Annotations • How to use @JavaAnnotation on a getter method when defining fields? • Scala meta annotations @beanGetter, @beanGetter, @field, @getter, @setter 44
  • 45. 45
  • 46. 46
  • 47. 47
  • 48. Option vs. null • In Scala Option represents an optional value. • Don’t let the billion-dollar mistake leak into your Scala code! • Solution: wrap nullable method calls in an Option 48
  • 49. 49
  • 50. 50
  • 51. 51
  • 52. 52
  • 53. Show me the code! 53
  • 57. 57
  • 58. 58
  • 59. 59
  • 60. 60
  • 61. Example Scala plugin • Servlet & XWork action • Confluence macro • REST via Jersey/Jackson • Build setup • Use as a starting point https://bitbucket.org/ssaasen/atlassian-scala-example-plugin 61
  • 62. Let’s recap • Replace Java entirely with Scala or mix and match • Runtime library: 3 deployment options • Fits into the AMPS development workflow • Scala Gotchas: watch out for common problems • “Scala is a great fit to write Atlassian plugins in” 62
  • 63. Agenda • Why? • How? • Example 1: Scala • Example 2: JRuby • Go and do it! 63
  • 64. (J)Ruby • Dynamic, object-oriented language • Everything is an object (no “primitives”) • Meta-programming and open, executable classes allow for easy creation of DSLs • JRuby: Ruby runtime written in Java 64
  • 65. 65
  • 66. How? 66
  • 67. Overview Ruby Ruby files *.rb Your Ruby Standard RubyGems jrubyc Code Library Ruby Runtime Java Bytecode *.class Java Java Standard Library javac JVM 67
  • 68. 68
  • 69. 69
  • 70. 70
  • 72. But we can still use Ruby in a plugin! 72
  • 73. Recipe for using JRuby • Embed the Ruby Runtime • Use Java to execute Ruby scripts and classes • Make components of the host application available • Use the script execution result in Java 73
  • 74. Embedding Strategies • javax.scripting - JSR 223 • Apache Bean Scripting Framework (BSF) • JRuby Embed - lower level JRuby API 74
  • 75. Embedding Strategies • javax.scripting - JSR 223 Only configurable via system properties :( • Apache Bean scripting framework (BSF) No point in adding a separate API • JRuby Embed - lower level JRuby API 75
  • 76. 76
  • 77. Building the plugin • Package Ruby files • Download and package Ruby gems • Make Ruby gems available on the Ruby LOAD_PATH • Bundle everything into a plugin jar file • A lot of Maven XML :) • Caveat: JRuby runtime is even larger! 77
  • 78. Show me the code! 78
  • 79. 79
  • 80. 80
  • 81. 1 minute 81
  • 82. 82
  • 83. 83
  • 84. 84
  • 85. 85
  • 86. 86
  • 87. 87
  • 88. 88
  • 89. 89
  • 90. Example JRuby plugin • How to use Rubygem based libraries in a plugin • How to access host components in Ruby • How to use the execution result • Sinatra in Confluence ;-) https://bitbucket.org/ssaasen/atlassian-jruby-example-plugin 90
  • 91. Let’s recap • The plugin can’t be written in Ruby! • Use the Ruby runtime • Development workflow bonus: dynamic reloading! • The overhead of adding the runtime and providing Java glue code makes this only feasible for high value use cases :( 91
  • 92. Conclusion • Scala is a great alternative when developing an Atlassian plugin • JRuby is powerful, makes developing in a dynamic language possible but requires more work and custom Java adapter code • Java interoperability needs to be well understood in both cases 92
  • 93. TAKE-AWAYS “ The power of the Atlassian plugin framework allows you to build your plugins the way you want and ” Java doesn’t have to be the limit. #atlascamp 93
  • 95. • “Happy Foods” by swanksalot http://www.flickr.com/photos/swanksalot/5021262869/ • “Construction” by Daniel Morris http://www.flickr.com/photos/danielmorris/275438405/ • “70 80 90” by roujo http://www.flickr.com/photos/tekmagika/437989361 95