Building a Microservice Federation with Grails
data:image/s3,"s3://crabby-images/36bac/36bace366573f972df91149927222d064923bf1a" alt=""
Grails... microservices...
you must be crazy man!
About me
data:image/s3,"s3://crabby-images/5feea/5feea32ff3b5ad116f6185b5599dd7c39cf99dea" alt=""
20 years Java
12+ years Spring
7+ years Groovy/Grails
PowerBuilder
C/C++
FORTRAN
1st career in aviation/aerospace
twitter: @jackfrosch linkedin: ../in/jackfrosch
email: jackfrosch@gmail.com groovy-community.slack.com: @jackfrosch
The Story I'm About to Tell You is True
- Microservices are here
- Doing it the easy way
- Building a small microservice federation
Microservices are here
- Foundational concepts
- Microservice advantages
- Microservice disadvantages
data:image/s3,"s3://crabby-images/4116a/4116a641a9211f2e98ed60bc2b740604f3d02aee" alt=""
What are microservices?
data:image/s3,"s3://crabby-images/a998e/a998e8de85408c60007a762d7028a09454b04e7c" alt=""
Microservices are small, autonomous services that work together.
Sam Newman Building Microservices, O'Reilly Media
Small is key
data:image/s3,"s3://crabby-images/b022c/b022c8ec1f2a174ccfb73dd1ff47c8ae5b51cf8a" alt=""
Small in functional scope...
data:image/s3,"s3://crabby-images/794fa/794fa952254270f3a9b9445d4a41be106835239e" alt=""
http://www.whattofix.com/images/ComplexERDExample.gif
Probably not small functional scope...
... not necesarily size
data:image/s3,"s3://crabby-images/51f35/51f356f5ef80075416d1c71fc064aa42fd089075" alt=""
Smaller is simpler
Everything should be made as simple as possible, but not simpler*
* though attributed to Einstein, this simple quote is actually from
Roger Sessions paraphrasing (and simplifying!) a statement by Albert Einstein
data:image/s3,"s3://crabby-images/5b792/5b792dece3f37ebb5f0adefe72ec6c1da348ab49" alt=""
Simpler is better
data:image/s3,"s3://crabby-images/60f51/60f511eba4d3344a8746ebe977c0f9279f949a5c" alt=""
A system that is hard to understand is hard to change.
— Eric Evans, Domain-Driven Design
Autonomy is key
data:image/s3,"s3://crabby-images/4e297/4e2970a27e5cf78f1ebf68b504484f2f65542109" alt=""
Development autonomy*
data:image/s3,"s3://crabby-images/c0bf4/c0bf443d2cac733f8bae1ed5d0069dfd9a21547f" alt=""
data:image/s3,"s3://crabby-images/284b9/284b9eef76159559cf806615ec545c82cf935dcb" alt=""
data:image/s3,"s3://crabby-images/1446f/1446f40cfff824aa785a006b1a2688b8e4737a85" alt=""
data:image/s3,"s3://crabby-images/80d2a/80d2a654dda4b628ada2c7d6f74eebc36c1f9271" alt=""
data:image/s3,"s3://crabby-images/889eb/889ebeeb195480d4073569d1f166ac447f21276d" alt=""
data:image/s3,"s3://crabby-images/589ff/589ffdb3bb7044a8b3e1e61140dcdfea98c826e6" alt=""
data:image/s3,"s3://crabby-images/30ac6/30ac61ee0727ee347cd82a2491e8ab3c723b44aa" alt=""
data:image/s3,"s3://crabby-images/b8e37/b8e373364793e0cf395c01229c0e04569d255ae7" alt=""
*Caveat: You can't break my stuff
http://www.memes.com/meme/498049
Consumer Driven Contracts
data:image/s3,"s3://crabby-images/33251/3325197bc9875b4cf9c21cfbde5e7d53683f0e2b" alt=""
http://bit.ly/thoughtworks-consumer-driven-contracts
Testing: How much & what kind?
data:image/s3,"s3://crabby-images/e7e0f/e7e0f45f3457fb06ae9b0300df12c7bf76d67116" alt=""
http://famouswonders.com/wp-content/gallery/pyramids-of-egypt/pyramid-of-khafre.jpg
Data Autonomy
data:image/s3,"s3://crabby-images/3c829/3c829a03c03be7c92139cab3d86282e42b729c5d" alt=""
http://martinfowler.com/articles/microservices.html
Build/Deploy autonomy
data:image/s3,"s3://crabby-images/a0c64/a0c643790e5c0cab129da57bfb6163d01ddfcb09" alt=""
http://www.openmakesoftware.com/images/ReleaseEngineer/CI-CD.png
data:image/s3,"s3://crabby-images/a0c64/a0c643790e5c0cab129da57bfb6163d01ddfcb09" alt=""
data:image/s3,"s3://crabby-images/a0c64/a0c643790e5c0cab129da57bfb6163d01ddfcb09" alt=""
Scale where the load is...
data:image/s3,"s3://crabby-images/ad72b/ad72bbf5baa7c0748cb8cd664d6ac72eed9a22bf" alt=""
Operational Autonomy
data:image/s3,"s3://crabby-images/65995/65995fb50f53036fb001339790f29b6df694d794" alt=""
https://media.licdn.com/mpr/mpr/p/8/005/083/1a8/257d716.jpg
All this sounds great, but...
data:image/s3,"s3://crabby-images/8190d/8190d22d85779d62f5267b6a9353abd26c40653a" alt=""
...what about our monolith?
Identify bounded contexts...
data:image/s3,"s3://crabby-images/0f211/0f211432c304e2166ab82c4e3320e19a0622270c" alt=""
... and divide along the seams
Favor choreography over orchestration
data:image/s3,"s3://crabby-images/b0acb/b0acb5e32c0608ad568d8a063bbd53b158770d97" alt=""
data:image/s3,"s3://crabby-images/4ca0c/4ca0cfa6f90508ec50e2864b38e3dd61f29f6abc" alt=""
http://kennysilva.net/wp-content/uploads/2010/12/orchestra-conductor.jpg
Monolith
Microservice
class Passenger {
String accountNo
String firstName
String lastName
Address billingAddress
Payment paymentPreference
List<Payment> paymentHistory
Phone home
Phone mobile
Phone work
BloodType bloodType
...
}
What about shared domain?
// For Trip Management
class Passenger {
String accountNo
String firstName
String lastName
Phone mobile
...
}
// For Billing
class Passenger {
String accountNo
String firstName
String lastName
Address billingAddress
List<Payment> paymentHistory
Phone home
Phone work
...
}
The evils of too much coupling between services are far worse than the problems caused by code duplication.
- Sam Newman, Building Microservices
Microservices Federation Cross-cutting Concerns
- Logging
- Security
- Metrics and monitoring
Logging Tips
- Use a Correlation ID
- Use a consistent log message format
- Use log aggregation
Logging Aggregators
data:image/s3,"s3://crabby-images/aea6a/aea6a12eac5a51422e011a5e9b4f8bc9c4b8a77f" alt=""
data:image/s3,"s3://crabby-images/9cb9c/9cb9cd967bcdddecc3d7579cc98c64d32311f4d6" alt=""
data:image/s3,"s3://crabby-images/40628/406280804230becd61509e6ff73970fcd222a2eb" alt=""
data:image/s3,"s3://crabby-images/9eb97/9eb97cf25e89b3bfa15851db3463ea15de0587ec" alt=""
data:image/s3,"s3://crabby-images/0c405/0c405d11ac2852777c13e4226f069124a8b3960c" alt=""
data:image/s3,"s3://crabby-images/a4b22/a4b22eff2e11320d1dd312ebaf3c5d329ebffd44" alt=""
"ELK Stack"
Security
- None
- At the gateway only
- At every microservice
Security at the gateway only
data:image/s3,"s3://crabby-images/1f40b/1f40bd3205ab4419971f4fe53c214775ebbcbd5d" alt=""
data:image/s3,"s3://crabby-images/bb847/bb847f0de1ddbe5755b1dc424c3a07eda1d26076" alt=""
data:image/s3,"s3://crabby-images/43cc8/43cc8ae692cc907b6fb830c349f530e2d0fd4c7d" alt=""
"They're inside the room!"
"Yikes!"
"Mmm, you look tasty."
Perimeter security reminds me of the movie, Aliens
Security at every microservice
data:image/s3,"s3://crabby-images/fd124/fd1245a98b0f61dc304b633b9af961796784d6b9" alt=""
Metrics, Monitoring & More
- Microservices need to report metrics
- Resiliency requires monitoring & circuit breakers
- Automatic service discovery
- Gateway / reverse proxy
- Load balancing
Coming up:
Spring Boot & Spring Cloud to the rescue!
Microservice Architecture Advantages
- Scalable
- Adaptable
- Resilient
- Testable
- More future-proof
- More greenfield
data:image/s3,"s3://crabby-images/597af/597af7fc4b7b7348137165956ebc3869edccfc4e" alt=""
Microservice Architecture Disadvantages
- Solid domain understanding
- Development more complicated
- More builds
- More databases
- Log aggregation is essential
- Metrics, monitoring & more essential
Doing it the easy way
- Spring Boot
- Spring Cloud with Netflix OSS
- Grails
Spring Boot includes metrics endpoint
data:image/s3,"s3://crabby-images/a76eb/a76eb52be0d252597047db59597f2633f5889e8d" alt=""
There are many metrics out of the box, but you can create your own.
http://docs.spring.io/spring-boot/docs/current/reference/html/production-ready-endpoints.html
Spring Boot includes many endpoints
data:image/s3,"s3://crabby-images/0c583/0c583087eb74bcc8aa57c4e2bd41409110719fa2" alt=""
http://docs.spring.io/spring-boot/docs/current/reference/html/production-ready-endpoints.html
Spring Cloud provides monitoring
... and much more
data:image/s3,"s3://crabby-images/a0343/a0343f0c315d0b25e33eafde22f2a8795e83917f" alt=""
Service discovery with Eureka
data:image/s3,"s3://crabby-images/dd30c/dd30ca8fd46ee00ba612e48bd606a05d978735e7" alt=""
Gateway /Rev Proxy with Zuul
data:image/s3,"s3://crabby-images/0c6e9/0c6e9c321b27d1d63016dd1cc698bfc57a598dbb" alt=""
http://bit.ly/vignette2_wikia_nocookie_net_ghostbusters
Gateway /Rev Proxy with Zuul
data:image/s3,"s3://crabby-images/59831/5983121b08a569533673d2de1e36891c47d8db9c" alt=""
Hystrix for Resilience
data:image/s3,"s3://crabby-images/8b4b7/8b4b76d16a362406ad257ece8ae6a669fd8abbf9" alt=""
Monitoring & Circuit Breakers w/ Hystrix
data:image/s3,"s3://crabby-images/00dd0/00dd0743eb2fa40b4077ac06400130c9895d7198" alt=""
data:image/s3,"s3://crabby-images/8ddea/8ddeaac6f16cec6d58ab083f5d399bb291d4d0fd" alt=""
http://martinfowler.com/bliki/CircuitBreaker.html
data:image/s3,"s3://crabby-images/77f29/77f2940cd573e1c5f003ca2733372c96190e9998" alt=""
Grails plays well
- Grails full stack development
- Grails REST
- Grails plugin architecture
Full stack != monolith
Grails full stack development
data:image/s3,"s3://crabby-images/e9bc6/e9bc6a556f6f4c8237f2eca9b8e3562fd6ceacc1" alt=""
Web UI, SQL / NOSQL Database Support, Spring, Hibernate, Security, REST, Java, Groovy...
Create App
data:image/s3,"s3://crabby-images/6ee1b/6ee1bd38f9ebcc2a1e952391667f4858ab19de64" alt=""
$grails create-app HelloWorld
Hello World app... easy as 1 - 2 - 3
grails> create-controller demo.Hello
package demo
class HelloController {
def index() {
render 'Hello SpringOne2GX 2015!'
}
}
data:image/s3,"s3://crabby-images/47d96/47d965d559de211d60f649f3c6f057d7e35da89e" alt=""
Full stack CRUD apps are about as easy...
1. Create controller
2. Make it do something
3. See it to believe it!
Create a Domain Class
data:image/s3,"s3://crabby-images/f4d95/f4d95ced61d7d17c392d78927ce35321c508d177" alt=""
Define demo.Person
package demo
class Person {
String firstName
String lastName
String email
String twitterHandle
static constraints = {
firstName blank:false, maxSize: 32
lastName blank: false, maxSize: 64
email blank: false, maxSize: 128, email:true
twitterHandle nullable:true, maxSize:64
}
}
Generate Scaffolding...
data:image/s3,"s3://crabby-images/ae707/ae7070def8d256549e147eb523040a60f0e1132a" alt=""
.. to bootstrap CRUD app
data:image/s3,"s3://crabby-images/37d2f/37d2f84708dc160171b3db4e2890d708256f650a" alt=""
After generation
data:image/s3,"s3://crabby-images/fff01/fff01867f62b0923dcd403b31adac0c00c1d739b" alt=""
Run App
data:image/s3,"s3://crabby-images/fb16c/fb16c78c82ba997611cea470967ca954a4e4ec60" alt=""
grails> run-app
Retrieve Persons List
data:image/s3,"s3://crabby-images/6814f/6814fe30171f11eb87fe51af8b6faf48b5ef6b43" alt=""
Create Person form
data:image/s3,"s3://crabby-images/fdd81/fdd819ed55a71983b8438b2b4c461a4be6459f0f" alt=""
Create a Person record
data:image/s3,"s3://crabby-images/adfd7/adfd7a586e5d511c7bd4c8e90b3f394aee4937d4" alt=""
Show a Person record
data:image/s3,"s3://crabby-images/1ac82/1ac8225fbaffa23eef6cc6d3c35fc13e2466090f" alt=""
Retrieve Persons List
data:image/s3,"s3://crabby-images/0ea7e/0ea7e18926b50a8e0a4b8e275fda849456d87e6b" alt=""
Delete a Person record
data:image/s3,"s3://crabby-images/e5f83/e5f83b5f8471087ee698e966a5abb0fecbfe1506" alt=""
Grails REST URI patterns
data:image/s3,"s3://crabby-images/30419/304192f2fbb23104ed1b33a4bba0800f29d71d23" alt=""
What if I want to do more than CRUD?
package demo
import grails.rest.RestfulController
class PersonController extends RestfulController<Person> {
static responseFormats =['json', 'xml']
PersonController() {
super(Person)
}
def findPersonsWithLastNameLike(String likeness) {
respond Person.findByLastNameLike("${likeness}%")
}
}
class UrlMappings {
static mappings = {
"/persons"(resources:'person')
"/persons/search/withLastNameLike/$likeness"(controller: 'person',
action: 'findPersonsWithLastNameLike')
}
}
Restfully create a Person...
data:image/s3,"s3://crabby-images/b2341/b2341de276c01929e6a71becf3c661dace9f20dc" alt=""
... yields a 201 created response
data:image/s3,"s3://crabby-images/f9cd1/f9cd1cdfdc888bdc41620ce7b198193d5ac8fa11" alt=""
Test Rest
$ curl -i -X GET http://localhost:8080/persons.xml
HTTP/1.1 200 OK
Server: Apache-Coyote/1.1
X-Application-Context: application:development
Content-Type: text/xml;charset=UTF-8
Transfer-Encoding: chunked
Date: Mon, 07 Sep 2015 21:26:16 GMT
<?xml version="1.0" encoding="UTF-8"?><list />
$
200 OK good ... XML weird, but you wanted weird
Test Rest
$ curl -i -X GET http://localhost:8080/persons
HTTP/1.1 200 OK
Server: Apache-Coyote/1.1
X-Application-Context: application:development
Content-Type: text/xml;charset=UTF-8
Transfer-Encoding: chunked
Date: Mon, 07 Sep 2015 21:26:16 GMT
[]
$
200 OK good ... JSON good.
Change @Resource to prefer JSON
package demo
import grails.rest.Resource
@Resource(uri='/persons', formats=['json', 'xml'])
class Person {
String firstName
String lastName
String email
String twitterHandle
static constraints = {
firstName blank:false, maxSize: 32
lastName blank: false, maxSize: 64
email blank: false, maxSize: 128, email:true
twitterHandle nullable:true, maxSize:64
}
}
Grails content negotiation
$ curl -i -X GET --header "Accept:application/json" http://localhost:8080/persons
HTTP/1.1 200 OK
Server: Apache-Coyote/1.1
X-Application-Context: application:development
Content-Type: text/xml;charset=UTF-8
Transfer-Encoding: chunked
Date: Mon, 07 Sep 2015 21:26:16 GMT
[]
$
200 OK good ... JSON good.
Grails content negotiation
$ curl -i -X GET http://localhost:8080/persons.json
HTTP/1.1 200 OK
Server: Apache-Coyote/1.1
X-Application-Context: application:development
Content-Type: text/xml;charset=UTF-8
Transfer-Encoding: chunked
Date: Mon, 07 Sep 2015 21:26:16 GMT
[]
$
200 OK good ... JSON good.
Besides UI CRUD, what about REST?
package demo
import grails.rest.Resource
@Resource(uri='/persons')
class Person {
String firstName
String lastName
String email
String twitterHandle
static constraints = {
firstName blank:false, maxSize: 32
lastName blank: false, maxSize: 64
email blank: false, maxSize: 128, email:true
twitterHandle nullable:true, maxSize:64
}
}
grails create-app RestfulCrud
With an IDE...
data:image/s3,"s3://crabby-images/0d6af/0d6af353c58adcb82a53b5cff05916f03caf13e7" alt=""
Test Rest
$ curl -i -X GET http://localhost:8080/persons
[jfrosch@localhost demo]$ HTTP/1.1 200 OK
Server: Apache-Coyote/1.1
X-Application-Context: application:development
Content-Type: text/xml;charset=UTF-8
Transfer-Encoding: chunked
Date: Mon, 07 Sep 2015 21:26:16 GMT
<?xml version="1.0" encoding="UTF-8"?><list />
$
Hmm... 200 OK good ... XML bad.
Building a Grails microservice federation
data:image/s3,"s3://crabby-images/3acbd/3acbd8cc522b62b1639ffceef801d9de1664c146" alt=""
Download all the sources for this from: https://github.com/jfrosch/GrailsMicroFederation
Build a Eureka Server
data:image/s3,"s3://crabby-images/af269/af269aae02dfc0c6fa94838f6ee8d7e0e0f64691" alt=""
data:image/s3,"s3://crabby-images/5c053/5c053ae3e76ab39a6ec4ad9f5212cc9e66dfb567" alt=""
data:image/s3,"s3://crabby-images/8f584/8f584113c00baba97e6690ae4de26e38eedaaa15" alt=""
Build a Eureka Server
data:image/s3,"s3://crabby-images/b1581/b15814b055223ce587424bd15ea07d5a429a7fb4" alt=""
data:image/s3,"s3://crabby-images/a6124/a6124e4b24a05e42f22cb4cb9fe88607eb3732bc" alt=""
Build a Zuul Server
data:image/s3,"s3://crabby-images/69c1e/69c1e482ab7434b7f0196bbe996e045210e8ce0e" alt=""
/product-catalog acts as the root of the uri
data:image/s3,"s3://crabby-images/bb53c/bb53cf55d564f70e71ab490f5ee7008b83540ee0" alt=""
data:image/s3,"s3://crabby-images/865b0/865b017aba50e6022d792f0688e091b21c3a9a90" alt=""
Build a Zuul Server
data:image/s3,"s3://crabby-images/303a6/303a606b38c1e8c66514a05a6634ad8bbd433a4b" alt=""
Inspecting Zuul Routes
data:image/s3,"s3://crabby-images/983ed/983ede17622c4141d957c581fa19276abdeb6777" alt=""
Microservice Setup
data:image/s3,"s3://crabby-images/9fa1b/9fa1bfa6f94b76090fb13c111ab96df3df3a0713" alt=""
data:image/s3,"s3://crabby-images/ac6b8/ac6b82b27cc6b382335bee7fae3c4db8af39c752" alt=""
data:image/s3,"s3://crabby-images/7a650/7a650e7431cdb6b6cd0e6c70b67e982343f94602" alt=""
product-catalog is the name we map the Zuul route to
Microservice Setup
Can use @EnableDiscoveryClient if not intending to use Netflix implementation
data:image/s3,"s3://crabby-images/98193/981935eeeaca07bc2eb9f66f6c34aad2a89cfdf2" alt=""
@HystrixCommand
data:image/s3,"s3://crabby-images/42e24/42e2424107685f79a70eaa1b0e8cd9dcba0ff110" alt=""
Notice we're using Zuul server at port 8765 and Zuul route, not Recommendation Engine app's endpoint directly
I've had some difficulty applying circuit-breakers in controllers, so do so only in services
Build Hystrix Dashboard
data:image/s3,"s3://crabby-images/ca6f5/ca6f51415a74001d18a42abad8df6b5144d476d7" alt=""
data:image/s3,"s3://crabby-images/0bde8/0bde8eafb3fd201c90b4e8f17f82170243f73b88" alt=""
data:image/s3,"s3://crabby-images/d25ae/d25ae31a123fe94330fdaa0beb46f1b94568061a" alt=""
data:image/s3,"s3://crabby-images/a372a/a372a2675feb2ac0aa9c2d3b70e749d58e4bac1f" alt=""
Note we're using Zuul server, not a microservice endpoint, to catch all data streams
data:image/s3,"s3://crabby-images/099ca/099ca9b4c88b0535bcbe89555381d2f32371ceeb" alt=""
All services are up
... After killing recommendation engine ...
data:image/s3,"s3://crabby-images/fa0c1/fa0c154c086021d525c314d5237e3721aa4b6a74" alt=""
... After restarting recommendation engine ...
data:image/s3,"s3://crabby-images/60790/607903aa74077e98674eea025c9cbd7e67fbbdd8" alt=""
Summary
- Adaptable
- Scalable
- Resilient
- Maintainable
- Monitorable
- Cloud ready
You'd be crazy to adopt a microservice architecture using Grails...
data:image/s3,"s3://crabby-images/36bac/36bace366573f972df91149927222d064923bf1a" alt=""
unless your application needs to be ...
(even if your company isn't ready for the cloud yet)
Resources
- http://projects.spring.io/spring-cloud/
- http://cloud.spring.io/spring-cloud-netflix/
- http://martinfowler.com/articles/microservices.html
-
Github repo for demo: https://github.com/jfrosch/GrailsMicroFederation
data:image/s3,"s3://crabby-images/0ff14/0ff144cce5aa82536bba3faab6c9695270943305" alt=""
data:image/s3,"s3://crabby-images/77f29/77f2940cd573e1c5f003ca2733372c96190e9998" alt=""
Want to learn more about Groovy and Grails? Come to our monthly Groovy / Grails Meetup:
https://www.meetup.com/st-louis-groovy-and-grails-meetup/
Questions?
STLJUG NFJS Building a Microservice Federation with Grails
By Jack Frosch
STLJUG NFJS Building a Microservice Federation with Grails
- 1,965