要件はちゃんと確認しないと・・・VFSで作り直し

ちょっと前に書いたログファイルをダウンロードする機能ですが、実機となるLINUX上にデプロイすると動きませんでした。。。サーバへ接続はできているようなのですが瞬時に切断されているようです。原因切り分けの為、作業中に使っているユーザーIDで試してみると特に問題無く動いてます。TeraTermから指定のIDでログインしてみると即切断されたようになり、WinSCPでSFTPログインすると問題無くアクセスできます。
 で、要件を思い出すと、「SFTPはOK」ではなく、「SFTPのみOK」であった事に気が付きました。与えられたユーザーIDでは、SSH接続ができないセキュリティ設定になっていたという事なんです。(sshd_configでSFTP専用にされてました)
 要件上はたった2文字の違いですが、実装上は明らかに別物であり、SSHで接続するような実装ではNGという事になります。
 となるとGamynedのようにSSHを前提にしたライブラリでなく、SFTPのみでサーバへアクセスできるライブラリを探し、ApacheCommonsのVFSに落ち着きました。なお、VSFにはJschが必要みたいです。
 login,logoutはなさそうなんで、ManagedBeanの呼び出しメソッド引数にID、パスワード、ホスト名を追加して、SFTP処理クラスが殆どの変更箇所となります。とりあえず、ファイル受信は出来ましたので別途整理します。

  <dependencies>
	<dependency>
		<groupId>org.apache.commons</groupId>
		<artifactId>commons-vfs2</artifactId>
		<version>2.0</version>
	</dependency>
	<dependency>
		<groupId>com.jcraft</groupId>
		<artifactId>jsch</artifactId>
		<version>0.1.53</version>
	</dependency>
	<dependency>
		<groupId>org.apache.commons</groupId>
		<artifactId>commons-lang3</artifactId>
		<version>3.4</version>
	</dependency>
  </dependencies>
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.time.DateFormatUtils;
import org.apache.commons.vfs2.FileObject;
import org.apache.commons.vfs2.FileSystemManager;
import org.apache.commons.vfs2.FileSystemOptions;
import org.apache.commons.vfs2.FileType;
import org.apache.commons.vfs2.VFS;
import org.apache.commons.vfs2.provider.sftp.SftpFileSystemConfigBuilder;

/**
 * SFTPファイルアクセスクラス by apache commons vfs
 * */
@Stateless
public class SFTPFileAccess {
	
	public SFTPFileAccess(){
	}

	/**
	 * リモートホスト上の指定ディレクトリのファイル一覧を取得する
	 * @param String RemoteDir ディレクトリ名
	 * @param List<LogFile> セットするファイル一覧
	 * @param String User ユーザーID
	 * @param String Password パスワード
	 * @param String HostName ホスト名
	 * @throws IOException
	 * */
	public void getSFTPList(String RemoteDir, List<LogFile> FileList, String User, String Password, String HostName) throws IOException {

		FileSystemOptions opts = new FileSystemOptions();
		SftpFileSystemConfigBuilder.getInstance().setStrictHostKeyChecking(opts, "no");
		FileSystemManager fsManager = VFS.getManager();
		
    	String URL = "sftp://"+URLEncoder.encode(User, "UTF-8");
    	
    	URL = URL +":" + URLEncoder.encode(Password, "UTF-8");
    	URL = URL +"@" + URLEncoder.encode(HostName, "UTF-8");
    	URL = URL + RemoteDir;
    	
		FileObject localFileObject=fsManager.resolveFile(URL);
		
		//Directoryチェック
		if ( !localFileObject.getType().equals(FileType.FOLDER) ){
			return;
		}
		//ファイル存在チェック
		if( ! localFileObject.getType().hasChildren() ){
			return;
		}

		FileObject[] children = localFileObject.getChildren();
        for (int i = 0; i < children.length; i++) {
        	if(StringUtils.isEmpty(children[ i ].getName().getBaseName())){
	        	LogFile file = new LogFile();
	        	file.setFileName(children[ i ].getName().getBaseName());
	        	file.setDirectory(RemoteDir);
	        	file.setFileSize(children[ i ].getContent().getSize());
	        	Date tm = new Date(children[ i ].getContent().getLastModifiedTime());
	        	file.setUpDateTime(DateFormatUtils.format(tm, "yyyy/MM/dd HH:mm:ss"));
	        	FileList.add(file);
        	}
        }
    }
    /**
     * SFTPファイル受信
     * @param remoteFile 受信元ファイル
     * @param stream ストリーム
     * @param String User ユーザーID
     * @param String Password パスワード
     * @param String HostName ホスト名
     * */
    public void getFile( String remoteFile, OutputStream stream, String User, String Password, String HostName) throws IOException{
		FileSystemOptions opts = new FileSystemOptions();

		SftpFileSystemConfigBuilder.getInstance().setStrictHostKeyChecking(opts, "no");
		FileSystemManager fsManager = VFS.getManager();
        
		String URL = "sftp://"+URLEncoder.encode(User, "UTF-8");
    	URL = URL +":" + URLEncoder.encode(Password, "UTF-8");
    	URL = URL +"@" + URLEncoder.encode(HostName, "UTF-8");
    	URL = URL + remoteFile;
    	
		FileObject localFileObject = fsManager.resolveFile(URL);
		
		if(!localFileObject.exists()){
			throw new IOException(); 
		}
		
		InputStream inputStream = localFileObject.getContent().getInputStream();

		BufferedOutputStream bos = new BufferedOutputStream(stream);
        
        byte[] buffer = new byte[1024];
        int readLength = 0;
        
        while( (readLength = inputStream.read(buffer)) != -1){
          bos.write(buffer, 0, readLength);
        }
        // close
        inputStream.close();
        bos.close();
    }
}

源泉徴収の知らせがきた

税務署から「源泉所得税及び復興所得税の納付期限のお知らせ」というハガキが来ていました。
9月末に「給与支払事務所等の開設申請書」の届けを出してから、それ以降、いつ手続きすればよいのかわからなかったんですが、ハガキに納付期限が1月20日と書いてありました。となると、「源泉所得税の納期の特例の承認に関する申請書」によって半年に一度手続きすればよいはずなので、1月と7月がそのタイミングなんですね。ようやく解りました。

bootstrapのdatepickerを使ってみる

前回、今作成中のWebLogicアプリにbootstrapを使う事にしたので、次は日付入力にdatepickerを使ってみることにしました。datepickerって、使う方も入力しやすいし、作る方も余計な日付チェック処理を作らなくて済むし、素晴らしいコンポーネントだと思います。昔はテキスト入力エリアを年/月/日と3個繋げて、日付妥当性やメンドクサイうるう年チェックとかしてたんですよ・・・
しかも、ダウンロードURLではオプション設定の書き方も教えてくれてます。
bootstrap-datepicker
JSF2画面xhtmlの該当箇所は下記ですが、そのままだとBean側が更新されないので、f:ajaxで通知してます。

<h:outputStylesheet name="bootstrap/css/bootstrap-datepicker.min.css" />
<h:outputScript name="bootstrap/js/bootstrap-datepicker.min.js" />
<h:outputScript name="bootstrap/js/bootstrap-datepicker.ja.min.js" />
<script type="text/javascript">
$(function() {
  $('.datepicker').datepicker({
    language: 'ja',
    format: 'yyyy/mm/dd',
    autoclose: true,
    todayHighlight: true
  });
});
</script>
・・・中略
          <h:inputText value="#{bean.inDate}" class="from-control datepicker" >
             <f:ajax  execute="@this"/>
          </h:inputText>
・・・以下省略

画面メニューにbootstrapのnavbarを使ってみる

今作っているWebアプリですが、ログインユーザー権限によりメニューのリンク表示を切り替える要件の実装について、一緒に作ってるメンバーに聞いてみたら、bootstrapのnavbarがかっこいいとの事で使ってみました。
各画面で指定するテンプレートxhtmlにbootstrapのnavbarをセットする形です。なお、bootstrapのバージョンは少し古いですが2.3.2を使ってます。
bootstrapをダウンロードしたら、Webソース内のresourcesにダウンロードした各ディレクトリを入れ、メニューを表示する各画面xhtmlでは、下記のようにテンプレートを指定します。

<ui:composition template="../templates/menu.xhtml">

テンプレートとして指定したmenu.xhtmlですが、その中の「SessionManager」はログイン情報を保持しているSessionScopedBeanで、ログイン時にログイン日時をlogintimeに、各権限の値をroleCdにセットし、xhtml側ではui:fragmentのrenderedで表示を切り替えます。

<h:outputStylesheet name="bootstrap/css/bootstrap.css" />
<h:outputStylesheet name="bootstrap/css/bootstrap-responsive.css" />
<h:outputScript name="bootstrap/js/bootstrap.js" />
<h:body>
  <nav class="navbar navbar-inverse navbar-default">
    <div class="navbar-inner">
    <div id="container">
      <h:form>
        <ul class="nav navbar-nav">
          <ui:fragment rendered="#{SessionManager.roleCd == 1}">
            <li><a href="../sales/page1.xhtml">ファイル照会</a></li>
            <li><a href="../sales/page2.xhtml">状態確認</a></li>
            <li><a href="../common/pwchange.xhtml">パスワード更新</a></li>
          </ui:fragment>
          <ui:fragment rendered="#{SessionManager.roleCd == 2}">
            <li><a href="../usermanage/page1.xhtml">ユーザー管理</a></li>
            <li><a href="../usermanage/page2.xhtml">ユーザー一覧</a></li>
            <li><a href="../common/pwchange.xhtml">パスワード更新</a></li>
          </ui:fragment>
・・・他の設定は割愛
          <ui:insert name="navlinks" ></ui:insert>
        </ul>
        <h:commandButton class="btn btn-primary btn-lg pull-right" value="ログアウト" action="#{SessionManager.logout}" />
        <h:outputText class="span3 pull-right" style="color: #ffffff;" value="#{SessionManager.logintime}"/>
        </h:form>
      <ui:insert name="navinfo"/>
    </div>
    </div>
  </nav>

ログインした権限に従い下記のようにメニューを表示されます。
navbar