Making Modules
Utilizing Reusable Code through ColdBox Modules
Follow Along:
What This Talk Is
- An overview of modules
- Module + ColdBox Superpowers
- Testing strategies for modules
- Ways to quickly go from idea to ForgeBox
Who Am I?
data:image/s3,"s3://crabby-images/1531b/1531b103fb825ba649b9581bbcbec21ca3d9c436" alt=""
Utah
data:image/s3,"s3://crabby-images/6962e/6962e7934cab366eccb07d34e7dea24487f020e9" alt=""
Ortus Solutions
data:image/s3,"s3://crabby-images/884aa/884aab476476789ae259d76eafad097fda9022d3" alt=""
Prolific Module Author
data:image/s3,"s3://crabby-images/e29a2/e29a2aed2d0c24f67abe765e7d3d3186d5d1573d" alt=""
1 wife, 3 kids
data:image/s3,"s3://crabby-images/e2c2a/e2c2aaeda83a4e0f922545caf8b90cb96ab63566" alt=""
Type 1 Diabetic
What is a Module?
Reusable packages of funcationality
/**
* Returns the character at a certain position in a string.
*
* @param str String to be checked.
* @param pos Position to get character from.
* @return Returns a character.
* @author Raymond Camden (ray@camdenfamily.com)
* @version 1, December 3, 2001
*/
function CharAt(str,pos) {
return Mid(str,pos,1);
}
data:image/s3,"s3://crabby-images/0339a/0339ae72af53ffa451e02e89ffe427235d23c0f0" alt=""
So what defines a module?
data:image/s3,"s3://crabby-images/4550f/4550f22c909082632d30221a7a3a3e8744c33634" alt=""
Reusable
Semantically versioned
data:image/s3,"s3://crabby-images/36b7c/36b7c354db58d157517ea99a0a749ea9661822a5" alt=""
data:image/s3,"s3://crabby-images/12397/123975100d19af62f5a908903498ad29caeaa707" alt=""
Can depend on other modules
Shared on forgebox
data:image/s3,"s3://crabby-images/2a0c5/2a0c5754ab04462aadf418bdbc41c76f011ad4de" alt=""
Why Use Modules?
data:image/s3,"s3://crabby-images/3bc47/3bc472e4ebc6cc24decd74a013555d5f1c8f0beb" alt=""
Encapsulate complexity
Encapsulate complexity
data:image/s3,"s3://crabby-images/101da/101da0374c71c178df46ab287f4c94f5358484a0" alt=""
Figure out the java interop once
data:image/s3,"s3://crabby-images/9abf4/9abf4ecc5f40a5b73583bf0dc23b91af54a31671" alt=""
data:image/s3,"s3://crabby-images/141ed/141edd49adbe282be762f85f006283275af7240f" alt=""
Easier to test in isolation
Enforce coding standards
data:image/s3,"s3://crabby-images/76b08/76b0844d105731d00504a2fb84c77d04a09964f9" alt=""
data:image/s3,"s3://crabby-images/4c038/4c0381df2107113c86f0ad13d369d4f6a350a24f" alt=""
Build on the shoulders of giants
Do I need to be using coldbox to use modules?
No...
Benefits of modules without coldbox
- Versioning
- Dependency Management
- Installation Location by Convention (or Setting)
What do I need to make a module?
box.json
Describes your module
- Version
- Dependencies
- Module Type
- Installation Data
- Package Scripts
- Other Metadata
box.json
box.json
{
"name":"cbyaml",
"version":"1.0.1",
"author":"Eric Peterson",
"location":"elpete/cbyaml#v1.0.1",
"homepage":"https://github.com/elpete/cbyaml",
"documentation":"https://github.com/elpete/cbyaml",
"repository":{
"type":"git",
"URL":"https://github.com/elpete/cbyaml"
},
"bugs":"https://github.com/elpete/cbyaml/issues",
"slug":"cbyaml",
"shortDescription":"Provides easy serialization and deserialization of yaml files",
"description":"Provides easy serialization and deserialization of yaml files",
"type":"modules",
"dependencies":{
"cbjavaloader":"^2.0.0+44"
},
"devDependencies":{
"coldbox":"^5.6.2+1021",
"testbox":"^3.0.0"
},
"installPaths":{
"testbox":"testbox/",
"coldbox":"tests/resources/app/coldbox/",
"cbjavaloader":"modules/cbjavaloader/"
},
"scripts":{
"postVersion":"package set location='elpete/cbyaml#v`package version`'"
},
"ignore":[
"**/.*",
"test",
"tests"
]
}
Discover the full schema at https://commandbox.ortusbooks.com/package-management/box.json
Do I need to be using coldbox to use modules?
No...
BUT...
Coldbox gives you superpowers
data:image/s3,"s3://crabby-images/901f8/901f83b2d9deb0a6a99404057655c9493a091abf" alt=""
Not using ColdBox yet? Check out this guide to starting one piece at a time:
Title Text
Subtitle
What do I get with ColdBox modules?
data:image/s3,"s3://crabby-images/f3c8e/f3c8e9bbca913759c1cc8ea655eba9ee798049f1" alt=""
Self-contained ColdBox Applications
What does that mean?
- Full MVC Sub-application
- Models
- Interceptors
- Layouts & Views
- Handlers
- Overridable Settings
- Automatic WireBox Registration
data:image/s3,"s3://crabby-images/d6b0a/d6b0a6fbd68d21b6066fc74f732e70e940fca76d" alt=""
So how do we take advantage of all this?
data:image/s3,"s3://crabby-images/553e6/553e68bba334c93b58e174b9a9d3fa2a06b5b007" alt=""
ModuleConfig.cfc
Bootstraps your module
- Version
- Dependencies
- Module Type
- Installation Data
- Package Scripts
- Other Metadata
ModuleConfig.cfc
Properties
ModuleConfig.cfc
Property | Purpose |
---|---|
name | The unique name of the module |
entryPoint | The default route into this module (i.e. /api/v1) |
cfmapping | The mapping to access this module |
dependencies | Other modules that must be loaded before this module |
autoMapModels | Set to true to have WireBox automatically map your models folder |
Find all the properties at
https://coldbox.ortusbooks.com/hmvc/modules/moduleconfig/public-module-properties-directives
Methods
ModuleConfig.cfc
Method | Purpose |
---|---|
configure() | Only interact with this module. Use onLoad() for cross-module, cross-framework dependencies. |
onLoad() | When you need the framework loaded first. |
onUnload() | Undo what you did in load. |
WireBox Integration
ModuleConfig.cfc
this.autoMapModels = true;
/*
Automatically maps all of your models to
{modelName}@{moduleName}
property name="builder" inject="Builder@qb";
*/
Advanced WireBox Integration
ModuleConfig.cfc
function configure() {
binder.map( "DefaultGrammar" )
.to( "#moduleMapping#.models.grammars.MySQLGrammar" );
}
BONUS:
Moduleconfig.cfc
is also an Interceptor
component {
this.name = "JSONToRC";
this.author = "Eric Peterson";
this.description = "Add the HTTP Request Body to the RC on each request";
this.version = "1.0.0";
function configure() {}
function preProcess( event, interceptData, buffer, rc, prc ) {
var jsonContent = event.getHTTPContent( json = true );
if ( isStruct( jsonContent ) ) {
structAppend( rc, jsonContent );
}
}
}
Examples Of Modules
Single Purpose
- vue-helpers
- redirectBack
- JSONToRC
Utility Libraries
- str
- bcrypt
- cbvalidation
- mementifier
API Wrappers
- cbgithub
- sendgrid-sdk
- aws-cfml
Interceptors
- verify-csrf-interceptor
- cors
- ses-on-request
Frameworks
- qb
- Quick
- hyper
Module Deep Dive
redirectBack
redirectBack
Usage
component {
property name="auth" inject="AuthenticationService@cbauth";
property name="flash" inject="coldbox:flash";
function new( event, rc, prc ){
param prc.errors = flash.get( "registration_form_errors", {} );
event.setView( "registrations/new" );
}
function create( event, rc, prc ){
var result = validateModel(
target = rc,
constraints = {
"email" : {
"required" : true,
"type" : "email",
"uniqueInDatabase" : { "table" : "users", "column" : "email" }
},
"password" : { "required" : true },
"passwordConfirmation" : { "required" : true, "sameAs" : "password" }
}
);
if ( result.hasErrors() ) {
flash.put( "registration_form_errors", result.getAllErrorsAsStruct() );
redirectBack();
return;
}
var user = getInstance( "User" ).create( { "email" : rc.email, "password" : rc.password } );
auth.login( user );
relocate( uri = "/" );
}
}
redirectBack
Directory Structure
data:image/s3,"s3://crabby-images/79f6e/79f6ed473b290625237c8e8bf132c68079512fea" alt=""
redirectBack
box.json
{
"name": "redirectBack",
"version": "1.0.4",
"location": "elpete/redirectBack#v1.0.4",
"slug": "redirectBack",
"shortDescription":
"Caches the last request in the flash scope to give easy redirects back",
"type": "modules",
"scripts": {
"postVersion": "package set location='elpete/redirectBack#v`package version`'",
"onRelease": "publish",
"postPublish": "!git push && git push --tags"
},
"ignore": [
"**/.*",
"test",
"tests"
]
}
redirectBack
ModuleConfig.cfc
this.title = "redirectBack";
this.author = "Eric Peterson";
this.webURL = "https://github.com/elpete/redirectBack";
this.description = "Caches the last request in the flash scope to give easy redirects back";
this.version = "1.0.4";
Properties
redirectBack
ModuleConfig.cfc
function configure() {
/*
* can override any of these settings in your `config/ColdBox.cfc` file.
* `moduleSettings = { redirectBack = { key = "overridden_key" } };`
* Or in CommandBox * `config set modules.redirectBack.key="overridden_key"`
*/
settings = {
"key" = "last_url"
};
interceptors = [
{
class = "#moduleMapping#/interceptors/RedirectBack",
name = "RedirectBack",
properties = {}
}
];
}
configure()
redirectBack
ModuleConfig.cfc
/*
* We use onLoad instead of configure here
* because we want to interact with other
* ColdBox settings, such as application helpers.
*/
function onLoad() {
var helpers = controller.getSetting( "applicationHelper" );
arrayAppend( helpers, "#moduleMapping#/helpers/RedirectBackHelpers.cfm" );
controller.setSetting( "applicationHelper", helpers );
}
onLoad()
redirectBack
ModuleConfig.cfc
function onUnload() {
controller.setSetting(
"applicationHelper",
arrayFilter( controller.getSetting( "applicationHelper" ), function( helper ) {
return helper != "#moduleMapping#/helpers/RedirectBackHelpers.cfm";
} )
);
}
onUnload()
redirectBack
interceptors/RedirectBack.cfc
component extends="coldbox.system.Interceptor" {
property name="moduleSettings" inject="coldbox:moduleSettings:redirectBack";
function postProcess( event, interceptData, buffer, rc, prc ) {
var flash = wirebox.getInstance( dsl = "coldbox:flash" );
if ( ! event.isAjax() ) {
flash.put(
name = moduleSettings.key,
value = event.isSES() ? event.getCurrentRoutedUrl() : event.getCurrentEvent(),
autoPurge = false
);
}
}
}
redirectBack
RedirectBackHelpers.cfm
function redirectBack() {
var moduleSettings = wirebox.getInstance( dsl = "coldbox:moduleSettings:redirectBack" );
var flash = wirebox.getInstance( dsl = "coldbox:flash" );
arguments.event = flash.get( moduleSettings.key, "" );
relocate( argumentCollection = arguments );
}
Deep Dive #2
cbyaml
To the Code!!
Other notable Modules
To the Code!!
Where Do I find Modules?
data:image/s3,"s3://crabby-images/2a0c5/2a0c5754ab04462aadf418bdbc41c76f011ad4de" alt=""
data:image/s3,"s3://crabby-images/f61ce/f61cee7b46e60030bda0a45c73ccb11042c5edb9" alt=""
data:image/s3,"s3://crabby-images/cc651/cc651fdfa468dace6842777da756902ac6dbc129" alt=""
Git Repo
modules_app
Other Module Locations
Pros
Cons
- Versioning (via tags)
- Can be shared (Public repos)
-
Can be private (Private Repos)
- No semantic version ranges
- No ForgeBox slugs
Pros
Cons
- Committed in your repo
- Private to your application
-
Easy to iterate on and develop
-
Great for internal API versions
- No reuse outside your project
- No versions
You said modules are easier to test?
Unit Testing
component extends="testbox.system.BaseSpec" {
function beforeAll() {
include "/root/functions/normalizeToArray.cfm";
}
function run() {
describe( "normalizeToArray", function() {
it( "returns an array unmodified", function() {
var actual = normalizeToArray( [ 1, 2, 3, 4 ] );
expect( actual ).toBe( [ 1, 2, 3, 4 ] );
} );
it( "converts a list to an array", function() {
var actual = normalizeToArray( "1,2,3,4" );
expect( actual ).toBe( [ 1, 2, 3, 4 ] );
} );
} );
}
}
Integration Testing
component extends="tests.resources.ModuleIntegrationSpec" appMapping="/app" {
variables.travisYmlSamplePath = expandPath(
"/tests/resources/sample-files/.travis.sample.yml"
);
function run() {
describe( "Yaml Parser", function() {
it( "can deserialize yaml strings", function() {
var parser = getWireBox().getInstance( "Parser@cbyaml" );
var travisYmlSample = fileRead( variables.travisYmlSamplePath );
var actual = parser.deserialize( travisYmlSample );
expect( actual ).toBe( getTravisYmlAsCF() );
} );
} );
}
function getTravisYmlAsCF() { /* ... */ }
}
Scaffolding Modules
data:image/s3,"s3://crabby-images/7553d/7553d12bc66683c5cf4d96fb91c67433e8cb44f5" alt=""
Steps to share a module
- Create box.json
- Create ModuleConfig.cfc
- Scaffold tests folder
- Copy over Travis CI configuration
- Create git repo
- Create GitHub repo
data:image/s3,"s3://crabby-images/89c30/89c30cbcc52b2a44da4de907c4b0688d4bd9f596" alt=""
data:image/s3,"s3://crabby-images/6c9cc/6c9ccbda19be5b73a217b82586e9d6f8e85208b1" alt=""
For ColdBox Applications
coldbox create app
For New Modules
install cb-module-template
module scaffold my-awesome-module "It will blow your mind!"
data:image/s3,"s3://crabby-images/4202d/4202d168d0252696270e38706ce2bde8baccd383" alt=""
Recap
- Build reusable code
- Test easier
- Share with others
Thanks!
data:image/s3,"s3://crabby-images/a9829/a982919dc3a796124c10f8c5fcdcc9caac496419" alt=""
@_elpete
data:image/s3,"s3://crabby-images/77f8b/77f8b439b1a10ec2eec252bb892149c327f5d3de" alt=""
elpete
data:image/s3,"s3://crabby-images/1513b/1513b2850ea1af841a71a6014020dfae8eff40e8" alt=""
dev.elpete.com
Copy of Making Modules
By Eric Peterson
Copy of Making Modules
Utilizing reusable code through ColdBox Modules
- 573