Struts2-tiles-plugin 2.3.28 — StrutsTilesListener генерирует исключение NullPointerException

Я обновляюсь со struts-2.3.24.1 до struts-2.3.28, и при этом мне пришлось обновить плитки с 2.1.2 до 2.2.2. Я развертываю на WebLogic Server 12.1.3. Мое приложение построено без ошибок с использованием maven 3, однако при развертывании оно генерирует следующие ошибки:

<User defined listener org.apache.struts2.tiles.StrutsTilesListener failed: java.lang.NullPointerException.java.lang.NullPointerException
    at java.io.File.<init>(Unknown Source)
    at org.apache.struts2.tiles.StrutsWildcardServletTilesApplicationContext.<init>(StrutsWildcardServletTilesApplicationContext.java:53)

В приложении ничего не изменилось с тех пор, как оно работало со struts 2.3.24.1. В месте, указанном в файлеtiles-defs.xml, есть файлы .jsp. Я пробовал много разных подходов, чтобы исправить это, но продолжаю получать ту же ошибку. Я также не видел, чтобы этот конкретный вопрос обсуждался раньше. Что еще мне нужно изменить с его конкретным обновлением? Что мне не хватает?

Спасибо,

Полная трассировка стека:

    <User defined listener org.apache.struts2.tiles.StrutsTilesListener failed: java.lang.NullPointerException.java.lang.NullPointerException
    at java.io.File.<init>(Unknown Source)
    at org.apache.struts2.tiles.StrutsWildcardServletTilesApplicationContext.<init>(StrutsWildcardServletTilesApplicationContext.java:53)
    at org.apache.struts2.tiles.StrutsTilesInitializer.createTilesApplicationContext(StrutsTilesInitializer.java:37)
    at org.apache.tiles.startup.AbstractTilesInitializer.initialize(AbstractTilesInitializer.java:68)
    at org.apache.tiles.web.startup.AbstractTilesListener.contextInitialized(AbstractTilesListener.java:62)
    at weblogic.servlet.internal.EventsManager$FireContextListenerAction.run(EventsManager.java:678)
    at weblogic.security.acl.internal.AuthenticatedSubject.doAs(AuthenticatedSubject.java:321)
    at weblogic.security.service.SecurityManager.runAs(SecurityManager.java:120)
    at weblogic.servlet.provider.WlsSubjectHandle.run(WlsSubjectHandle.java:57)
    at weblogic.servlet.internal.EventsManager.executeContextListener(EventsManager.java:243)
    at weblogic.servlet.internal.EventsManager.notifyContextCreatedEvent(EventsManager.java:200)
    at weblogic.servlet.internal.EventsManager.notifyContextCreatedEvent(EventsManager.java:185)
    at weblogic.servlet.internal.WebAppServletContext.preloadResources(WebAppServletContext.java:1838)
    at weblogic.servlet.internal.WebAppServletContext.start(WebAppServletContext.java:2876)
    at weblogic.servlet.internal.WebAppModule.startContexts(WebAppModule.java:1661)
    at weblogic.servlet.internal.WebAppModule.start(WebAppModule.java:823)
    at weblogic.application.internal.ExtensibleModuleWrapper$StartStateChange.next(ExtensibleModuleWrapper.java:360)
    at weblogic.application.internal.ExtensibleModuleWrapper$StartStateChange.next(ExtensibleModuleWrapper.java:356)
    at weblogic.application.utils.StateMachineDriver.nextState(StateMachineDriver.java:42)
    at weblogic.application.internal.ExtensibleModuleWrapper.start(ExtensibleModuleWrapper.java:138)
    at weblogic.application.internal.flow.ModuleListenerInvoker.start(ModuleListenerInvoker.java:124)
    at weblogic.application.internal.flow.ModuleStateDriver$3.next(ModuleStateDriver.java:216)
    at weblogic.application.internal.flow.ModuleStateDriver$3.next(ModuleStateDriver.java:211)
    at weblogic.application.utils.StateMachineDriver.nextState(StateMachineDriver.java:42)
    at weblogic.application.internal.flow.ModuleStateDriver.start(ModuleStateDriver.java:73)
    at weblogic.application.internal.flow.StartModulesFlow.activate(StartModulesFlow.java:24)
    at weblogic.application.internal.BaseDeployment$2.next(BaseDeployment.java:729)
    at weblogic.application.utils.StateMachineDriver.nextState(StateMachineDriver.java:42)
    at weblogic.application.internal.BaseDeployment.activate(BaseDeployment.java:258)
    at weblogic.application.internal.SingleModuleDeployment.activate(SingleModuleDeployment.java:48)
    at weblogic.application.internal.DeploymentStateChecker.activate(DeploymentStateChecker.java:165)
    at weblogic.deploy.internal.targetserver.AppContainerInvoker.activate(AppContainerInvoker.java:80)
    at weblogic.deploy.internal.targetserver.BasicDeployment.activate(BasicDeployment.java:226)
    at weblogic.deploy.internal.targetserver.BasicDeployment.activateFromServerLifecycle(BasicDeployment.java:418)
    at weblogic.management.deploy.internal.DeploymentAdapter$1.doActivate(DeploymentAdapter.java:51)
    at weblogic.management.deploy.internal.DeploymentAdapter.activate(DeploymentAdapter.java:200)
    at weblogic.management.deploy.internal.AppTransition$2.transitionApp(AppTransition.java:30)
    at weblogic.management.deploy.internal.ConfiguredDeployments.transitionApps(ConfiguredDeployments.java:240)
    at weblogic.management.deploy.internal.ConfiguredDeployments.activate(ConfiguredDeployments.java:169)
    at weblogic.management.deploy.internal.ConfiguredDeployments.deploy(ConfiguredDeployments.java:123)
    at weblogic.management.deploy.internal.DeploymentServerService.resume(DeploymentServerService.java:210)
    at weblogic.management.deploy.internal.DeploymentServerService.start(DeploymentServerService.java:118)
    at weblogic.server.AbstractServerService.postConstruct(AbstractServerService.java:78)
    at sun.reflect.GeneratedMethodAccessor7.invoke(Unknown Source)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
    at java.lang.reflect.Method.invoke(Unknown Source)
    at org.glassfish.hk2.utilities.reflection.ReflectionHelper.invoke(ReflectionHelper.java:1017)
    at org.jvnet.hk2.internal.ClazzCreator.postConstructMe(ClazzCreator.java:388)
    at org.jvnet.hk2.internal.ClazzCreator.create(ClazzCreator.java:430)
    at org.jvnet.hk2.internal.SystemDescriptor.create(SystemDescriptor.java:456)
    at org.glassfish.hk2.runlevel.internal.AsyncRunLevelContext.findOrCreate(AsyncRunLevelContext.java:225)
    at org.glassfish.hk2.runlevel.RunLevelContext.findOrCreate(RunLevelContext.java:82)
    at org.jvnet.hk2.internal.Utilities.createService(Utilities.java:2488)
    at org.jvnet.hk2.internal.ServiceHandleImpl.getService(ServiceHandleImpl.java:98)
    at org.jvnet.hk2.internal.ServiceLocatorImpl.getService(ServiceLocatorImpl.java:606)
    at org.jvnet.hk2.internal.ThreeThirtyResolver.resolve(ThreeThirtyResolver.java:77)
    at org.jvnet.hk2.internal.ClazzCreator.resolve(ClazzCreator.java:231)
    at org.jvnet.hk2.internal.ClazzCreator.resolveAllDependencies(ClazzCreator.java:254)
    at org.jvnet.hk2.internal.ClazzCreator.create(ClazzCreator.java:413)
    at org.jvnet.hk2.internal.SystemDescriptor.create(SystemDescriptor.java:456)
    at org.glassfish.hk2.runlevel.internal.AsyncRunLevelContext.findOrCreate(AsyncRunLevelContext.java:225)
    at org.glassfish.hk2.runlevel.RunLevelContext.findOrCreate(RunLevelContext.java:82)
    at org.jvnet.hk2.internal.Utilities.createService(Utilities.java:2488)
    at org.jvnet.hk2.internal.ServiceHandleImpl.getService(ServiceHandleImpl.java:98)
    at org.jvnet.hk2.internal.ServiceHandleImpl.getService(ServiceHandleImpl.java:87)
    at org.glassfish.hk2.runlevel.internal.CurrentTaskFuture$QueueRunner.oneJob(CurrentTaskFuture.java:1162)
    at org.glassfish.hk2.runlevel.internal.CurrentTaskFuture$QueueRunner.run(CurrentTaskFuture.java:1147)
    at org.glassfish.hk2.runlevel.internal.CurrentTaskFuture$UpOneLevel.run(CurrentTaskFuture.java:753)
    at weblogic.work.SelfTuningWorkManagerImpl$WorkAdapterImpl.run(SelfTuningWorkManagerImpl.java:553)
    at weblogic.work.ExecuteThread.execute(ExecuteThread.java:311)
    at weblogic.work.ExecuteThread.run(ExecuteThread.java:263)

выдержка pom.xml

    <properties>
    <struts2-version>2.3.28</struts2-version>
    <tiles-version>2.2.2</tiles-version>
    </properties>
    <dependency>
        <groupId>org.apache.struts</groupId>
        <artifactId>struts2-core</artifactId>
        <version>${struts2-version}</version>
        <exclusions>
            <exclusion>
                <groupId>javassist</groupId>
                <artifactId>javassist</artifactId>
            </exclusion>
        </exclusions>
    </dependency>

    <dependency>
        <groupId>org.apache.struts</groupId>
        <artifactId>struts2-convention-plugin</artifactId>
        <version>${struts2-version}</version>
    </dependency>

    <dependency>
        <groupId>org.apache.struts</groupId>
        <artifactId>struts2-junit-plugin</artifactId>
        <version>${struts2-version}</version>
        <scope>test</scope>
        <exclusions>
            <exclusion>
                <groupId>junit</groupId>
                <artifactId>junit</artifactId>
            </exclusion>
        </exclusions>
    </dependency>

    <dependency>
        <groupId>org.apache.struts</groupId>
        <artifactId>struts2-rest-plugin</artifactId>
        <version>${struts2-version}</version>
    </dependency>

    <dependency>
        <groupId>org.apache.struts</groupId>
        <artifactId>struts2-spring-plugin</artifactId>
        <version>${struts2-version}</version>
    </dependency>

    <dependency>
        <groupId>org.apache.struts</groupId>
        <artifactId>struts2-tiles-plugin</artifactId>
        <version>${struts2-version}</version>
    </dependency>

    <dependency>
        <groupId>org.apache.tiles</groupId>
        <artifactId>tiles-jsp</artifactId>
        <version>${tiles-version}</version>
    </dependency>

    <dependency>
        <groupId>org.apache.tiles</groupId>
        <artifactId>tiles-core</artifactId>
        <version>${tiles-version}</version>
        <scope>compile</scope>
    </dependency>

    <dependency>
        <groupId>org.apache.tiles</groupId>
        <artifactId>tiles-el</artifactId>
        <version>${tiles-version}</version>
        <scope>compile</scope>
    </dependency>

    <dependency>
        <groupId>org.apache.tiles</groupId>
        <artifactId>tiles-freemarker</artifactId>
        <version>${tiles-version}</version>
        <scope>compile</scope>
    </dependency>

    <dependency>
        <groupId>org.apache.tiles</groupId>
        <artifactId>tiles-ognl</artifactId>
        <version>${tiles-version}</version>
        <scope>compile</scope>
    </dependency>

    <dependency>
        <groupId>org.apache.tiles</groupId>
        <artifactId>tiles-api</artifactId>
        <version>${tiles-version}</version>
        <scope>compile</scope>
    </dependency>

    <dependency>
        <groupId>org.apache.tiles</groupId>
        <artifactId>tiles-servlet</artifactId>
        <version>${tiles-version}</version>
    </dependency>   

Struts.xml

<!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN" "http://struts.apache.org/dtds/struts-2.0.dtd"> 
<struts>

<constant name="struts.enable.DynamicMethodInvocation" value="TRUE" />

<bean type="org.apache.struts2.dispatcher.mapper.ActionMapper" class="path.mapper.GfpDynamicMapper" name="gfpDynamicMapper"/>
<constant name="struts.mapper.class" value="gfpDynamicMapper" /> 

<constant name="struts.convention.action.includeJars" value=".*_wl_cls_gen.*"/>
<constant name="struts.convention.action.fileProtocols" value="jar,zip" />


<constant name="struts.convention.action.suffix" value="Controller" />
<constant name="struts.convention.action.mapAllMatches" value="true" />
<constant name="struts.convention.default.parent.package" value="default" />
<constant name="struts.custom.i18n.resources" value="path/application-gfp" />
<constant name="struts.objectFactory" value="spring" />
<constant name="struts.convention.result.path" value="tiles/gfp"/>


<!-- Handler setup -->
<bean name="gfpXml"  type="org.apache.struts2.rest.handler.ContentTypeHandler" class="path.handler.GfpXmlHandler" />
<bean type="com.opensymphony.xwork2.UnknownHandler" name="handler" class="path.handler.UnknownHandler"/>
<package name="default" namespace="/actions" extends="rest-default">
    <result-types>
        <result-type name="tiles" class="org.apache.struts2.views.tiles.TilesResult" />
    </result-types>

    <interceptors>

        <interceptor name="userInterceptor"  class="path.interceptors.UserInterceptor" />

        <interceptor name="exceptionMapper" class="path.interceptors.ExceptionMappingInterceptor">
            <param name="logEnabled">true</param>
        </interceptor>

        <interceptor name="cachingHeadersInterceptor" class="path.interceptors.CachingHeadersInterceptor"/>
        <interceptor-stack name="interceptorStack">
            <interceptor-ref name="exceptionMapper"/>
            <interceptor-ref name="tokenSession" >
                <param name="excludeMethods">create,edit,index,show,compare,ulns,showInvalidUlns,toggleExternalUics,status,exportSummary,summary,exportSlice,exportSlices,export,exportUlns,exportPids,exportPcode,exportArt,exportAst,exportDrrs,exportOrgAssoc,exportBillet,exportEquipment,exportLeaderAssoc,exportGfmts,exportSorts,exportUSMCSliderPull,exportWebSked,exportExcel,exportSliceCompare,exportSliceCompareCritieria,exportCargo,exportPersonnel,showSliceSummary</param>
            </interceptor-ref>
            <interceptor-ref name="alias"/>
            <interceptor-ref name="userInterceptor"/>
            <interceptor-ref name="servletConfig"/>
            <interceptor-ref name="messages">
                <param name="operationMode">AUTOMATIC</param>
            </interceptor-ref>
            <interceptor-ref name="prepare"/>
            <interceptor-ref name="i18n"/>
            <interceptor-ref name="chain"/>
            <interceptor-ref name="debugging"/>
            <interceptor-ref name="profiling"/>
            <interceptor-ref name="scopedModelDriven"/>
            <interceptor-ref name="modelDriven">
                <param name="refreshModelBeforeResult">true</param>
            </interceptor-ref>
            <interceptor-ref name="fileUpload"/>
            <interceptor-ref name="checkbox"/>
            <interceptor-ref name="multiselect"/>
            <interceptor-ref name="staticParams"/>
            <interceptor-ref name="actionMappingParams"/>
            <interceptor-ref name="params">
                <param name="excludeParams">dojo\..*,statusCode,token,struts.token.name,import,submit,label.next,label.back,_method</param>
            </interceptor-ref>
            <interceptor-ref name="rest" />
            <interceptor-ref name="conversionError"/>
            <interceptor-ref name="validation">
                <param name="excludeMethods">input,back,cancel,browse,index,show,edit,editNew</param>
            </interceptor-ref>
            <interceptor-ref name="restWorkflow">
                <param name="excludeMethods">input,back,cancel,browse,index,show,edit,editNew</param>
            </interceptor-ref>
            <interceptor-ref name="annotationWorkflow"/>
            <interceptor-ref name="cachingHeadersInterceptor"/>
        </interceptor-stack>
      </interceptors>
</package>
</struts> 

веб.xml

<web-app id="WebApp_9" version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee"    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"   xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
<context-param>
    <param-name>org.apache.tiles.impl.BasicTilesContainer.DEFINITIONS_CONFIG</param-name>
    <param-value>/WEB-INF/classes/path/tiles-defs.xml</param-value>
</context-param>    
<listener>
    <listener-class>org.apache.struts2.tiles.StrutsTilesListener</listener-class>
</listener>

tiles-defs.xml

<?xml version="1.0" encoding="ISO-8859-1" ?>
<!DOCTYPE tiles-definitions PUBLIC"-//Apache Software Foundation//DTD Tiles Configuration 2.0//EN" "http://tiles.apache.org/dtds/tiles-config_2_0.dtd">
<tiles-definitions>
    <definition name="layout.default" template="/WEB-INF/tiles/common/default.jsp">
        <put-attribute name="infoKey" value="" />
        <put-attribute name="titleKey" value="" />
        <put-attribute name="browserTitleKey" value="" />
        <put-attribute name="header" value="/WEB-INF/tiles/common/header.jsp" />
        <put-attribute name="content" value="" />
        <put-attribute name="menubar" value="" />
        <put-attribute name="adminMenu" value="" />
        <put-attribute name="footer" value="/WEB-INF/tiles/common/footer.jsp" />
</definition>
<definition name="layout.update" template="/WEB-INF/tiles/common/updateDefault.jsp">
    <put-attribute name="infoKey" value="" />
    <put-attribute name="titleKey" value="" />
    <put-attribute name="header" value="/WEB-INF/tiles/common/classmarkTop.jsp"/>
    <put-attribute name="content" value="" />
    <put-attribute name="menubar" value="" />
    <put-attribute name="footer" value="/WEB-INF/tiles/common/footer.jsp" />
</definition>
<definition name="layout.admin" extends="layout.default">
    <put-attribute name="adminMenu" value="/WEB-INF/tiles/gfp/admin/adminMenu.jsp" />
    </definition>
</tiles-definitions>

Изменить: это не связано с вопросом о том, что такое исключение нулевого указателя, на которое ссылаются в комментариях, потому что это проблема с конфигурациями стоек и плиток.

Редактировать 31.03.2016: я включил удаленную отладку и обнаружил, что контекст имеет все соответствующие пути (см. изображение ниже). Однако в строке 53 StrutsWildcardServletTilesApplicationContext.java context.getRealPath() возвращает значение null, поскольку невозможно получить абсолютный путь к этому ресурсу. Контекстный путь также корректен в объекте контекста. Я читал, что функция getRealPath() может возвращать значение null, если она не может преобразовать виртуальный путь в реальный путь, что происходит, когда файлы поступают из .war. Как это исправить?

Изображение удаленной отладки


person user255377    schedule 29.03.2016    source источник
comment
Попробуйте удалить <context-param>, если это не поможет, зарегистрируйте ошибку в JIRA issues.apache.org/ jira/browse/WW   -  person Lukasz Lenart    schedule 30.03.2016
comment
Обновите dtd-s, удалите context-param, убедитесь, что на сервере нет старых версий. Отладьте свое приложение, чтобы узнать, что является нулевым.   -  person Aleksandr M    schedule 30.03.2016
comment
Возможный дубликат Что такое нулевой указатель Исключение и как его исправить?   -  person beresfordt    schedule 30.03.2016
comment
@beresfordt Не обман. NPE исходит из плагина фреймворка. Может быть неверная конфигурация.   -  person Aleksandr M    schedule 30.03.2016
comment
@AleksandrM Каковы правильные DTDS для этих версий стоек и плиток? Я читал, что где-то это может быть проблемой, но не видел в документации по выпуску ничего, указывающего на наличие зависимости. Нужно ли обновлять dtd tile-defs.xml и struts.xml до одной и той же версии?   -  person user255377    schedule 30.03.2016
comment
Версия, которую вы используете и для которой есть dtd. Плитки — tiles.apache.org/dtds, Struts2 — struts.apache.org/dtds.   -  person Aleksandr M    schedule 30.03.2016
comment
Я обновил плитки dtd до 2.1 и стойки до 2.3, а также удалил ‹context-param›, указывающий на файлtiles-defs.xml. Это не сработало, и я все еще получаю ту же ошибку. Я заметил, что org.apache.struts2.tiles.StrutsTilesListener отличается в 2.3.28 от 2.3.24.1. Кому-нибудь еще удалось это обновление?   -  person user255377    schedule 30.03.2016
comment
Да, но кажется, что вы переопределили некоторые вещи, например. ActionMapper. Можете ли вы отладить свое приложение, чтобы узнать, что именно равно нулю.   -  person Aleksandr M    schedule 31.03.2016
comment
@AleksandrM Я запустил приложение с включенной отладкой и обнаружил, что context.getRealPath() не может преобразовать виртуальный путь в абсолютный путь (см. Редактирование от 31 марта 2016 г. выше). Переменная «путь» верна, как и контекстный путь. Не уверен, почему getRealPath() возвращает значение null. Любая помощь приветствуется. Спасибо!   -  person user255377    schedule 31.03.2016


Ответы (1)


Только что решил проблему с NPE, которая у меня была. Как я уже говорил выше, проблема заключалась в том, что getRealPath не мог преобразовать виртуальный путь в абсолютный путь. Воспользовался советом из этой темы.

Почему getRealPath() возвращает значение null при развертывании с файл .war?

и добавил конфигурацию

<show-archived-real-path-enabled>true</show-archived-real-path-enabled>

в мой weblogic.xml.

person user255377    schedule 31.03.2016