From a4ae0d230c9cca60eeceadb73f6acb60be67010e Mon Sep 17 00:00:00 2001 From: Moritz Kobel Date: Tue, 23 Aug 2011 08:53:02 +0200 Subject: [PATCH] - Upgrade to Grails 2.0.0M1 - Added sendXmppPacket method to send a "raw" Packet --- XmppGrailsPlugin.groovy | 325 +++++++++++++++++++++++-- application.properties | 8 +- plugin.xml | 6 +- web-app/WEB-INF/applicationContext.xml | 41 ++-- web-app/WEB-INF/sitemesh.xml | 2 +- web-app/WEB-INF/tld/c.tld | 25 +- web-app/WEB-INF/tld/fmt.tld | 14 +- 7 files changed, 350 insertions(+), 71 deletions(-) diff --git a/XmppGrailsPlugin.groovy b/XmppGrailsPlugin.groovy index 5058888..3b06597 100644 --- a/XmppGrailsPlugin.groovy +++ b/XmppGrailsPlugin.groovy @@ -1,32 +1,61 @@ -import org.jivesoftware.smack.RosterListener import org.codehaus.groovy.grails.commons.GrailsClassUtils -import org.codehaus.groovy.grails.commons.ConfigurationHolder as CFG -import grails.util.GrailsUtil import grails.util.GrailsNameUtils +import org.jivesoftware.smack.RosterListener +import org.codehaus.groovy.grails.commons.GrailsClassUtils +import grails.util.GrailsUtil +import grails.util.GrailsNameUtils import org.jivesoftware.smack.packet.Packet import org.jivesoftware.smack.packet.Message -import org.jivesoftware.smack.PacketListener import org.jivesoftware.smack.RosterListener -import org.jivesoftware.smack.filter.PacketFilter import org.jivesoftware.smack.Chat import org.jivesoftware.smack.packet.Presence import org.codehaus.groovy.grails.commons.ServiceArtefactHandler import org.apache.commons.lang.StringUtils +import org.jivesoftware.smack.PacketListener +import org.jivesoftware.smack.RosterListener +import org.jivesoftware.smack.filter.PacketFilter +import org.jivesoftware.smack.Chat +import org.jivesoftware.smack.packet.Presence + +import org.codehaus.groovy.grails.commons.ServiceArtefactHandler +import org.apache.commons.lang.StringUtils + class XmppGrailsPlugin { // the plugin version - def version = "0.2" + def version = "0.4.3" // the version or versions of Grails the plugin is designed for - def grailsVersion = "1.1 > *" // resources that are excluded from plugin packaging + def grailsVersion = "1.1 > *" + // resources that are excluded from plugin packaging def pluginExcludes = [ "grails-app/views/error.gsp", - "grails-app/services/org/grails/xmpp/*Service.groovy", "grails-app/controllers/org/grails/xmpp/*Controller.groovy", "grails-app/conf/XmppBootStrap.groovy", "src/groovy/org/grails/xmpp/ClosureMessageListenerAdapter.groovy", "scripts/*.groovy" ] + "grails-app/services/org/grails/xmpp/*Service.groovy", + "grails-app/controllers/org/grails/xmpp/*Controller.groovy", + "grails-app/conf/XmppBootStrap.groovy", + "src/groovy/org/grails/xmpp/ClosureMessageListenerAdapter.groovy", + "scripts/*.groovy" + ] def author = "Fábio Franco Uechi" def authorEmail = "fabio.uechi@gmail.com" def title = "Grails XMPP Plugin" def description = '''\\ -Provides XMPP/Jabber based services to grails applications. Built upon the Smack API this plugin allows an application to send and receive xmpp packets (presence, message, etc). Basically it creates a Roster which automatically connects, to a pre configured host, at application startup. All the mentioned xmpp services rely upon this Roster. Eases the development of real-time web applications. One of the main features is the ability to expose methods as commands (method name = command) making it easier to develop XmppBots, for example. Another feature is auto-detection of PacketListeners and RosterListeners. +Provides XMPP/Jabber based services to grails applications. Built upon the Smack API this plugin allows +an application to send and receive xmpp packets (presence, message, etc). Basically it creates a Roster +which automatically connects, to a pre configured host, at application startup. All the mentioned xmpp +services rely upon this Roster. Eases the development of real-time web applications. +One of the main features is the ability to expose methods as commands (method name = command) making it +easier to develop XmppBots, for example. +Another feature is auto-detection of PacketListeners and RosterListeners. ''' // URL to the plugin's documentation def documentation = "http://grails.org/plugin/xmpp" def loadAfter = ['services', 'controllers'] def observe = ['services', 'controllers'] - def doWithSpring = { - registerXmppBaseBeans(delegate, log) - xmppAgent (org.grails.xmpp.XmppAgent){ bean -> bean.destroyMethod = "destroy" connection = xmppConnection username = CFG.config.xmpp.username password = CFG.config.xmpp.password } + + + def doWithSpring = { + + registerXmppBaseBeans(delegate, log) + + xmppAgent (org.grails.xmpp.XmppAgent){ bean -> + bean.destroyMethod = "destroy" + connection = xmppConnection + username = application.config.xmpp.username + password = application.config.xmpp.password + } /* * For each service containing the expose xmpp attribute * creates and registers the respectice MessageListenerAdapter bean @@ -36,20 +65,272 @@ Provides XMPP/Jabber based services to grails applications. Built upon the Smack registerXmppMessageListenerBean(service, delegate, log) } } - def registerXmppBaseBeans (beanBuilder, log) { beanBuilder.with { connectionConfiguration (org.jivesoftware.smack.ConnectionConfiguration, CFG.config.xmpp.connection.host, CFG.config.xmpp.connection.port, CFG.config.xmpp.connection.service ) { SASLAuthenticationEnabled = CFG.config.xmpp.connection.SASLAuthenticationEnabled reconnectionAllowed = true } xmppConnection (org.jivesoftware.smack.XMPPConnection, ref(connectionConfiguration)) { } } } def registerXmppMessageListenerBean(service, beanBuilder, log) { def serviceClass = service.getClazz() def exposeList = GrailsClassUtils.getStaticPropertyValue(serviceClass, 'expose') if (exposeList!=null && exposeList.contains('xmpp')) { def listenerMethod = GrailsClassUtils.getStaticPropertyValue(serviceClass, 'listenerMethod') if (!listenerMethod) listenerMethod = org.grails.xmpp.MessageListenerAdapter.ORIGINAL_DEFAULT_LISTENER_METHOD def xmppCommandPrefix = GrailsClassUtils.getStaticPropertyValue(serviceClass, 'xmppCommandPrefix') // comparing to null to allow "prefixless" commands if (xmppCommandPrefix == null) xmppCommandPrefix = org.grails.xmpp.MessageListenerAdapter.ORIGINAL_DEFAULT_COMMAND_PREFIX def xmppCommandMethodSuffix = GrailsClassUtils.getStaticPropertyValue(serviceClass, 'xmppCommandMethodSuffix') // comparing to null to allow "suffixless" commands if (xmppCommandMethodSuffix == null) xmppCommandMethodSuffix = org.grails.xmpp.MessageListenerAdapter.ORIGINAL_DEFAULT_COMMAND_METHOD_SUFFIX beanBuilder.with { "${service.shortName}XmppMessageListener"(org.grails.xmpp.MessageListenerAdapter, ref("${service.propertyName}")) { defaultListenerMethod = listenerMethod defaultXmppCommandPrefix = xmppCommandPrefix defaultXmppCommandMethodSuffix = xmppCommandMethodSuffix } } } } + + def registerXmppBaseBeans (beanBuilder, log) { + beanBuilder.with { + connectionConfiguration (org.jivesoftware.smack.ConnectionConfiguration, application.config.xmpp.connection.host, application.config.xmpp.connection.port, application.config.xmpp.connection.service ) { + SASLAuthenticationEnabled = application.config.xmpp.connection.SASLAuthenticationEnabled + reconnectionAllowed = true + } + + xmppConnection (org.jivesoftware.smack.XMPPConnection, ref(connectionConfiguration)) { + } + + } + } + + def registerXmppMessageListenerBean(service, beanBuilder, log) { + def serviceClass = service.getClazz() + def exposeList = GrailsClassUtils.getStaticPropertyValue(serviceClass, 'expose') + if (exposeList!=null && exposeList.contains('xmpp')) { + + def listenerMethod = GrailsClassUtils.getStaticPropertyValue(serviceClass, 'listenerMethod') + if (!listenerMethod) + listenerMethod = org.grails.xmpp.MessageListenerAdapter.ORIGINAL_DEFAULT_LISTENER_METHOD + + def xmppCommandPrefix = GrailsClassUtils.getStaticPropertyValue(serviceClass, 'xmppCommandPrefix') + // comparing to null to allow "prefixless" commands + if (xmppCommandPrefix == null) + xmppCommandPrefix = org.grails.xmpp.MessageListenerAdapter.ORIGINAL_DEFAULT_COMMAND_PREFIX + + def xmppCommandMethodSuffix = GrailsClassUtils.getStaticPropertyValue(serviceClass, 'xmppCommandMethodSuffix') + // comparing to null to allow "suffixless" commands + if (xmppCommandMethodSuffix == null) + xmppCommandMethodSuffix = org.grails.xmpp.MessageListenerAdapter.ORIGINAL_DEFAULT_COMMAND_METHOD_SUFFIX + + beanBuilder.with { + "${service.shortName}XmppMessageListener"(org.grails.xmpp.MessageListenerAdapter, ref("${service.propertyName}")) { + defaultListenerMethod = listenerMethod + defaultXmppCommandPrefix = xmppCommandPrefix + defaultXmppCommandMethodSuffix = xmppCommandMethodSuffix + } + } + } + } + def doWithApplicationContext = { applicationContext -> - // Checks all services and controllers which implement // PackageListener or MessageListener and add to xmppAgent application.serviceClasses?.each { service -> registerXmppMessageListener(service, applicationContext) + // Checks all services and controllers which implement + // PackageListener or MessageListener and add to xmppAgent + application.serviceClasses?.each { service -> + registerXmppMessageListener(service, applicationContext) } - //initializes xmppAgent - if (CFG.config.xmpp.autoStartup) + + //initializes xmppAgent + if (application.config.xmpp.autoStartup) applicationContext.xmppAgent.connect() } - def registerXmppMessageListener(service, applicationContext) { def serviceClass = service.getClazz() def exposeList = GrailsClassUtils.getStaticPropertyValue(serviceClass, 'expose') if (exposeList!=null && exposeList.contains('xmpp')) { //println ">>>> Adding XMPP listener for ${service.shortName} to xmppAgent" applicationContext.xmppAgent.packetListeners.add(applicationContext.getBean("${service.shortName}XmppMessageListener")) } if(PacketListener.class.isAssignableFrom(serviceClass)) { def svc = applicationContext.getBean("${service.propertyName}") applicationContext.xmppAgent.packetListeners.add(svc) } if(RosterListener.class.isAssignableFrom(serviceClass)) { def svc = applicationContext.getBean("${service.propertyName}") applicationContext.xmppAgent.rosterListeners.add(svc) } } def doWithDynamicMethods = { ctx -> def xmppAgent = ctx.getBean("xmppAgent") [application.controllerClasses, application.serviceClasses].each { it.each { if (it.clazz.name != "XmppService") { addXmppMethodsToClass(xmppAgent, it.clazz) } } } } - def addXmppMethodsToClass(xmppAgent, clazz) { [ sendXmppMessage: "sendXmppMessage", changeXmppRosterStatus: "changeXmppRosterStatus", addXmppContact: "addXmppContact", removeXmppContact: "removeXmppContact", // blockXmppContact: "blockXmppContact", getAllXmppContacts: "getAllXmppContacts" ].each { m, i -> clazz.metaClass."$m" << this."$i".curry(xmppAgent) } } def onChange = { event -> + + def registerXmppMessageListener(service, applicationContext) { + def serviceClass = service.getClazz() + def exposeList = GrailsClassUtils.getStaticPropertyValue(serviceClass, 'expose') + + if (exposeList!=null && exposeList.contains('xmpp')) { + //println ">>>> Adding XMPP listener for ${service.shortName} to xmppAgent" + applicationContext.xmppAgent.packetListeners.add(applicationContext.getBean("${service.shortName}XmppMessageListener")) + } + + if(PacketListener.class.isAssignableFrom(serviceClass)) { + def svc = applicationContext.getBean("${service.propertyName}") + applicationContext.xmppAgent.packetListeners.add(svc) + } + + if(RosterListener.class.isAssignableFrom(serviceClass)) { + def svc = applicationContext.getBean("${service.propertyName}") + applicationContext.xmppAgent.rosterListeners.add(svc) + } + } + + def doWithDynamicMethods = { ctx -> + def xmppAgent = ctx.getBean("xmppAgent") + [application.controllerClasses, application.serviceClasses].each { + it.each { + if (it.clazz.name != "XmppService") { + addXmppMethodsToClass(xmppAgent, it.clazz) + } + } + } + } + + def addXmppMethodsToClass(xmppAgent, clazz) { + [ + sendXmppMessage: "sendXmppMessage", + sendXmppPacket: "sendXmppPacket", + changeXmppRosterStatus: "changeXmppRosterStatus", + addXmppContact: "addXmppContact", + removeXmppContact: "removeXmppContact", + // blockXmppContact: "blockXmppContact", + getAllXmppContacts: "getAllXmppContacts" + ].each { m, i -> + clazz.metaClass."$m" << this."$i".curry(xmppAgent) + } + } + + def onChange = { event -> // Code that is executed when any artefact that this plugin is // watching is modified and reloaded. The event contains: event.source, - // event.application, event.manager, event.ctx, and event.plugin. if (event.source && event.ctx) { def xmppAgent = event.ctx.getBean('xmppAgent') if (application.isControllerClass(event.source)) { def controllerClass = event.application.getControllerClass(event.source?.name) addXmppMethodsToClass(xmppAgent, controllerClass.clazz) } else if (application.isServiceClass(event.source)) { boolean isNew = event.application.getServiceClass(event.source?.name) == null def serviceClass = application.addArtefact(ServiceArtefactHandler.TYPE, event.source).clazz def tgt = event.ctx.getBean(GrailsNameUtils.getPropertyName(event.source)) if (!isNew) { def exposeList = GrailsClassUtils.getStaticPropertyValue(event.source, 'expose') if (exposeList!=null && exposeList.contains('xmpp')) { xmppAgent.connection.removePacketListener(event.ctx.getBean("${GrailsNameUtils.getShortName(event.source)}XmppMessageListener")) event.ctx.removeBeanDefinition("${GrailsNameUtils.getShortName(event.source)}XmppMessageListener") } if(PacketListener.class.isAssignableFrom(event.source)) { xmppAgent.connection.removePacketListener(tgt) } if(RosterListener.class.isAssignableFrom(event.source)) { xmppAgent.roster.removeRosterListener(tgt) } } serviceClass = event.application.getServiceClass(event.source?.name) def newBeans = beans { registerXmppMessageListenerBean(serviceClass, delegate, log) } newBeans.beanDefinitions.each { n,d -> event.ctx.registerBeanDefinition(n, d) } addXmppMethodsToClass(xmppAgent, serviceClass.clazz) xmppAgent.unregisterListeners() doWithApplicationContext(event.ctx) xmppAgent.registerListeners() } } } + // event.application, event.manager, event.ctx, and event.plugin. + if (event.source && event.ctx) { + def xmppAgent = event.ctx.getBean('xmppAgent') + if (application.isControllerClass(event.source)) { + def controllerClass = event.application.getControllerClass(event.source?.name) + addXmppMethodsToClass(xmppAgent, controllerClass.clazz) + + } else if (application.isServiceClass(event.source)) { + + boolean isNew = event.application.getServiceClass(event.source?.name) == null + def serviceClass = application.addArtefact(ServiceArtefactHandler.TYPE, event.source).clazz + def tgt = event.ctx.getBean(GrailsNameUtils.getPropertyName(event.source)) + + if (!isNew) { + + def exposeList = GrailsClassUtils.getStaticPropertyValue(event.source, 'expose') + if (exposeList!=null && exposeList.contains('xmpp')) { + xmppAgent.connection.removePacketListener(event.ctx.getBean("${GrailsNameUtils.getShortName(event.source)}XmppMessageListener")) + event.ctx.removeBeanDefinition("${GrailsNameUtils.getShortName(event.source)}XmppMessageListener") + } + if(PacketListener.class.isAssignableFrom(event.source)) { + xmppAgent.connection.removePacketListener(tgt) + } + if(RosterListener.class.isAssignableFrom(event.source)) { + xmppAgent.roster.removeRosterListener(tgt) + } + } + + serviceClass = event.application.getServiceClass(event.source?.name) + + def newBeans = beans { + registerXmppMessageListenerBean(serviceClass, delegate, log) + } + newBeans.beanDefinitions.each { n,d -> + event.ctx.registerBeanDefinition(n, d) + } + addXmppMethodsToClass(xmppAgent, serviceClass.clazz) + xmppAgent.unregisterListeners() + doWithApplicationContext(event.ctx) + xmppAgent.registerListeners() + } + } + } + + def onConfigChange = { event -> + + def wasConnectedBeforeConfigChange = applicationContext.xmppAgent.connection?.isConnected() + def pList = applicationContext.xmppAgent.packetListeners + def rList = applicationContext.xmppAgent.rosterListeners + + if (wasConnectedBeforeConfigChange) { + applicationContext.xmppAgent.disconnect() + } - def onConfigChange = { event -> def wasConnectedBeforeConfigChange = applicationContext.xmppAgent.connection?.isConnected() def pList = applicationContext.xmppAgent.packetListeners def rList = applicationContext.xmppAgent.rosterListeners if (wasConnectedBeforeConfigChange) { applicationContext.xmppAgent.disconnect() } event.ctx.removeBeanDefinition("connectionConfiguration") event.ctx.removeBeanDefinition("xmppConnection") def newBeans = beans { registerXmppBaseBeans(delegate, log) } newBeans.beanDefinitions.each { n,d -> event.ctx.registerBeanDefinition(n, d) } applicationContext.xmppAgent.connection = applicationContext.xmppConnection applicationContext.xmppAgent.username = CFG.config.xmpp.username applicationContext.xmppAgent.password = CFG.config.xmpp.password applicationContext.xmppAgent.packetListeners = pList applicationContext.xmppAgent.rosterListeners = rList doWithDynamicMethods(event.ctx) - if (CFG.config.xmpp.autoStartup || wasConnectedBeforeConfigChange) { applicationContext.xmppAgent.connect() } } // xmpp utility closures def addXmppContact = { xmppAgent, to -> if (to instanceof List) { to.each { user -> xmppAgent.roster.createEntry(user, user, null) } } else { xmppAgent.roster.createEntry(to, to, null) } } def removeXmppContact = { xmppAgent, to -> if (to instanceof List) { to.each { user -> user = StringUtils.substringBefore(user,"/") log.debug "removing ${user}" xmppAgent.roster.removeEntry(xmppAgent.roster.getEntry(user)) } } else { to = StringUtils.substringBefore(to,"/") log.debug "looking for entry ${to}" def entry = xmppAgent.roster.getEntry(to) if (entry) { log.debug "found ${to}" xmppAgent.roster.removeEntry(entry) } else { log.debug "not found entry ${to}" } } } def sendXmppMessage = { xmppAgent, to, text -> try{ def chat def msgObj if (to instanceof List) { to.each { addr -> chat = xmppAgent.connection.chatManager.createChat(addr, null) msgObj = new Message(addr, Message.Type.chat) msgObj.body = text chat.sendMessage(msgObj) } } else { chat = xmppAgent.connection.chatManager.createChat(to, null) msgObj = new Message(to, Message.Type.chat) msgObj.body = text chat.sendMessage(msgObj) } } catch (Exception e) { log.error "Failed to send message", e } } def getAllXmppContacts = { xmppAgent -> return xmppAgent.roster.entries } def changeXmppRosterStatus = { xmppAgent, newStatus, presenceType -> if (!presenceType) changeXmppRosterStatus2(xmppAgent, newStatus) Presence presence = new Presence(presenceType); presence.setStatus(newStatus); xmppAgent.connection.sendPacket(presence); } def changeXmppRosterStatus2 = { xmppAgent, newStatus -> Presence presence = new Presence(org.jivesoftware.smack.packet.Presence.Type.available); presence.setStatus(newStatus); xmppAgent.connection.sendPacket(presence); } -} \ No newline at end of file + event.ctx.removeBeanDefinition("connectionConfiguration") + event.ctx.removeBeanDefinition("xmppConnection") + + def newBeans = beans { + registerXmppBaseBeans(delegate, log) + } + newBeans.beanDefinitions.each { n,d -> + event.ctx.registerBeanDefinition(n, d) + } + + applicationContext.xmppAgent.connection = applicationContext.xmppConnection + applicationContext.xmppAgent.username = application.config.xmpp.username + applicationContext.xmppAgent.password = application.config.xmpp.password + + applicationContext.xmppAgent.packetListeners = pList + applicationContext.xmppAgent.rosterListeners = rList + + doWithDynamicMethods(event.ctx) + + if (application.config.xmpp.autoStartup || wasConnectedBeforeConfigChange) { + applicationContext.xmppAgent.connect() + } + } + + +// xmpp utility closures + + def addXmppContact = { xmppAgent, to -> + if (to instanceof List) { + to.each { user -> + xmppAgent.roster.createEntry(user, user, null) + } + } else { + xmppAgent.roster.createEntry(to, to, null) + } + } + + def removeXmppContact = { xmppAgent, to -> + + if (to instanceof List) { + to.each { user -> + user = StringUtils.substringBefore(user,"/") + log.debug "removing ${user}" + xmppAgent.roster.removeEntry(xmppAgent.roster.getEntry(user)) + } + } else { + to = StringUtils.substringBefore(to,"/") + log.debug "looking for entry ${to}" + def entry = xmppAgent.roster.getEntry(to) + if (entry) { + log.debug "found ${to}" + xmppAgent.roster.removeEntry(entry) + } + else { + log.debug "not found entry ${to}" + } + + } + } + + def sendXmppMessage = { xmppAgent, to, text -> + try{ + def chat + def msgObj + if (to instanceof List) { + to.each { addr -> + chat = xmppAgent.connection.chatManager.createChat(addr, null) + msgObj = new Message(addr, Message.Type.chat) + msgObj.body = text + chat.sendMessage(msgObj) + } + } else { + chat = xmppAgent.connection.chatManager.createChat(to, null) + msgObj = new Message(to, Message.Type.chat) + msgObj.body = text + chat.sendMessage(msgObj) + } + } catch (Exception e) { + log.error "Failed to send message", e + } + } + def sendXmppPacket = { xmppAgent, pkg -> + try{ + xmppAgent.connection.sendPacket(pkg) + } catch (Exception e) { + log.error "Failed to send packet", e + } + } + + def getAllXmppContacts = { xmppAgent -> + return xmppAgent.roster.entries + } + + def changeXmppRosterStatus = { xmppAgent, newStatus, presenceType -> + if (!presenceType) + changeXmppRosterStatus2(xmppAgent, newStatus) + + Presence presence = new Presence(presenceType); + presence.setStatus(newStatus); + xmppAgent.connection.sendPacket(presence); + } + + def changeXmppRosterStatus2 = { xmppAgent, newStatus -> + Presence presence = new Presence(org.jivesoftware.smack.packet.Presence.Type.available); + presence.setStatus(newStatus); + xmppAgent.connection.sendPacket(presence); + } + + +} + + diff --git a/application.properties b/application.properties index be30ae7..a3a43cf 100644 --- a/application.properties +++ b/application.properties @@ -1,6 +1,6 @@ #Grails Metadata file -#Sat Jun 19 03:59:40 PDT 2010 -app.grails.version=1.3.1 +#Mon Aug 22 15:53:09 CEST 2011 +app.grails.version=2.0.0.M1 app.name=grails-xmpp -plugins.hibernate=1.3.1 -plugins.tomcat=1.3.1 +plugins.hibernate=2.0.0.M1 +plugins.tomcat=2.0.0.M1 diff --git a/plugin.xml b/plugin.xml index 40256df..96f7499 100644 --- a/plugin.xml +++ b/plugin.xml @@ -1,4 +1,4 @@ - + Fábio Franco Uechi fabio.uechi@gmail.com Grails XMPP Plugin @@ -12,10 +12,8 @@ easier to develop XmppBots, for example. Another feature is auto-detection of PacketListeners and RosterListeners. http://grails.org/plugin/xmpp + XmppGrailsPlugin - Config - DataSource - UrlMappings XmppBootStrap org.grails.xmpp.XmppController org.grails.xmpp.DefaultXmppBotService diff --git a/web-app/WEB-INF/applicationContext.xml b/web-app/WEB-INF/applicationContext.xml index 6f42796..69fbef3 100644 --- a/web-app/WEB-INF/applicationContext.xml +++ b/web-app/WEB-INF/applicationContext.xml @@ -6,37 +6,28 @@ http://www.springframework.org/schema/beans http://www.springframework.org/schem Grails application factory bean - - + + A bean that manages Grails plugins - - + + - - - - - - + + + + + + - - - + - - - classpath*:**/grails-app/**/*.groovy - - - - - - utf-8 - - + + + utf-8 + + \ No newline at end of file diff --git a/web-app/WEB-INF/sitemesh.xml b/web-app/WEB-INF/sitemesh.xml index a547b41..72399ce 100644 --- a/web-app/WEB-INF/sitemesh.xml +++ b/web-app/WEB-INF/sitemesh.xml @@ -5,7 +5,7 @@ + class="org.codehaus.groovy.grails.web.sitemesh.GrailsHTMLPageParser" /> diff --git a/web-app/WEB-INF/tld/c.tld b/web-app/WEB-INF/tld/c.tld index 22698c9..5e18236 100644 --- a/web-app/WEB-INF/tld/c.tld +++ b/web-app/WEB-INF/tld/c.tld @@ -1,13 +1,13 @@ - - - JSTL 1.1 core library + xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-jsptaglibrary_2_1.xsd" + version="2.1"> + + JSTL 1.2 core library JSTL core - 1.1 + 1.2 c http://java.sun.com/jsp/jstl/core @@ -74,7 +74,7 @@ not the body content should be processed. Name of the exported scoped variable for the resulting value of the test condition. The type -of the scoped variable is Boolean. +of the scoped variable is Boolean. var false @@ -174,6 +174,9 @@ Collection of items to iterate over. false true java.lang.Object + + java.lang.Object + @@ -253,6 +256,9 @@ String of tokens to iterate over. true true java.lang.String + + java.lang.String + @@ -322,7 +328,7 @@ visibility. Like <%= ... >, but for expressions. - + out org.apache.taglibs.standard.tag.rt.core.OutTag JSP @@ -467,6 +473,9 @@ Expression to be evaluated. value false true + + java.lang.Object + diff --git a/web-app/WEB-INF/tld/fmt.tld b/web-app/WEB-INF/tld/fmt.tld index 3b9a54a..74a6594 100644 --- a/web-app/WEB-INF/tld/fmt.tld +++ b/web-app/WEB-INF/tld/fmt.tld @@ -2,12 +2,12 @@ - - JSTL 1.1 i18n-capable formatting library + xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-jsptaglibrary_2_1.xsd" + version="2.1"> + + JSTL 1.2 i18n-capable formatting library JSTL fmt - 1.1 + 1.2 fmt http://java.sun.com/jsp/jstl/fmt @@ -55,7 +55,7 @@ and may contain a two-letter (upper-case) country code (as defined by ISO-3166). Language and country codes must be separated by hyphen (-) or underscore -(_). +(_). value true @@ -496,7 +496,7 @@ Date and/or time to be formatted. Specifies whether the time, the date, or both the time and date components of the given -date are to be formatted. +date are to be formatted. type false