EnumでPropertyファイル読み込みクラスを作る

WebシステムでPropertyファイルを起動時に読み込んでメモリ上に展開することはよくあることだと思います。

で、それをEnumでやったらこうなる!というのを作ってみました。

PropertyHolder.java

package property;

import java.util.Properties;
import java.util.Set;

public enum PropertyHolder {

    Hoge("hoge.properties"), Foo("foo.properties"), Bar("bar.properties");

    private Properties properties;
    private String fileName;

    static {
        System.out.println("staticイニシャライザ開始");
        for (PropertyHolder p : values()) {
            System.out.println("static:" + p.name());
            p.init();
        }
    }

    private PropertyHolder(String fileName) {
        System.out.println("constructor:" + fileName);
        this.fileName = fileName;
    }

    private void init() {
        System.out.println("init:" + fileName);
        properties = new Properties();
        try {
            properties.load(getClass().getClassLoader().getResourceAsStream(
                    fileName));
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    public String get(String key) {
        return properties.getProperty(key);
    }

    public String getAll() {
        System.out.println("getAll:" + fileName);
        Set<Object> keys = properties.keySet();
        StringBuilder sb = new StringBuilder();
        for (Object obj : keys) {
            String key = (String) obj;
            sb.append("[");
            sb.append(key);
            sb.append(":");
            sb.append(get((String) obj));
            sb.append("]");
        }
        return sb.toString();
    }
}


Main.java
package property;

public class Main {

    public static void main(String[] args) {
        System.out.println("Hoge.get(\"a\"):" + PropertyHolder.Hoge.get("a"));
        System.out.println("Hoge.getAll():" + PropertyHolder.Hoge.getAll());
        System.out.println("Foo.getAll():" + PropertyHolder.Foo.getAll());
        System.out.println("Bar.getAll():" + PropertyHolder.Bar.getAll());
    }
}
hoge.properties
a=A b=B c=C
foo.properties
aa=AA bb=BB cc=CC
bar.properties
aaa=AAA bbb=BBB ccc=CCC
実行結果
constructor:hoge.properties constructor:foo.properties constructor:bar.properties staticイニシャライザ開始 static:Hoge init:hoge.properties static:Foo init:foo.properties static:Bar init:bar.properties Hoge.get("a"):A getAll:hoge.properties Hoge.getAll():[b:B][a:A][c:C] getAll:foo.properties Foo.getAll():[aa:AA][bb:BB][cc:CC] getAll:bar.properties Bar.getAll():[bbb:BBB][aaa:AAA][ccc:CCC]
ポイントは、staticイニシャライザでEnumクラスの定数一つ一つがもっているinitメソッドを呼んでいるところですかね。

staticイニシャライザは1クラスに対して1回しか動かないので、Enumクラスの中に定数がいくつあっても、その定数ひとつに対して1回しかstaticイニシャライザが呼ばれないことは実行結果をみるとわかりますね。

staticイニシャライザを使うと例外がRuntimeExceptionしか出せないのがきついですね。 システム共通ランタイムExceptionクラスがある場合、そちらを使ってラップしてからスローするといいですね。

あとは、いちいちinitメソッドを作らずにコンストラクタでファイルを読んでもいいと思います。 場合によると思うので、使い分けてみてください。

※サンプルのgetAllメソッドは実際に使う時は不要だと思うので、削除して使ってください。

  • Digg
  • Del.icio.us
  • StumbleUpon
  • Reddit
  • Twitter
  • RSS

EnumでDate変換Utilクラスを作る

いろいろな使い方があるEnumですが、Utilクラスを作るのにもつかえます!
ためしに日付を変換するUtilを作ってみました。

日付<>文字列を変換するときは、いろいろなフォーマットの日付を表す文字列に対応しなければならないので、フォーマットの定義(yyyyMMddみたいなの)をたくさん定義して、定義した分だけ同じよう名メソッドを作らなければなりません。。

ですが、Enumを使えばすっきりかけます~。

以下コードです。

DateUtil.java

package util;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

// TODO: 自動生成された Javadoc
/**
* The Enum DateUtil.
*/
public enum DateUtil {

// ------------------------------
// 変換したいフォーマットを追加(start)
// ------------------------------
/** The YYYYMMDD. */
YYYYMMDD("yyyy/MM/dd"),
/** The YYYYMMDDHHMM. */
YYYYMMDDHHMM("yyyy/MM/dd hh:mm"),
/** The YYYYMMDDHHMMSS. */
YYYYMMDDHHMMSS("yyyy/MM/dd HH:mm.ss"),
/** The YYYYMMDDHHMMSSSSS. */
YYYYMMDDHHMMSSSSS("yyyy/MM/dd HH:mm.ss.SSS"),
/** The Jp_ yyyymmdd. */
JP_YYYYMMDD("yyyy年MM月dd日"),
/** The Jp_ yyyymmddhhmm. */
JP_YYYYMMDDHHMM("yyyy年MM月dd日HH時mm分"),
/** The Jp_ yyyymmddhhmmss. */
JP_YYYYMMDDHHMMSS("yyyy年MM月dd日HH時mm分ss秒");
// ------------------------------
// 変換したいフォーマットを追加(end)
// ------------------------------
/** The format. */
private String format;

/**
* Instantiates a new date util.
*
* @param format
* the format
*/
private DateUtil(String format) {
this.format = format;
}

/**
* To date.
*
* @param dateString
* the date string
* @return the date
*/
public Date toDate(String dateString) {
try {
return new SimpleDateFormat(format).parse(dateString);
} catch (ParseException e) {
// RuntimeExceptionでラップしてしまっていますが、システム共通Exceptionを投げるようにする方がいいと思います。
// 場合によって使い分けてください。
throw new RuntimeException("文字列→Dateへの変換に失敗しました(format:" + format
+ ",string:" + dateString + ")", e);
}
}

/**
* To string.
*
* @param date
* the date
* @return the string
*/
public String toString(Date date) {
return new SimpleDateFormat(format).format(date);
}

}

Main.java


package util;

import java.util.Date;

public class Main {

public static void main(String[] args) {
System.out.println("YYYYMMDD.toDate:"
+ DateUtil.YYYYMMDD.toDate("2012/1/29"));
System.out.println("YYYYMMDDHHMM.toDate:"
+ DateUtil.YYYYMMDDHHMM.toDate("2012/1/29 1:5"));
System.out.println("YYYYMMDDHHMMSS.toDate:"
+ DateUtil.YYYYMMDDHHMMSS.toDate("2012/1/29 1:5.7"));
System.out.println("YYYYMMDDHHMMSSSSS.toDate:"
+ DateUtil.YYYYMMDDHHMMSSSSS.toDate("2012/1/29 1:5.7.100"));
System.out.println("JP_YYYYMMDD.toDate:"
+ DateUtil.JP_YYYYMMDD.toDate("2012年1月29日"));
System.out.println("JP_YYYYMMDDHHMM.toDate:"
+ DateUtil.JP_YYYYMMDDHHMM.toDate("2012年1月29日1時5分"));
System.out.println("JP_YYYYMMDDHHMMSS.toDate:"
+ DateUtil.JP_YYYYMMDDHHMMSS.toDate("2012年1月29日1時5分7秒"));

System.out.println("------------------------------");
System.out.println(DateUtil.YYYYMMDD.toString(new Date()));
System.out.println(DateUtil.YYYYMMDDHHMM.toString(new Date()));
System.out.println(DateUtil.YYYYMMDDHHMMSS.toString(new Date()));
System.out.println(DateUtil.YYYYMMDDHHMMSSSSS.toString(new Date()));
System.out.println(DateUtil.JP_YYYYMMDD.toString(new Date()));
System.out.println(DateUtil.JP_YYYYMMDDHHMM.toString(new Date()));
System.out.println(DateUtil.JP_YYYYMMDDHHMMSS.toString(new Date()));

}
}



実行結果

YYYYMMDD.toDate:Sun Jan 29 00:00:00 JST 2012
YYYYMMDDHHMM.toDate:Sun Jan 29 01:05:00 JST 2012
YYYYMMDDHHMMSS.toDate:Sun Jan 29 01:05:07 JST 2012
YYYYMMDDHHMMSSSSS.toDate:Sun Jan 29 01:05:07 JST 2012
JP_YYYYMMDD.toDate:Sun Jan 29 00:00:00 JST 2012
JP_YYYYMMDDHHMM.toDate:Sun Jan 29 01:05:00 JST 2012
JP_YYYYMMDDHHMMSS.toDate:Sun Jan 29 01:05:07 JST 2012
------------------------------
2012/01/29
2012/01/29 06:27
2012/01/29 18:27.51
2012/01/29 18:27.51.073
2012年01月29日
2012年01月29日18時27分
2012年01月29日18時27分51秒


まぁこんな感じです。
DateUtilクラスでは、フォーマットを定義をして、Date<>Stringのメソッドを共通的に実装してます!
フォーマットごとにDate<>String変換メソッドを実装しなくてすむので、楽です~


※サンプルでは、toDateメソッドにて、ParseExceptionをRuntimeExceptionにラップしてしまっているので、使うときは、システムで使用しているExceptionにラップして使うようにしてください!

  • Digg
  • Del.icio.us
  • StumbleUpon
  • Reddit
  • Twitter
  • RSS

tilesが動かない!springを使う場合は、tiles用のRequestプロセッサを指定してね。

以下の環境で、strutsのtilesをしようしようとしたら503エラーが。。

struts 1.3.10
spring 2.5

なんでかって。。それは、struts-configで宣言しているcontrollerタグで指定するクラスが間違ってたからでした。

↓の用に設定していましたが、DelegatingRequestProcessorではなDelegatingTilesRequestProcessorでした。。

ダメ

<controller>
<set-property property="processorClass"
value="org.springframework.web.struts.DelegatingRequestProcessor"/>
</controller>


正解

<controller>
<set-property property="processorClass"
value="org.springframework.web.struts.DelegatingTilesRequestProcessor"/>
</controller>

これに数時間はまってましたorz..

  • Digg
  • Del.icio.us
  • StumbleUpon
  • Reddit
  • Twitter
  • RSS

JSPの上のほうに書く@page contentTypeやカスタムタグの宣言を省略する

きっかけ
JSPを作成する際に上のほうに↓のような宣言をよくかきますよね。。



<%@ page contentType="text/html;charset=UTF-8"%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>


これをどのすべてのJSPファイルに書くのは面倒だし重複コードになる気がしたので、どうにか省略できる方法はないかと探したら見つけたので、メモしておきます。

<%@ page contentType="text/html;charset=UTF-8"%>の省略
これは、以下のページを見るとわかります。

http://docs.oracle.com/cd/E18355_01/web.1013/B31860-01/jspnls.htm

pageディレクティブでのコンテンツ・タイプの設定

あたり


呼んでいくと
contentTypeエンコーディングを指定しない
かつ
pageEncodingを指定
した場合、contentTypeエンコーディングは、pageEncodingを指定 に従うとあるので、pageEncodingをweb.xmlのjsp-configにて、設定してあげれば、省略することが出来ます。

例えば、①の@pageタグを省略したい場合は、jsp-configには②のように書きます。


<%@ page contentType="text/html;charset=UTF-8"%>



<jsp-property-group>
<url-pattern>*.jsp</url-pattern>
<page-encoding>UTF-8</page-encoding>
</jsp-property-group>
</jsp-config>

これで、あえて、@pageタグを記載しなければ、すべてのJSPページのcontextTypeは
"text/html;charset=UTF-8"
となります。(JSPファイルの場合、text/html;はデフォルトで付与されるそうです)

カスタムタグの宣言を省略
以下のページに書いてありました。
http://www.atmarkit.co.jp/fjava/javatips/119jspservlet39.html

jsp-configの
include-preludタグを使えばすべてのJSPの上部に任意のJSPを設定できるようです。

で、include-preludタグにカスタムタグの宣言だけ書いてあるJSPを指定すれば、すべてのJSPの上部にカスタムタグの宣言が挿入されるってことです!

ecilpseでjspを作成する際に、このタグを設定してから保管機能(Ctrl+SPとか)を使うと宣言したカスタムタグが保管されるので、便利です。


これでJSPの上のほうがすっきり!

  • Digg
  • Del.icio.us
  • StumbleUpon
  • Reddit
  • Twitter
  • RSS

tiles:getAsString で取得した値をbean:message のkey属性に値を設定する

tiles.xmlでputタグに設定したvalueの値を、JSP側のbean:messageタグのkey属性に設定しようとしたらはまったので、メモします。

前提

tiles.xml
 ↓の用に定義しています。

<definition name="baseLayout" path="/WEB-INF/jsp/shared/baseLayout.jsp">
<put name="screenId" value="A001"/>
</definition>

bean:messageで取得するリソースファイル
リソースファイルの内容を↓の用に設定して、struts-configにリソースファイルとして定義しておきます。

    A001=A画面

baseLayout.jspでは、tiles.xmlで指定したscreenIdを取得し、リソースファイルに設定した"A画面"という文字をbean:messageタグを使って取得しようとしたとします。

問題

↓の様に書いてもだめでした。。。まぁ当然ですよね。

<bean:message key="<tiles:getAsString name="screenId"/>" />

対策 
bean:defineを使って一旦変数にとり、その後EL式を使って取得するようにしました。

<bean:define id="screenId"><tiles:getAsString name="screenId"/></bean:define>
<bean:message key="${screenId}" />
これで何とかリソースファイルから値を取得することが出来ました。

欠点 

この方法だとtiles.xmlでscreenIdがputタグで定義されていない場合、bean:defineにnullを設定しようとするため、エラーとなってしまいます。。。orz..

欠点を克服できる方法がありましたら、教えていただけるとありがたいです。

  • Digg
  • Del.icio.us
  • StumbleUpon
  • Reddit
  • Twitter
  • RSS

antのreplaceregrexタスクを使って複数行置換!

前回に引き続きantでreplace文字列のreplaceです~
今回は、ある複数行からある複数行に置換する方法です。

例えばあるファイル(test.xml)内にある①のような複数行の文字列を②に置換したいとします。


<hoge name="a">
aaaa
</hoge>


<piyo name="b">
bbbb
</piyo>

その場合以下のようにreplaceregrexタスクで記載します。

その場合、replace

<replaceregexp file="./test.xml" encoding="UTF-8" flags="gs" replace="&lt;piyo name=&quot;b&quot;&gt;&#10;bbbb&#10;&lt;/piyo&gt;">
<regexp pattern="&lt;hoge name=&quot;a&quot;&gt;.*?aaaa.*?/hoge&gt;" />
</replaceregexp>

のようになります。

ポイントは、flagsに設定している"gs"と文字のエスケープです。


【flagsに設定している"gs"】

g:繰り返し
regexpタスクで指定したpatternに指定した1度該当しても置換を継続して行う。
s:シングルラインモード
通常"."は\nを除く任意の一文字を示すが、シングルラインモードでは、
"."は\nを含む一文字を示します。
シングルラインモードに設定することで、複数行の内容のファイルを1行として扱うことが出来るので、複数行にまたがる正規表現が比較的簡単にかけます。
【文字をエスケープ】
replaceregexpタスクのreplace属性に置換後の複数行の内容を記載するときや
regexpタスクのpattern属性を記載するときは、
antはxml形式で記載するので、xmlの記載方法にのっとって記載しなければなりません。
xmlでは、改行コード、ダブルクオート、<,>をエスケープしないとうまく文字を認識してくれないので、以下のようにエスケープします。

< &lt;
> &gt;
LF &#10;
CR &#13;

例では、改行をLFのみで示しています。
インデントの部分を広報参照で書ければ汎用性が高くなおよいと思いますが、
例では、おこなっていません。

  • Digg
  • Del.icio.us
  • StumbleUpon
  • Reddit
  • Twitter
  • RSS

Antのrepraceで“(ダブルクォーテーション)を指定する方法

久々にはまりましたので、メモします。

antのreplaceタスクで置換対象文字列の抽出を正規表現で書いていたときのことです。

置換対象文字列に"(ダブルクォーテーション)が入っていたので、
↓のように「"」を「\"」としたのですが、eclipseに怒られてしまいました。。

<replaceregexp encoding="UTF-8" flags="g" replace="\[\0\]">
    <regexp pattern="\".*\""></regexp>
</replaceregexp>


で、いろいろ悩んだ結果「"」はアスキーコードで0x22なので、↓の用に書いたらいけました。

<replaceregexp encoding="UTF-8" flags="g" replace="\[\0\]">
    <regexp pattern="\x22.*\x22"></regexp>
</replaceregexp>



ちなみにreplaceregexpタグのreplace属性側に「\x22」を指定したところ「x22」と認識されてしまいました。
replaceregexpタグのreplace属性に設定する場合は、もう一工夫いるようでしたが、わかりませんでしたorz...

誰か知っている方、いらっしゃいましたら、コメントください。

  • Digg
  • Del.icio.us
  • StumbleUpon
  • Reddit
  • Twitter
  • RSS

gaeでローカル起動したらJDOUserException

やっと新機能を実装してローカル環境でgaeを起動したら↓のエラーが。。。

javax.jdo.JDOUserException: Persistent class "Class org.pupit.model.PupitThread does not seem to have been enhanced.  You may want to rerun the enhancer and check for errors in the output." has no table in the database, but the operation requires it. Please check the specification of the MetaData for this class.
 at org.datanucleus.jdo.NucleusJDOHelper.getJDOExceptionForNucleusException(NucleusJDOHelper.java:375)
 at org.datanucleus.jdo.JDOQuery.execute(JDOQuery.java:252)
 at org.pupit.model.PupitThreadDao.selectByLastUpdateTime(PupitThreadDao.java:37)
 at org.pupit.service.thread.GetPupitThreadListServiceImpl.execute(GetPupitThreadListServiceImpl.java:13)
 at org.pupit.servlet.thread.IndexServlet.getThreadList(IndexServlet.java:54)
 at org.pupit.servlet.thread.IndexServlet.doGet(IndexServlet.java:25)
 at javax.servlet.http.HttpServlet.service(HttpServlet.java:617)
 at javax.servlet.http.HttpServlet.service(HttpServlet.java:717)
 at org.mortbay.jetty.servlet.ServletHolder.handle(ServletHolder.java:511)
 at org.mortbay.jetty.servlet.ServletHandler.handle(ServletHandler.java:390)
 at org.mortbay.jetty.security.SecurityHandler.handle(SecurityHandler.java:216)
 at org.mortbay.jetty.servlet.SessionHandler.handle(SessionHandler.java:182)
 at org.mortbay.jetty.handler.ContextHandler.handle(ContextHandler.java:765)
 at org.mortbay.jetty.webapp.WebAppContext.handle(WebAppContext.java:418)
 at org.mortbay.jetty.servlet.Dispatcher.forward(Dispatcher.java:327)
 at org.mortbay.jetty.servlet.Dispatcher.forward(Dispatcher.java:126)
 at com.google.apphosting.runtime.jetty.ResourceFileServlet.serveWelcomeFileAsForward(ResourceFileServlet.java:342)
 at com.google.apphosting.runtime.jetty.ResourceFileServlet.maybeServeWelcomeFile(ResourceFileServlet.java:281)
 at com.google.apphosting.runtime.jetty.ResourceFileServlet.doGet(ResourceFileServlet.java:142)
 at javax.servlet.http.HttpServlet.service(HttpServlet.java:617)
 at javax.servlet.http.HttpServlet.service(HttpServlet.java:717)
 at org.mortbay.jetty.servlet.ServletHolder.handle(ServletHolder.java:511)
 at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1166)
 at com.google.inject.servlet.FilterChainInvocation.doFilter(FilterChainInvocation.java:67)
 at com.google.inject.servlet.ManagedFilterPipeline.dispatch(ManagedFilterPipeline.java:122)
 at com.google.inject.servlet.GuiceFilter.doFilter(GuiceFilter.java:110)
 at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157)
 at com.google.apphosting.utils.servlet.ParseBlobUploadFilter.doFilter(ParseBlobUploadFilter.java:102)
 at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157)
 at com.google.apphosting.runtime.jetty.SaveSessionFilter.doFilter(SaveSessionFilter.java:35)
 at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157)
 at com.google.apphosting.utils.servlet.TransactionCleanupFilter.doFilter(TransactionCleanupFilter.java:43)
 at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157)
 at org.mortbay.jetty.servlet.ServletHandler.handle(ServletHandler.java:388)
 at org.mortbay.jetty.security.SecurityHandler.handle(SecurityHandler.java:216)
 at org.mortbay.jetty.servlet.SessionHandler.handle(SessionHandler.java:182)
 at org.mortbay.jetty.handler.ContextHandler.handle(ContextHandler.java:765)
 at org.mortbay.jetty.webapp.WebAppContext.handle(WebAppContext.java:418)
 at com.google.apphosting.runtime.jetty.AppVersionHandlerMap.handle(AppVersionHandlerMap.java:249)
 at org.mortbay.jetty.handler.HandlerWrapper.handle(HandlerWrapper.java:152)
 at org.mortbay.jetty.Server.handle(Server.java:326)
 at org.mortbay.jetty.HttpConnection.handleRequest(HttpConnection.java:542)
 at org.mortbay.jetty.HttpConnection$RequestHandler.headerComplete(HttpConnection.java:923)
 at com.google.apphosting.runtime.jetty.RpcRequestParser.parseAvailable(RpcRequestParser.java:76)
 at org.mortbay.jetty.HttpConnection.handle(HttpConnection.java:404)
 at com.google.apphosting.runtime.jetty.JettyServletEngineAdapter.serviceRequest(JettyServletEngineAdapter.java:135)
 at com.google.apphosting.runtime.JavaRuntime$RequestRunnable.run(JavaRuntime.java:422)
 at com.google.tracing.TraceContext$TraceContextRunnable.runInContext(TraceContext.java:449)
 at com.google.tracing.TraceContext$TraceContextRunnable$1.run(TraceContext.java:455)
 at com.google.tracing.TraceContext.runInContext(TraceContext.java:695)
 at com.google.tracing.TraceContext$AbstractTraceContextCallback.runInInheritedContextNoUnref(TraceContext.java:333)
 at com.google.tracing.TraceContext$AbstractTraceContextCallback.runInInheritedContext(TraceContext.java:325)
 at com.google.tracing.TraceContext$TraceContextRunnable.run(TraceContext.java:453)
 at com.google.apphosting.runtime.ThreadGroupPool$PoolEntry.run(ThreadGroupPool.java:251)
 at java.lang.Thread.run(Thread.java:679)
NestedThrowablesStackTrace:
Persistent class "Class org.pupit.model.PupitThread does not seem to have been enhanced.  You may want to rerun the enhancer and check for errors in the output." has no table in the database, but the operation requires it. Please check the specification of the MetaData for this class.
org.datanucleus.store.exceptions.NoTableManagedException: Persistent class "Class org.pupit.model.PupitThread does not seem to have been enhanced.  You may want to rerun the enhancer and check for errors in the output." has no table in the database, but the operation requires it. Please check the specification of the MetaData for this class.
 at org.datanucleus.store.appengine.DatastoreManager.getDatastoreClass(DatastoreManager.java:722)
 at org.datanucleus.store.appengine.query.DatastoreQuery.performExecute(DatastoreQuery.java:216)
 at org.datanucleus.store.appengine.query.JDOQLQuery.performExecute(JDOQLQuery.java:89)
 at org.datanucleus.store.query.Query.executeQuery(Query.java:1489)
 at org.datanucleus.store.query.Query.executeWithArray(Query.java:1371)
 at org.datanucleus.jdo.JDOQuery.execute(JDOQuery.java:243)
 at org.pupit.model.PupitThreadDao.selectByLastUpdateTime(PupitThreadDao.java:37)
 at org.pupit.service.thread.GetPupitThreadListServiceImpl.execute(GetPupitThreadListServiceImpl.java:13)
 at org.pupit.servlet.thread.IndexServlet.getThreadList(IndexServlet.java:54)
 at org.pupit.servlet.thread.IndexServlet.doGet(IndexServlet.java:25)
 at javax.servlet.http.HttpServlet.service(HttpServlet.java:617)
 at javax.servlet.http.HttpServlet.service(HttpServlet.java:717)
 at org.mortbay.jetty.servlet.ServletHolder.handle(ServletHolder.java:511)
 at org.mortbay.jetty.servlet.ServletHandler.handle(ServletHandler.java:390)
 at org.mortbay.jetty.security.SecurityHandler.handle(SecurityHandler.java:216)
 at org.mortbay.jetty.servlet.SessionHandler.handle(SessionHandler.java:182)
 at org.mortbay.jetty.handler.ContextHandler.handle(ContextHandler.java:765)
 at org.mortbay.jetty.webapp.WebAppContext.handle(WebAppContext.java:418)
 at org.mortbay.jetty.servlet.Dispatcher.forward(Dispatcher.java:327)
 at org.mortbay.jetty.servlet.Dispatcher.forward(Dispatcher.java:126)
 at com.google.apphosting.runtime.jetty.ResourceFileServlet.serveWelcomeFileAsForward(ResourceFileServlet.java:342)
 at com.google.apphosting.runtime.jetty.ResourceFileServlet.maybeServeWelcomeFile(ResourceFileServlet.java:281)
 at com.google.apphosting.runtime.jetty.ResourceFileServlet.doGet(ResourceFileServlet.java:142)
 at javax.servlet.http.HttpServlet.service(HttpServlet.java:617)
 at javax.servlet.http.HttpServlet.service(HttpServlet.java:717)
 at org.mortbay.jetty.servlet.ServletHolder.handle(ServletHolder.java:511)
 at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1166)
 at com.google.inject.servlet.FilterChainInvocation.doFilter(FilterChainInvocation.java:67)
 at com.google.inject.servlet.ManagedFilterPipeline.dispatch(ManagedFilterPipeline.java:122)
 at com.google.inject.servlet.GuiceFilter.doFilter(GuiceFilter.java:110)
 at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157)

調べてみたところeclipseでプロジェクトをクリーンすればいいとのことだったので、クリーンしたのですが、、↓のエラーがでてプロジェクトのクリーンにも失敗してましたorz...

Encountered a problem: Unexpected exception
Please see the logs [C:\Users\xxx\AppData\Local\Temp\enhance45590413067311643.log] for further information.

↓ファイルの内容↓

java.lang.RuntimeException: Unexpected exception
 at com.google.appengine.tools.enhancer.Enhancer.execute(Enhancer.java:59)
 at com.google.appengine.tools.enhancer.Enhance.(Enhance.java:60)
 at com.google.appengine.tools.enhancer.Enhance.main(Enhance.java:41)
Caused by: java.lang.reflect.InvocationTargetException
 at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
 at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
 at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
 at java.lang.reflect.Method.invoke(Method.java:601)
 at com.google.appengine.tools.enhancer.Enhancer.execute(Enhancer.java:57)
 ... 2 more
Caused by: org.datanucleus.exceptions.NucleusException: Plugin (Bundle) "org.datanucleus.store.appengine" is already registered. Ensure you dont have multiple JAR versions of the same plugin in the classpath. The URL "file:/C:/eclipse_web/plugins/com.google.appengine.eclipse.sdkbundle.1.4.0_1.4.0.v201012021502/appengine-java-sdk-1.4.0/lib/user/orm/datanucleus-appengine-1.0.8.final.jar" is already registered, and you are trying to register an identical plugin located at URL "file:/C:/eclipse_web/workspace/pu-pit/war/WEB-INF/lib/datanucleus-appengine-1.0.8.final.jar."
 at org.datanucleus.plugin.NonManagedPluginRegistry.registerBundle(NonManagedPluginRegistry.java:434)
 at org.datanucleus.plugin.NonManagedPluginRegistry.registerBundle(NonManagedPluginRegistry.java:340)
 at org.datanucleus.plugin.NonManagedPluginRegistry.registerExtensions(NonManagedPluginRegistry.java:222)
 at org.datanucleus.plugin.NonManagedPluginRegistry.registerExtensionPoints(NonManagedPluginRegistry.java:153)
 at org.datanucleus.plugin.PluginManager.registerExtensionPoints(PluginManager.java:82)
 at org.datanucleus.OMFContext.(OMFContext.java:160)
 at org.datanucleus.enhancer.DataNucleusEnhancer.(DataNucleusEnhancer.java:172)
 at org.datanucleus.enhancer.DataNucleusEnhancer.(DataNucleusEnhancer.java:150)
 at org.datanucleus.enhancer.DataNucleusEnhancer.main(DataNucleusEnhancer.java:1157)
 ... 7 more


これは、 WEB-INF/lib 以下から
datanucleus-appengine-1.0.8.final.jar
を削除して、プロジェクト>クリーン したらクリーンでました!

で、gaeをローカルで起動したところ
datanucleus-appengine-1.0.8.final.jar
がないので、起動できないと起こられたので、
datanucleus-appengine-1.0.8.final.jar
を元の場所(WEB-INF/lib以下)に戻して起動しました!!!!

<<追記>>
このJDOUserExceptionが出る状態でサーバにアップロードするとサーバ側でも同様のエラーが出るので、注意しましょう!

  • Digg
  • Del.icio.us
  • StumbleUpon
  • Reddit
  • Twitter
  • RSS

gaeのweb.xmlでは日本語コメントが使えない

gaeで自分の作ったアプリケーションをサーバにデプロイしようとしたら↓のエラーが!!



Unable to update:
com.google.apphosting.utils.config.AppEngineConfigException: Received SAXException parsing the input stream for C:\Users\iichico\AppData\Local\Temp\appcfg6630924556865691858.tmp\WEB-INF/web.xml
at com.google.apphosting.utils.config.AbstractConfigXmlReader.getTopLevelNode(AbstractConfigXmlReader.java:215)
at com.google.apphosting.utils.config.AbstractConfigXmlReader.parse(AbstractConfigXmlReader.java:229)
at com.google.apphosting.utils.config.WebXmlReader.processXml(WebXmlReader.java:146)
at com.google.apphosting.utils.config.WebXmlReader.processXml(WebXmlReader.java:22)
at com.google.apphosting.utils.config.AbstractConfigXmlReader.readConfigXml(AbstractConfigXmlReader.java:112)
at com.google.apphosting.utils.config.WebXmlReader.readWebXml(WebXmlReader.java:73)
at com.google.appengine.tools.admin.Application.compileJsps(Application.java:461)
at com.google.appengine.tools.admin.Application.createStagingDirectory(Application.java:334)
at com.google.appengine.tools.admin.AppAdminImpl.doUpdate(AppAdminImpl.java:282)
at com.google.appengine.tools.admin.AppAdminImpl.update(AppAdminImpl.java:48)
at com.google.appengine.eclipse.core.proxy.AppEngineBridgeImpl.deploy(AppEngineBridgeImpl.java:400)
at com.google.appengine.eclipse.core.deploy.DeployProjectJob.runInWorkspace(DeployProjectJob.java:148)
at org.eclipse.core.internal.resources.InternalWorkspaceJob.run(InternalWorkspaceJob.java:38)
at org.eclipse.core.internal.jobs.Worker.run(Worker.java:54)
Caused by: org.apache.xerces.impl.io.MalformedByteSequenceException: Invalid byte 3 of 3-byte UTF-8 sequence.
at org.apache.xerces.util.ErrorHandlerWrapper.createSAXParseException(Unknown Source)
at org.apache.xerces.util.ErrorHandlerWrapper.fatalError(Unknown Source)
at org.apache.xerces.impl.XMLErrorReporter.reportError(Unknown Source)
at org.apache.xerces.impl.XMLErrorReporter.reportError(Unknown Source)
at org.apache.xerces.impl.XMLDocumentFragmentScannerImpl$FragmentContentDispatcher.dispatch(Unknown Source)
at org.apache.xerces.impl.XMLDocumentFragmentScannerImpl.scanDocument(Unknown Source)
at org.apache.xerces.parsers.XML11Configuration.parse(Unknown Source)
at org.apache.xerces.parsers.XML11Configuration.parse(Unknown Source)
at org.apache.xerces.parsers.XMLParser.parse(Unknown Source)
at org.apache.xerces.parsers.AbstractSAXParser.parse(Unknown Source)
at org.apache.xerces.jaxp.SAXParserImpl$JAXPSAXParser.parse(Unknown Source)
at org.apache.xerces.jaxp.SAXParserImpl.parse(Unknown Source)
at org.mortbay.xml.XmlParser.parse(XmlParser.java:230)
at com.google.apphosting.utils.config.AbstractConfigXmlReader.getTopLevelNode(AbstractConfigXmlReader.java:207)
... 13 more
Caused by: org.apache.xerces.impl.io.MalformedByteSequenceException: Invalid byte 3 of 3-byte UTF-8 sequence.
at org.apache.xerces.impl.io.UTF8Reader.invalidByte(Unknown Source)
at org.apache.xerces.impl.io.UTF8Reader.read(Unknown Source)
at org.apache.xerces.impl.XMLEntityScanner.load(Unknown Source)
at org.apache.xerces.impl.XMLEntityScanner.peekChar(Unknown Source)
at org.apache.xerces.impl.XMLScanner.scanComment(Unknown Source)
at org.apache.xerces.impl.XMLDocumentFragmentScannerImpl.scanComment(Unknown Source)
... 23 more

  • Digg
  • Del.icio.us
  • StumbleUpon
  • Reddit
  • Twitter
  • RSS