/* * Aipo is a groupware program developed by TOWN, Inc. * Copyright (C) 2004-2015 TOWN, Inc. * http://www.aipo.com * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ package com.aimluck.eip.blog; import java.io.IOException; import java.io.StringWriter; import java.util.ArrayList; import java.util.Calendar; import java.util.Date; import java.util.List; import org.apache.cayenne.exp.Expression; import org.apache.cayenne.exp.ExpressionFactory; import org.apache.jetspeed.om.security.UserIdPrincipal; import org.apache.jetspeed.services.JetspeedSecurity; import org.apache.jetspeed.services.logging.JetspeedLogFactoryService; import org.apache.jetspeed.services.logging.JetspeedLogger; import org.apache.jetspeed.services.resources.JetspeedResources; import org.apache.turbine.services.TurbineServices; import org.apache.turbine.services.velocity.VelocityService; import org.apache.turbine.util.RunData; import org.apache.velocity.context.Context; import com.aimluck.commons.field.ALStringField; import com.aimluck.eip.blog.util.BlogUtils; import com.aimluck.eip.cayenne.om.portlet.EipTBlogComment; import com.aimluck.eip.cayenne.om.portlet.EipTBlogEntry; import com.aimluck.eip.cayenne.om.security.TurbineUser; import com.aimluck.eip.common.ALAbstractFormData; import com.aimluck.eip.common.ALBaseUser; import com.aimluck.eip.common.ALDBErrorException; import com.aimluck.eip.common.ALEipConstants; import com.aimluck.eip.common.ALEipUser; import com.aimluck.eip.common.ALPageNotFoundException; import com.aimluck.eip.common.ALPermissionException; import com.aimluck.eip.mail.ALAdminMailContext; import com.aimluck.eip.mail.ALAdminMailMessage; import com.aimluck.eip.mail.ALMailService; import com.aimluck.eip.mail.util.ALEipUserAddr; import com.aimluck.eip.mail.util.ALMailUtils; import com.aimluck.eip.modules.actions.blog.BlogAction; import com.aimluck.eip.modules.actions.common.ALAction; import com.aimluck.eip.modules.screens.BlogDetailScreen; import com.aimluck.eip.modules.screens.BlogEntryFormJSONScreen; import com.aimluck.eip.orm.Database; import com.aimluck.eip.orm.query.SelectQuery; import com.aimluck.eip.services.accessctl.ALAccessControlConstants; import com.aimluck.eip.services.orgutils.ALOrgUtilsService; import com.aimluck.eip.util.ALCellularUtils; import com.aimluck.eip.util.ALEipUtils; import com.aimluck.eip.util.ALLocalizationUtils; /** * ブログエントリー・コメントのフォームデータを管理するクラスです。 <BR> * */ public class BlogEntryCommentFormData extends ALAbstractFormData { /** logger */ private static final JetspeedLogger logger = JetspeedLogFactoryService .getLogger(BlogEntryCommentFormData.class.getName()); /** コメント */ private ALStringField comment; private boolean sendEmailToPC = false; private boolean sendEmailToCellular = false; /** メール送信時のメッセージ種別(ブログ) 3=PC,Celluler 2=Celluer 1=PC 0=送信しない */ private int MsgTypeBlog = 0; /** <code>login_user</code> ログインユーザー */ private ALEipUser login_user; /** アクセス権限の機能名 */ private String aclPortletFeature = null; /** * * @param action * @param rundata * @param context * * */ @Override public void init(ALAction action, RunData rundata, Context context) throws ALPageNotFoundException, ALDBErrorException { super.init(action, rundata, context); try { MsgTypeBlog = ALMailUtils.getSendDestType(ALMailUtils.KEY_MSGTYPE_BLOG); if ((MsgTypeBlog & 1) > 0) { sendEmailToPC = true; } else { sendEmailToPC = false; } if ((MsgTypeBlog & 2) > 0) { sendEmailToCellular = true; } else { sendEmailToCellular = false; } } catch (Throwable t) { sendEmailToPC = false; sendEmailToCellular = false; } login_user = ALEipUtils.getALEipUser(rundata); int uid = ALEipUtils.getUserId(rundata); // アクセス権 if ("commentdel".equals(action.getMode())) { String commentid = rundata.getParameters().getString("comment_id"); int comment_view_uid = BlogUtils.getCommentViewId(rundata, context, uid, commentid); if (uid == comment_view_uid) { aclPortletFeature = ALAccessControlConstants.POERTLET_FEATURE_BLOG_ENTRY_REPLY; } else { aclPortletFeature = ALAccessControlConstants.POERTLET_FEATURE_BLOG_ENTRY_OTHER_REPLY; } } } /** * 各フィールドを初期化します。 <BR> * * */ @Override public void initField() { // コメント comment = new ALStringField(); comment.setFieldName(ALLocalizationUtils.getl10n("BLOG_COMMENT")); comment.setTrim(false); } /** * 掲示板の各フィールドに対する制約条件を設定します。 <BR> * * */ @Override protected void setValidator() { // メモ必須項目 comment.setNotNull(true); // メモの文字数制限 comment.limitMaxLength(1000); } /** * トピックのフォームに入力されたデータの妥当性検証を行います。 <BR> * * @param msgList * @return TRUE 成功 FALSE 失敗 * */ @Override protected boolean validate(List<String> msgList) { // メモ comment.validate(msgList); return (msgList.size() == 0); } /** * トピックをデータベースから読み出します。 <BR> * * @param rundata * @param context * @param msgList * @return TRUE 成功 FALSE 失敗 */ @Override protected boolean loadFormData(RunData rundata, Context context, List<String> msgList) { return false; } /** * コメントをデータベースから削除します。 <BR> * * @param rundata * @param context * @param msgList * @return TRUE 成功 FALSE 失敗 */ @Override protected boolean deleteFormData(RunData rundata, Context context, List<String> msgList) throws ALPageNotFoundException, ALDBErrorException { try { String commentid = rundata.getParameters().getString("comment_id"); // オブジェクトモデルを取得 EipTBlogComment comment = BlogUtils.getEipTBlogComment(rundata, context, commentid); if (comment == null) { // 指定した トピック ID のレコードが見つからない場合 logger.debug("[BlogEntryCommentFormData] Not found ID..."); throw new ALPageNotFoundException(); } Database.delete(comment); Database.commit(); } catch (Exception e) { Database.rollback(); logger.error("BlogEntryCommentFormData.deleteFormData", e); throw new ALDBErrorException(); } return true; } /** * コメントをデータベースに格納します。 <BR> * * @param rundata * @param context * @param msgList * @return TRUE 成功 FALSE 失敗 */ @Override protected boolean insertFormData(RunData rundata, Context context, List<String> msgList) throws ALPageNotFoundException, ALDBErrorException { try { // オブジェクトモデルを取得 EipTBlogEntry parententry = BlogUtils.getEipTBlogParentEntry(rundata, context); if (parententry == null) { // 指定した エントリー ID のレコードが見つからない場合 logger.debug("[BlogEntryCommentFormData] Not found ID..."); throw new ALPageNotFoundException(); } int uid = ALEipUtils.getUserId(rundata); Date updateDate = Calendar.getInstance().getTime(); EipTBlogEntry entry = Database.get(EipTBlogEntry.class, Integer.valueOf(parententry .getEntryId() .intValue())); // 新規オブジェクトモデル EipTBlogComment blogcomment = Database.create(EipTBlogComment.class); // ユーザーID blogcomment.setOwnerId(Integer.valueOf(uid)); // コメント blogcomment.setComment(comment.getValue()); // エントリーID blogcomment.setEipTBlogEntry(entry); // 作成日 blogcomment.setCreateDate(Calendar.getInstance().getTime()); // 更新日 blogcomment.setUpdateDate(updateDate); // トピックを登録 Database.commit(); // 更新情報を送るユーザーを取得する List<ALEipUser> recipientList = getRecipientList(rundata, context); // アクティビティ ALEipUser loginName = ALEipUtils.getALEipUser(uid); BlogUtils.createNewCommentActivity( entry, loginName.getName().getValue(), blogcomment); // あなた宛のお知らせ List<String> recipientNameList = new ArrayList<String>(); for (ALEipUser recipient : recipientList) { recipientNameList.add(recipient.getName().toString()); } BlogUtils.createNewBlogTopicActivity(entry, loginName .getName() .toString(), recipientNameList, blogcomment); // メール送信 if (sendEmailToPC || sendEmailToCellular) { List<ALEipUserAddr> destMemberList = ALMailUtils.getALEipUserAddrs(recipientList, ALEipUtils .getUserId(rundata), false); String orgId = Database.getDomainName(); String subject = "[" + JetspeedResources.getString("aipo.alias") + "]ブログコメント"; // パソコン、携帯電話へメールを送信 List<ALAdminMailMessage> messageList = new ArrayList<ALAdminMailMessage>(); for (ALEipUserAddr destMember : destMemberList) { ALAdminMailMessage message = new ALAdminMailMessage(destMember); if (sendEmailToPC) { message.setPcSubject(subject); message.setPcBody(createMsgForPc(rundata)); } if (sendEmailToCellular) { message.setCellularSubject(subject); message.setCellularBody(createMsgForCellPhone(rundata, destMember .getUserId())); } messageList.add(message); } ALMailService.sendAdminMailAsync(new ALAdminMailContext( orgId, (int) login_user.getUserId().getValue(), messageList, ALMailUtils.getSendDestType(ALMailUtils.KEY_MSGTYPE_BLOG))); } } catch (Exception e) { Database.rollback(); logger.error("BlogEntryCommentFormData.insertFormData", e); throw new ALDBErrorException(); } return true; } /** * 当該ブログの更新通知ユーザーを習得します。 <BR> * * @param rundata * @param context * @return * @throws ALDBErrorException * @throws ALPageNotFoundException */ private List<ALEipUser> getRecipientList(RunData rundata, Context context) throws ALPageNotFoundException, ALDBErrorException { Integer loginUserId = ALEipUtils.getUserId(rundata); // 関連トピックをすべて取得する EipTBlogEntry parenttopic = BlogUtils.getEipTBlogParentEntry(rundata, context); SelectQuery<EipTBlogComment> topicQuery = Database.query(EipTBlogComment.class); Expression topicExp = ExpressionFactory.matchDbExp(EipTBlogComment.ENTRY_ID_COLUMN, parenttopic .getEntryId()); topicQuery.setQualifier(topicExp); // トピックに関連する全てのユーザーIDを取得する List<Integer> userIdList = new ArrayList<Integer>(); Integer userId = parenttopic.getOwnerId(); if (!loginUserId.equals(userId)) { userIdList.add(userId); } List<EipTBlogComment> topicList = topicQuery.fetchList(); for (EipTBlogComment topic : topicList) { userId = topic.getOwnerId(); if (!userId.equals(loginUserId) && !userIdList.contains(userId)) { userIdList.add(userId); } } if (userIdList.isEmpty()) { return new ArrayList<ALEipUser>(0); } // ユーザーIDからユーザー情報を取得する。 SelectQuery<TurbineUser> userQuery = Database.query(TurbineUser.class); Expression userExp = ExpressionFactory.inDbExp(TurbineUser.USER_ID_PK_COLUMN, userIdList); userQuery.setQualifier(userExp); return ALEipUtils.getUsersFromSelectQuery(userQuery); } /** * パソコンへ送信するメールの内容を作成する. * * @return */ private String createMsgForPc(RunData rundata) { ALEipUser user = ALEipUtils.getALEipUser(rundata); boolean enableAsp = JetspeedResources.getBoolean("aipo.asp", false); ALBaseUser user2 = null; try { user2 = (ALBaseUser) JetspeedSecurity.getUser(new UserIdPrincipal(user .getUserId() .toString())); } catch (Exception e) { return ""; } StringWriter out = null; String e_mail_addr = user2.getEmail(); try { VelocityService service = (VelocityService) ((TurbineServices) TurbineServices.getInstance()) .getService(VelocityService.SERVICE_NAME); Context context = service.getContext(); context.put("userLastName", user2.getLastName()); context.put("userFirstName", user2.getFirstName()); if (!e_mail_addr.equals("")) { context.put("mailAddress", "(" + e_mail_addr + ")"); } context.put("mailNoticeMSG", ALLocalizationUtils .getl10n("BLOG_YOU_GOT_MAIL_NOTICE")); context.put("comment", ALLocalizationUtils .getl10n("BLOG_MAIL_NOTICE_COMMENT")); context.put("commentValue", comment.getValue()); context.put("accessToAlias", "[" + ALOrgUtilsService.getAlias() + ALLocalizationUtils.getl10n("BLOG_MAIL_NOTICE_ACCESS_TO") + "]"); if (enableAsp) { context.put("globalUrl1", ALMailUtils.getGlobalurl()); } else { context.put("outsideOffice", ALLocalizationUtils .getl10n("BLOG_MAIL_NOTICE_OUTSIDE_OFFICE")); context.put("globalurl2", ALMailUtils.getGlobalurl()); context.put("insideOffice", ALLocalizationUtils .getl10n("BLOG_MAIL_NOTICE_INSIDE_OFFICE")); context.put("globalUrl3", ALMailUtils.getLocalurl()); } context.put("Alias", ALOrgUtilsService.getAlias()); out = new StringWriter(); service.handleRequest(context, "mail/getBlogNotice.vm", out); out.flush(); return out.toString(); } catch (Exception e) { StringBuffer trace = new StringBuffer(); String message = e.getMessage(); logger.warn(message, e); e.printStackTrace(); trace.append("\n").append(message).append("\n").append(e.toString()); } finally { if (out != null) { try { out.close(); } catch (IOException e) { // ignore } } } return null; } /** * 携帯電話へ送信するメールの内容を作成する. * * @return */ private String createMsgForCellPhone(RunData rundata, int destUserID) { ALEipUser user = ALEipUtils.getALEipUser(rundata); ALBaseUser user2 = null; try { user2 = (ALBaseUser) JetspeedSecurity.getUser(new UserIdPrincipal(user .getUserId() .toString())); } catch (Exception e) { return ""; } StringWriter out = null; String e_mail_addr = user2.getEmail(); try { VelocityService service = (VelocityService) ((TurbineServices) TurbineServices.getInstance()) .getService(VelocityService.SERVICE_NAME); Context context = service.getContext(); context.put("userLastName", user2.getLastName()); context.put("userFirstName", user2.getFirstName()); if (!e_mail_addr.equals("")) { context.put("mailAddress", "(" + e_mail_addr + ")"); } context.put("mailNoticeMSG", ALLocalizationUtils .getl10n("BLOG_YOU_GOT_MAIL_NOTICE")); context.put("comment", ALLocalizationUtils .getl10n("BLOG_MAIL_NOTICE_COMMENT")); context.put("commentValue", comment.getValue()); ALEipUser destUser; try { destUser = ALEipUtils.getALEipUser(destUserID); } catch (ALDBErrorException ex) { logger.error("BlogEntryCommentFormData.createMsgForCellPhone", ex); return ""; } context.put("accessToAlias", "[" + ALOrgUtilsService.getAlias() + ALLocalizationUtils.getl10n("BLOG_MAIL_NOTICE_ACCESS_TO") + "]"); context.put("globalUrl1", ALMailUtils.getGlobalurl() + "?key=" + ALCellularUtils.getCellularKey(destUser)); context.put("Alias", ALOrgUtilsService.getAlias()); out = new StringWriter(); service.handleRequest(context, "mail/getBlogNotice.vm", out); out.flush(); return out.toString(); } catch (Exception e) { StringBuffer trace = new StringBuffer(); String message = e.getMessage(); logger.warn(message, e); e.printStackTrace(); trace.append("\n").append(message).append("\n").append(e.toString()); } finally { if (out != null) { try { out.close(); } catch (IOException e) { // ignore } } } return null; } /** * データベースに格納されているコメントを更新します。 <BR> * * @param rundata * @param context * @param msgList * @return TRUE 成功 FALSE 失敗 */ @Override protected boolean updateFormData(RunData rundata, Context context, List<String> msgList) { return false; } /** * エントリー詳細表示ページからデータを新規登録します。 * * @param action * @param rundata * @param context * @return TRUE 成功 FALSE 失敗 */ @Override public boolean doInsert(ALAction action, RunData rundata, Context context) { try { if (!doCheckSecurity(rundata, context)) { return false; } init(action, rundata, context); doCheckAclPermission( rundata, context, ALAccessControlConstants.VALUE_ACL_INSERT); action.setMode(ALEipConstants.MODE_INSERT); ArrayList<String> msgList = new ArrayList<String>(); setValidator(); boolean res = false; if (isOverQuota()) { msgList.add(ALLocalizationUtils .getl10n("COMMON_FULL_DISK_DELETE_DETA_OR_CHANGE_PLAN")); } else { res = (setFormData(rundata, context, msgList) && validate(msgList) && insertFormData( rundata, context, msgList)); } if (!res) { action.setMode(ALEipConstants.MODE_NEW_FORM); setMode(action.getMode()); } if (action instanceof BlogEntryFormJSONScreen) { action.setResultData(this); action.addErrorMessages(msgList); action.putData(rundata, context); } else { BlogAction blogAction = (BlogAction) action; blogAction.setResultDataOnCommentDetail(this); blogAction.addErrorMessagesOnCommentDetail(msgList); blogAction.putDataOnCommentDetail(rundata, context); } return res; } catch (ALPermissionException e) { ALEipUtils.redirectPermissionError(rundata); return false; } catch (ALPageNotFoundException e) { ALEipUtils.redirectPageNotFound(rundata); return false; } catch (ALDBErrorException e) { ALEipUtils.redirectDBError(rundata); return false; } } /** * エントリー詳細表示ページにフォームを表示します。 * * @param action * @param rundata * @param context * @return TRUE 成功 FALSE 失敗 */ @Override public boolean doViewForm(ALAction action, RunData rundata, Context context) { try { init(action, rundata, context); // action.setMode(isedit ? ALEipConstants.MODE_EDIT_FORM // : ALEipConstants.MODE_NEW_FORM); // mode = action.getMode(); // doCheckAclPermission(rundata, context, // ALAccessControlConstants.VALUE_ACL_DETAIL); ArrayList<String> msgList = new ArrayList<String>(); boolean res = setFormData(rundata, context, msgList); if (action instanceof BlogDetailScreen) { BlogDetailScreen blogAction = (BlogDetailScreen) action; blogAction.setResultDataOnCommentDetail(this); blogAction.addErrorMessagesOnCommentDetail(msgList); blogAction.putDataOnCommentDetail(rundata, context); } else { BlogAction blogAction = (BlogAction) action; blogAction.setResultDataOnCommentDetail(this); blogAction.addErrorMessagesOnCommentDetail(msgList); blogAction.putDataOnCommentDetail(rundata, context); } return res; // } catch (ALPermissionException e) { // ALEipUtils.redirectPermissionError(rundata); // return false; } catch (ALPageNotFoundException e) { ALEipUtils.redirectPageNotFound(rundata); return false; } catch (ALDBErrorException e) { ALEipUtils.redirectDBError(rundata); return false; } } /** * * @param rundata * @param context * @param msgList * @return * @throws ALPageNotFoundException * @throws ALDBErrorException */ @Override protected boolean setFormData(RunData rundata, Context context, List<String> msgList) throws ALPageNotFoundException, ALDBErrorException { boolean res = super.setFormData(rundata, context, msgList); return res; } /** * メモを取得します。 <BR> * * @return */ public ALStringField getComment() { return comment; } /** * アクセス権限チェック用メソッド。<br /> * アクセス権限の機能名を返します。 * * @return */ @Override public String getAclPortletFeature() { return aclPortletFeature; } }