grails.rest.RestfulController.groovy Maven / Gradle / Ivy
/*
* Copyright 2013 the original author or authors.
*
* 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.
*/
package grails.rest
import grails.artefact.Artefact
import grails.gorm.transactions.ReadOnly
import grails.gorm.transactions.Transactional
import grails.util.GrailsNameUtils
import grails.web.http.HttpHeaders
import org.springframework.http.HttpStatus
import static org.springframework.http.HttpStatus.*
/**
* Base class that can be extended to get the basic CRUD operations needed for a RESTful API.
*
* @author Graeme Rocher
* @since 2.3
*/
@Artefact("Controller")
@ReadOnly
class RestfulController {
static allowedMethods = [save: "POST", update: ["PUT", "POST"], patch: "PATCH", delete: "DELETE"]
Class resource
String resourceName
String resourceClassName
boolean readOnly
RestfulController(Class resource) {
this(resource, false)
}
RestfulController(Class resource, boolean readOnly) {
this.resource = resource
this.readOnly = readOnly
resourceClassName = resource.simpleName
resourceName = GrailsNameUtils.getPropertyName(resource)
}
/**
* Lists all resources up to the given maximum
*
* @param max The maximum
* @return A list of resources
*/
def index(Integer max) {
if (max < 0) { max = null }
params.max = Math.min(max ?: 10, 100)
respond listAllResources(params), model: [("${resourceName}Count".toString()): countResources()]
}
/**
* Shows a single resource
* @param id The id of the resource
* @return The rendered resource or a 404 if it doesn't exist
*/
def show() {
respond queryForResource(params.id)
}
/**
* Displays a form to create a new resource
*/
def create() {
if(handleReadOnly()) {
return
}
respond createResource()
}
/**
* Saves a resource
*/
@Transactional
def save() {
if(handleReadOnly()) {
return
}
def instance = createResource()
instance.validate()
if (instance.hasErrors()) {
transactionStatus.setRollbackOnly()
respond instance.errors, view:'create' // STATUS CODE 422
return
}
saveResource instance
request.withFormat {
form multipartForm {
flash.message = message(code: 'default.created.message', args: [classMessageArg, instance.id])
redirect instance
}
'*' {
response.addHeader(HttpHeaders.LOCATION,
grailsLinkGenerator.link( resource: this.controllerName, action: 'show',id: instance.id, absolute: true,
namespace: hasProperty('namespace') ? this.namespace : null ))
respond instance, [status: CREATED, view:'show']
}
}
}
def edit() {
if(handleReadOnly()) {
return
}
respond queryForResource(params.id)
}
/**
* Updates a resource for the given id
* @param id
*/
@Transactional
def patch() {
update()
}
/**
* Updates a resource for the given id
* @param id
*/
@Transactional
def update() {
if(handleReadOnly()) {
return
}
T instance = queryForResource(params.id)
if (instance == null) {
transactionStatus.setRollbackOnly()
notFound()
return
}
instance.properties = getObjectToBind()
instance.validate()
if (instance.hasErrors()) {
transactionStatus.setRollbackOnly()
respond instance.errors, view:'edit' // STATUS CODE 422
return
}
updateResource instance
request.withFormat {
form multipartForm {
flash.message = message(code: 'default.updated.message', args: [classMessageArg, instance.id])
redirect instance
}
'*'{
response.addHeader(HttpHeaders.LOCATION,
grailsLinkGenerator.link( resource: this.controllerName, action: 'show',id: instance.id, absolute: true,
namespace: hasProperty('namespace') ? this.namespace : null ))
respond instance, [status: OK]
}
}
}
/**
* Deletes a resource for the given id
* @param id The id
*/
@Transactional
def delete() {
if(handleReadOnly()) {
return
}
def instance = queryForResource(params.id)
if (instance == null) {
transactionStatus.setRollbackOnly()
notFound()
return
}
deleteResource instance
request.withFormat {
form multipartForm {
flash.message = message(code: 'default.deleted.message', args: [classMessageArg, instance.id])
redirect action:"index", method:"GET"
}
'*'{ render status: NO_CONTENT } // NO CONTENT STATUS CODE
}
}
/**
* handles the request for write methods (create, edit, update, save, delete) when controller is in read only mode
*
* @return true if controller is read only
*/
protected boolean handleReadOnly() {
if(readOnly) {
render status: HttpStatus.METHOD_NOT_ALLOWED.value()
return true
} else {
return false
}
}
/**
* The object that can be bound to a domain instance. Defaults to the request. Subclasses may override this
* method to return anything that is a valid second argument to the bindData method in a controller. This
* could be the request, a {@link java.util.Map} or a {@link org.grails.databinding.DataBindingSource}.
*
* @return the object to bind to a domain instance
*/
protected getObjectToBind() {
request
}
/**
* Queries for a resource for the given id
*
* @param id The id
* @return The resource or null if it doesn't exist
*/
protected T queryForResource(Serializable id) {
resource.get(id)
}
/**
* Creates a new instance of the resource for the given parameters
*
* @param params The parameters
* @return The resource instance
*/
protected T createResource(Map params) {
resource.newInstance(params)
}
/**
* Creates a new instance of the resource. If the request
* contains a body the body will be parsed and used to
* initialize the new instance, otherwise request parameters
* will be used to initialized the new instance.
*
* @return The resource instance
*/
protected T createResource() {
T instance = resource.newInstance()
bindData instance, getObjectToBind()
instance
}
/**
* List all of resource based on parameters
*
* @return List of resources or empty if it doesn't exist
*/
protected List listAllResources(Map params) {
resource.list(params)
}
/**
* Counts all of resources
*
* @return List of resources or empty if it doesn't exist
*/
protected Integer countResources() {
resource.count()
}
protected void notFound() {
request.withFormat {
form multipartForm {
flash.message = message(code: 'default.not.found.message', args: [classMessageArg, params.id])
redirect action: "index", method: "GET"
}
'*'{ render status: NOT_FOUND }
}
}
/**
* Saves a resource
*
* @param resource The resource to be saved
* @return The saved resource or null if can't save it
*/
protected T saveResource(T resource) {
resource.save flush: true
}
/**
* Updates a resource
*
* @param resource The resource to be updated
* @return The updated resource or null if can't save it
*/
protected T updateResource(T resource) {
saveResource resource
}
/**
* Deletes a resource
*
* @param resource The resource to be deleted
*/
protected void deleteResource(T resource) {
resource.delete flush: true
}
protected String getClassMessageArg() {
message(code: "${resourceName}.label".toString(), default: resourceClassName)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy