Copyright 2013, Australian Access Federation
Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
- (c) Australian Access Federation
- Authors: Bradley Beddoes, Shaun Mangelsdorf
All code and assets located in the AAF Application Base repository at https://github.com/ausaccessfed/applicationbase is provided under the Apache 2.0 open source license.
The AAF Application Base is an extensible Grails plugin that provides a number of key features that are used across all AAF projects. With the use of this plugin it should be possible to simply start working on new business logic in a short time frame having all the additional plumbing for a production ready service simply provided for you.
- Grails 2.1.1+ Support
- JQuery and various JQuery libraries for expressive UI
- Twitter Bootstrap for styling and UI layout
- Report generation via HighCharts
- Refinable data via dataTables
- Responsive AAF branded layouts suitable for Phones, Tables and Desktops
- Cross reader HTML email
- Administrative dashboard
- Subject management
- Role management
- Workflow Engine including data store driven Processes and Scripts
- Internal Grails console
- Default application welcome page and dashboard
- Support for Federated authentication via Shibboleth SP
- Eases development with local accounts
- Full i18n implmentation
- Customizable branding per project including full navigation modification
- AAF specific assistance tools including help directives and custom error pages
- Externalizes configuration and logging ready for production deployment
- Supports Git Flow workflow and provides base .gitignore
If you're new to this space you'll want to get a few things up and running before you can start development.
-
A *nix based development machine - I work directly on my Mac but a Virtualbox VM works just as well
-
An install of Java JDK. I am currently using a 7 release:
java -version java version "1.7.0_07" Java(TM) SE Runtime Environment (build 1.7.0_07-b10) Java HotSpot(TM) 64-Bit Server VM (build 23.3-b01, mixed mode)
You can find offical Oracle JDK downloads here or alternatively use an OpenJDK which works just as well.
-
Ensure you have a JAVA_HOME and appropriate JAVA_OPTS environment variables defined in your
~/.bash_profile
, here is an example of mine:export JAVA_HOME='/Library/Java/JavaVirtualMachines/jdk1.7.0_07.jdk/Contents/Home' export JAVA_OPTS='-Xms1024m -Xmx1024m -XX:MaxPermSize=256m'
For Grails work the above memory allocations seem to give me the best performance.
-
Install GVM using the below commands. You will require bash, curl, zip and unzip to be available on your system
curl -s get.gvmtool.net | bash source "~/.gvm/bin/gvm-init.sh" gvm install grails 2.1.1 gvm use grails 2.1.1
Make sure Grails is correctly installed and referenced:
$> which grails
~/.gvm/grails/current/bin/grails
This couldn't be simpler on the surface. Though getting things like workflows up and running will require a little more work and familiarisiation of the underlying codebase.
The AAF applicationbase plugin is distributed in source form only and only available from private AAF git repositories. Any changes required to the applicationbase plugin should be done in the upstream project
-
Create a containing directory called something like 'development'
-
Run
git clone [email protected]:ausaccessfed/applicationbase.git
-
This will give you a local directory called applicationbase
-
Still in your development directory create a new Grails application using at least Grails release 2.1.1 we'll refer to this as appname
-
Your directory should now contain applicationbase and appname
-
Change to your application and edit the file: grails-app/conf/BuildConfig.groovy and add the following (~line 8):
grails.plugin.location.'aaf-application-base' = '../applicationbase'
-
At the root of your project directory structure run the command:
grails base-clean-new-app
-
Follow this with the command:
grails base-create-branding
-
This has given you a number of new files througout your Grails App which you can explore. To start with lets get something basic up and running.
-
Edit the new file application_config.groovy in the root of your project Setting the following:
appName="lowercasenameofapp" e.g. "federationregistry" grails.serverURL="http://brainslave.dev.bradleybeddoes.com:8080/lowercasenameofapp" This is your your dev machine, include appName, no trailing /
development { active = true } This will make dev easier, ensure FALSE in production
You can and should extend the values stored in application_config.groovy to suit your own applications needs. Be sure to also document new values in application_config.groovy.orig to assist deployers of your application.
application_config.groovy is excluded from git by default
-
Lets test everything. Export the environment variable config_dir:
export config_dir=.
-
Run your application:
grails run-app
-
Navigate to your application as indicated by Grails output
-
You should have a public welcome page, be able to login and then access all the default functionality.
-
To become an administrator start a MySQL session and run the following:
select id, cn, email from subjects; # Take not of your accounts ID. insert into base_role_subjects (subject_id, role_id) values(,1);
-
Refresh your browser, you should have full admin rights and can use the UI to on assign this in the future.
The commands you ran earlier (clean-new-app and create-branding) have taken your vanilla Grails app and both removed css/js/images that we don't use and added css/js/images that we do.
In particular the following files will help you customise your new application to suit your needs.
- grails-app/i18n/messages-branding.properties - This contains all the strings shown throughout the application. You should change them to be relevant to application you're building.
- grails-app/views/layout/*.gsp - This controls how the application itself looks. In most cases you won't need to modify these but they are there for special circumstances
- grails-app/controllers/DashboardController.groovy and grails-app/views/dashboard/*.gsp - Your public and authenticated dashboard views. You will want to customise the controller and the view files to suit your application. I strongly recommend moving the controller into a package structure such as aaf.yourappname - all your subsquent controllers should be packaged as well.
- grails-app/views/templates/branding/*.gsp - These templates give the application its header, footer and most importantly navigation views. All applications implementing the Application Base will want to extend navigation markup (along with UrlMappings.groovy) to provide access to the logic they serve.
The Application Base provides an extensive set of scaffolding templates to make creation of CRUD objects and views as fast as possible. Views fully utilise Bootstrap CSS and javascript libraries supported by AAF Application Base.
To make use of this functionality first design your domain objects to store all data that is relevant to your application then use Grails scaffolding as normal. Support for all the grails generate-* commands are provided as base-generate-* overrides (base-generate-controller, base-generate-views, base-generate-all) is provided see Grails Scaffolding for more details.
When creating a controller a full test suite is also generated in test/unit/ControllerSpec.groovy.
While scaffolding is very useful for developers it onlg gives you around 80% of what I'd consider production ready code. Be sure to check all required permissions, restrict the use of binding and perform other quality audits of the generated code.
You will need to create your own i18n codes and provide your own URL Mappings for generated controllers and views.
Scaffolding is provided by the grails base-create-app
command by default. If you're already using AAF Application Base 0.1 then a new command grails base-create-templates
will populate the scaffolding components for you.
Be sure to commit your scaffold output once you're happy with it so you don't accidently overwrite it if the commands get executed again.
In certain cases the scaffolding generated for your object's controller will need manual fine tuning before you can get test cases to pass. Situations where we've seen this occur:
- Use of a mandatory Date field in your domain object. This will require you to provide a date format in your test case that your controller is expecting to recieve to create the Date field from.
- A set/list of domain object references that much have at least one entry. You'll need to manually provide this data, possibly requiring additional Build annotations in your test code.
- Problems with fields that must be unique. In this case you'll need to fine tune how the build process works via the build-test-data plugin TestDataConfig configuration.
- Your edge case here please!
Within the Application Base a Subject is the key building block. Subjects are created (usually automatically) when a new users enters the system via supported authentication realm (federation for production, local accounts for development. See FederatedRealm.groovy)
Subjects can then be granted membership to many Roles. Both a Subject and Role can be granted Permissions. A permission directly enforces security on requests, actions, views, markup and internal data it is the lowest level of our security structure. A permission takes the form x:y:z to an unlimited number of levels. Optionally a Subject or Role might be granted a permission with a wildcard x:y:* meaning a state of true will be returned for any permission check under x:y. e.g A permissions check for x:y:delete
would return true for this Subject or Role.
To enforce access control code should always check for valid permissions. It can achieve this several ways. Within a groovy code block:
import org.apache.shiro.SecurityUtils
…
if(SecurityUtils.subject.isPermitted("x:y:z")) {
}
else {
log.warn("Attempt to do action by $subject was denied")
response.sendError(403)
}
Within a GSP:
<aaf:isAdministrator>
</aaf:isAdministrator>
<aaf:hasPermission target="x:y:z">
</aaf:hasPermission>
There are a lot more GSP tag options defined in AAFBaseTagLib.groovy within the Application Base project, check them out.
Roles and Permissions can be created by administrators using the provided UI. You may also choose to implement your own subset of this UI in which case RoleService and PermissionsService will be useful for you to use when interacting with this subsystem.
A full workflow engine is provided but as yet not well documented. We need to update this section over time as understanding needs to be shared through the team. In the meantime have a look at the extensive test cases to understand how this piece works.
As you use the Application Base you'll probably find gaps in documentation and test cases that explain how things function. Please bring these up so we can continue to refine over time.