All Downloads are FREE. Search and download functionalities are using the official Maven repository.

grails.testing.web.UrlMappingsUnitTest.groovy Maven / Gradle / Ivy

There is a newer version: 2023.2.0-M1
Show newest version
/*
 * Copyright 2016-2023 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
 *
 *      https://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.testing.web

import groovy.text.Template
import groovy.transform.CompileDynamic
import junit.framework.AssertionFailedError
import junit.framework.ComparisonFailure

import grails.core.GrailsControllerClass
import grails.web.UrlConverter
import grails.web.mapping.UrlCreator
import grails.web.mapping.UrlMappingInfo
import grails.web.mapping.UrlMappingsHolder

import org.grails.core.artefact.ControllerArtefactHandler
import org.grails.core.artefact.UrlMappingsArtefactHandler
import org.grails.gsp.GroovyPagesTemplateEngine
import org.grails.testing.ParameterizedGrailsUnitTest
import org.grails.web.mapping.UrlMappingsHolderFactoryBean
import org.grails.web.mapping.mvc.GrailsControllerUrlMappingInfo

trait UrlMappingsUnitTest implements ParameterizedGrailsUnitTest, GrailsWebUnitTest {

    public static final String KEY_EXCEPTION = 'exception'
    private final List assertionKeys = ['controller', 'action', 'view']

    Class[] getControllersToMock() {
        []
    }

    void configuredMockedControllers() {
        for (Class c : controllersToMock) {
            GrailsControllerClass controllerArtefact = (GrailsControllerClass) grailsApplication.addArtefact(ControllerArtefactHandler.TYPE, c)
            controllerArtefact.initialize()
            defineBeans {
                "$controllerArtefact.name"(c) { bean ->
                    bean.scope = 'prototype'
                    bean.autowire = true
                }
            }
        }
        getArtefactInstance()
    }

    /**
     * @return The {@link UrlMappingsHolder} bean
     */
    UrlMappingsHolder getUrlMappingsHolder() {
        applicationContext.getBean('grailsUrlMappingsHolder', UrlMappingsHolder)
    }

    /**
     * Maps a URI and returns the appropriate controller instance
     *
     * @param uri The URI to map
     * @return The controller instance
     */
    Object mapURI(String uri) {
        UrlMappingsHolder mappingsHolder = getUrlMappingsHolder()

        UrlMappingInfo[] mappingInfos = mappingsHolder.matchAll(uri, request.method)
        for (UrlMappingInfo info in mappingInfos) {
            def backupParams = new HashMap(webRequest.params)
            info.configure(webRequest)

            webRequest.params.putAll(backupParams)
            if (info.viewName == null && info.URI == null) {
                if (info instanceof GrailsControllerUrlMappingInfo) {
                    GrailsControllerClass controller = info.controllerClass
                    if (controller != null) {
                        return applicationContext.getBean(controller.name)
                    }
                }
            }
        }
    }

    private boolean checkController(String controller, boolean throwEx) {
        GrailsControllerClass controllerClass = getControllerClass(controller)
        if (!controllerClass && throwEx) {
            throw new AssertionFailedError("Url mapping assertion failed, '$controller' is not a valid controller")
        }
        controllerClass != null
    }

    /**
     * asserts a controller exists for the specified name and url
     *
     * @param controller The controller name
     * @param url The url
     */
    void assertController(String controller) {
        checkController(controller, true)
    }

    /**
     * @param controller The controller name
     * @param url The url
     * @return true If a controller exists for the specified name and url
     */
    boolean verifyController(String controller) {
        checkController(controller, false)
    }

    private boolean checkAction(String controller, String action, boolean throwEx) {
        GrailsControllerClass controllerClass = getControllerClass(controller)
        boolean valid = controllerClass?.mapsToURI("/$controller/$action")
        if (!valid && throwEx) {
            throw new AssertionFailedError("Url mapping assertion failed, '$action' is not a valid action of controller '$controller'")
        }
        valid
    }

    /**
     * Asserts an action exists for the specified controller name, action name and url
     *
     * @param controller The controller name
     * @param action The action name
     */
    void assertAction(String controller, String action) {
        checkAction(controller, action, true)
    }

    /**
     * @param controller The controller name
     * @param action The action name
     * @return true If an action exists for the specified controller name and action name
     */
    boolean verifyAction(String controller, String action) {
        checkAction(controller, action, false)
    }

    private boolean checkView(String controller, String view, boolean throwEx) {
        def pathPattern = ((controller) ? "$controller/" : '') + "${view}.gsp"
        if (!pathPattern.startsWith('/')) {
            pathPattern = "/$pathPattern"
        }
        GroovyPagesTemplateEngine templateEngine = applicationContext.getBean('groovyPagesTemplateEngine', GroovyPagesTemplateEngine)

        Template t = templateEngine.createTemplate(pathPattern)
        if (!t && throwEx) {
            throw new AssertionFailedError(
                    (controller) ? "Url mapping assertion failed, '$view' is not a valid view of controller '$controller'"
                            : "Url mapping assertion failed, '$view' is not a valid view")
        }
        t != null
    }

    /**
     * Asserts a view exists for the specified controller name and view name
     *
     * @param controller The controller name
     * @param view The view name
     */
    void assertView(String controller, String view) {
        checkView(controller, view, true)
    }

    /**
     *
     * @param controller The controller name
     * @param view The view name
     * @param url The url
     * @return true If a view exists for the specified controller and view
     */
    boolean verifyView(String controller, String view) {
        checkView(controller, view, false)
    }

    /**
     * Asserts a URL mapping maps to the specified controller, action, and optionally also parameters. Example:
     *
     * 
     * 
     *           assertUrlMapping("/action1", controller: "grailsUrlMappingsTestCaseFake", action: "action1") {
     *              param1 = "value1"
     *              param2 = "value2"
     *           }
     * 
     * 
* @param assertions The assertions as named parameters * @param url The URL as a string * @param paramAssertions The parameters to assert defined in the body of the closure */ void assertUrlMapping(Map assertions, String url, Closure paramAssertions = null) { assertForwardUrlMapping(assertions, url, paramAssertions) if (assertions.controller && !(url instanceof Integer)) { assertReverseUrlMapping(assertions, url, paramAssertions) } } /** * Verifies a URL mapping maps to the specified controller, action, and optionally also parameters. Example: * *
     * 
     *           verifyUrlMapping("/action1", controller: "grailsUrlMappingsTestCaseFake", action: "action1") {
     *              param1 = "value1"
     *              param2 = "value2"
     *           }
     * 
     * 
* @param assertions The assertions as named parameters * @param url The URL as a string * @param paramAssertions The parameters to assert defined in the body of the closure * * @return True if the url matches the assertions */ boolean verifyUrlMapping(Map assertions, String url, Closure paramAssertions = null) { boolean returnValue = verifyForwardUrlMapping(assertions, url, paramAssertions) if (assertions.controller && !(url instanceof Integer)) { returnValue = returnValue && verifyReverseUrlMapping(assertions, url, paramAssertions) } returnValue } private boolean checkForwardUrlMapping(Map assertions, Object url, Closure paramAssertions, boolean throwEx) { UrlMappingsHolder mappingsHolder = getUrlMappingsHolder() if (assertions.action && !assertions.controller) { throw new IllegalArgumentException('Cannot assert action for url mapping without asserting controller') } if (assertions.controller) { if (!checkController((String) assertions.controller, throwEx)) { return false } } if (assertions.action) { if (!checkAction((String) assertions.controller, (String) assertions.action, throwEx)) { return false } } if (assertions.view) { if (!checkView((String) assertions.controller, (String) assertions.view, throwEx)) { return false } } List mappingInfos if (url instanceof Integer) { mappingInfos = [] def mapping if (assertions."$KEY_EXCEPTION") { mapping = mappingsHolder.matchStatusCode(url, assertions."$KEY_EXCEPTION" as Throwable) } else { mapping = mappingsHolder.matchStatusCode(url) } if (mapping) { mappingInfos << mapping } } else { mappingInfos = mappingsHolder.matchAll((String) url, request.method).toList() } if (mappingInfos.size() == 0) { if (throwEx) { throw new AssertionFailedError("url '$url' did not match any mappings") } else { return false } } boolean returnVal = true boolean mappingMatched = mappingInfos.any { UrlMappingInfo mapping -> mapping.configure(webRequest) for (key in assertionKeys) { if (assertions.containsKey(key)) { String expected = (String) assertions[key] String actual = mapping."${key}Name" switch (key) { case 'controller': if (actual && !getControllerClass(actual)) { return false } break case 'view': if (actual[0] == '/') { actual = actual.substring(1) } if (expected[0] == '/') { expected = expected.substring(1) } break case 'action': if (key == 'action' && actual == null) { GrailsControllerClass controllerClass = getControllerClass(assertions.controller) actual = controllerClass?.defaultAction } break } if (expected != actual) { if (throwEx) { throw new ComparisonFailure("Url mapping $key assertion for '$url' failed".toString(), expected, actual) } else { returnVal = false } } } } if (paramAssertions) { def params = [:] paramAssertions.delegate = params paramAssertions.resolveStrategy = Closure.DELEGATE_ONLY paramAssertions.call() params.each { name, value -> String actual = mapping.parameters[name] String expected = value if (expected != actual) { if (throwEx) { throw new ComparisonFailure("Url mapping $name assertion for '$url' failed".toString(), expected, actual) } else { returnVal = false } } } } true } if (!mappingMatched) { throw new IllegalArgumentException("url '$url' did not match any mappings") } returnVal } void assertForwardUrlMapping(Map assertions, Object url, Closure paramAssertions = null) { checkForwardUrlMapping(assertions, url, paramAssertions, true) } boolean verifyForwardUrlMapping(Map assertions, Object url, Closure paramAssertions = null) { checkForwardUrlMapping(assertions, url, paramAssertions, false) } private boolean checkReverseUrlMapping(Map assertions, String url, Closure paramAssertions, boolean throwEx) { UrlMappingsHolder mappingsHolder = applicationContext.getBean('grailsUrlMappingsHolder', UrlMappingsHolder) UrlConverter urlConverter = applicationContext.getBean(UrlConverter.BEAN_NAME, UrlConverter) def controller = assertions.controller def action = assertions.action def method = assertions.method def plugin = assertions.plugin def namespace = assertions.namespace String convertedControllerName = null, convertedActionName = null if (controller) { convertedControllerName = urlConverter.toUrlElement(controller) ?: controller } if (action) { convertedActionName = urlConverter.toUrlElement(action) ?: action } def params = [:] if (paramAssertions) { paramAssertions.delegate = params paramAssertions.resolveStrategy = Closure.DELEGATE_ONLY paramAssertions.call() } UrlCreator urlCreator = mappingsHolder.getReverseMapping(controller, action, namespace, plugin, method, params) if (urlCreator == null) { if (throwEx) { throw new AssertionFailedError( "could not create reverse mapping of '$url' for {controller = $controller, action = $action, params = $params}") } else { return false } } String createdUrl = urlCreator.createRelativeURL(convertedControllerName, convertedActionName, params, 'UTF-8') if (url != createdUrl) { if (throwEx) { throw new ComparisonFailure( "reverse mapping assertion for {controller = $controller, action = $action, params = $params}", url, createdUrl) } else { return false } } true } /** * Asserts the given controller and action produce the given reverse URL mapping * *
     * 
     *           assertReverseUrlMapping("/action1", controller: "grailsUrlMappingsTestCaseFake", action: "action1")
     * 
     * 
* @param assertions The assertions as named parameters * @param url The URL as a string * @param paramAssertions The parameters to assert defined in the body of the closure */ void assertReverseUrlMapping(Map assertions, String url, Closure paramAssertions = null) { checkReverseUrlMapping(assertions, url, paramAssertions, true) } /** * Asserts the given controller and action produce the given reverse URL mapping * *
     * 
     *           verifyReverseUrlMapping("/action1", controller: "grailsUrlMappingsTestCaseFake", action: "action1")
     * 
     * 
* @param assertions The assertions as named parameters * @param url The URL as a string * @param paramAssertions The parameters to assert defined in the body of the closure * * @return True if the url matches the assertions */ boolean verifyReverseUrlMapping(Map assertions, String url, Closure paramAssertions = null) { checkReverseUrlMapping(assertions, url, paramAssertions, false) } GrailsControllerClass getControllerClass(String controller) { (GrailsControllerClass) grailsApplication.getArtefactByLogicalPropertyName(ControllerArtefactHandler.TYPE, controller) } @CompileDynamic void mockArtefact(Class urlMappingsClass) { grailsApplication.addArtefact(UrlMappingsArtefactHandler.TYPE, urlMappingsClass) defineBeans { grailsUrlMappingsHolder(UrlMappingsHolderFactoryBean) { getDelegate().grailsApplication = grailsApplication } } } String getBeanName(Class urlMappingsClass) { null } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy