/** * $RCSfile$ * $Revision: 1751 $ * $Date: 2005-08-07 20:08:47 -0300 (Sun, 07 Aug 2005) $ * * Copyright (C) 2004 Jive Software. All rights reserved. * * This software is published under the terms of the GNU Public License (GPL), * a copy of which is included in this distribution. */ package org.jivesoftware.wildfire.roster; import org.jivesoftware.database.DbConnectionManager; import org.jivesoftware.database.SequenceManager; import org.jivesoftware.util.JiveConstants; import org.jivesoftware.util.LocaleUtils; import org.jivesoftware.util.Log; import org.jivesoftware.wildfire.user.UserAlreadyExistsException; import org.jivesoftware.wildfire.user.UserNotFoundException; import org.xmpp.packet.JID; import java.sql.Connection; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.util.ArrayList; import java.util.Iterator; import java.util.LinkedList; import java.util.List; /** * Defines the provider methods required for creating, reading, updating and deleting roster * items.<p> * * Rosters are another user resource accessed via the user or chatbot's long ID. A user/chatbot * may have zero or more roster items and each roster item may have zero or more groups. Each * roster item is additionaly keyed on a XMPP jid. In most cases, the entire roster will be read * in from memory and manipulated or sent to the user. However some operations will need to retrive * specific roster items rather than the entire roster. * * @author Iain Shigeoka */ public class RosterItemProvider { private static final String CREATE_ROSTER_ITEM = "INSERT INTO jiveRoster (username, rosterID, jid, sub, ask, recv, nick) " + "VALUES (?, ?, ?, ?, ?, ?, ?)"; private static final String UPDATE_ROSTER_ITEM = "UPDATE jiveRoster SET sub=?, ask=?, recv=?, nick=? WHERE rosterID=?"; private static final String DELETE_ROSTER_ITEM_GROUPS = "DELETE FROM jiveRosterGroups WHERE rosterID=?"; private static final String CREATE_ROSTER_ITEM_GROUPS = "INSERT INTO jiveRosterGroups (rosterID, rank, groupName) VALUES (?, ?, ?)"; private static final String DELETE_ROSTER_ITEM = "DELETE FROM jiveRoster WHERE rosterID=?"; private static final String LOAD_USERNAMES = "SELECT DISTINCT username from jiveRoster WHERE jid=?"; private static final String COUNT_ROSTER_ITEMS = "SELECT COUNT(rosterID) FROM jiveRoster WHERE username=?"; private static final String LOAD_ROSTER = "SELECT jid, rosterID, sub, ask, recv, nick FROM jiveRoster WHERE username=?"; private static final String LOAD_ROSTER_ITEM_GROUPS = "SELECT groupName FROM jiveRosterGroups WHERE rosterID=? ORDER BY rank"; private static RosterItemProvider instance = new RosterItemProvider(); public static RosterItemProvider getInstance() { return instance; } /** * Creates a new roster item for the given user (optional operation).<p> * * <b>Important!</b> The item passed as a parameter to this method is strictly a convenience * for passing all of the data needed for a new roster item. The roster item returned from the * method will be cached by Wildfire. In some cases, the roster item passed in will be passed * back out. However, if an implementation may return RosterItems as a separate class * (for example, a RosterItem that directly accesses the backend storage, or one that is an * object in an object database).<p> * * If you don't want roster items edited through wildfire, throw * UnsupportedOperationException. * * @param username the username of the user/chatbot that owns the roster item * @param item the settings for the roster item to create * @return The created roster item */ public RosterItem createItem(String username, RosterItem item) throws UserAlreadyExistsException { Connection con = null; PreparedStatement pstmt = null; try { con = DbConnectionManager.getConnection(); long rosterID = SequenceManager.nextID(JiveConstants.ROSTER); pstmt = con.prepareStatement(CREATE_ROSTER_ITEM); pstmt.setString(1, username); pstmt.setLong(2, rosterID); pstmt.setString(3, item.getJid().toBareJID()); pstmt.setInt(4, item.getSubStatus().getValue()); pstmt.setInt(5, item.getAskStatus().getValue()); pstmt.setInt(6, item.getRecvStatus().getValue()); pstmt.setString(7, item.getNickname()); pstmt.executeUpdate(); item.setID(rosterID); insertGroups(rosterID, item.getGroups().iterator(), con); } catch (SQLException e) { throw new UserAlreadyExistsException(item.getJid().toBareJID()); } finally { try { if (pstmt != null) { pstmt.close(); } } catch (Exception e) { Log.error(e); } try { if (con != null) { con.close(); } } catch (Exception e) { Log.error(e); } } return item; } /** * Update the roster item in storage with the information contained in the given item * (optional operation).<p> * * If you don't want roster items edited through wildfire, throw UnsupportedOperationException. * * @param username the username of the user/chatbot that owns the roster item * @param item The roster item to update * @throws UserNotFoundException If no entry could be found to update */ public void updateItem(String username, RosterItem item) throws UserNotFoundException { Connection con = null; PreparedStatement pstmt = null; long rosterID = item.getID(); try { con = DbConnectionManager.getConnection(); // Update existing roster item pstmt = con.prepareStatement(UPDATE_ROSTER_ITEM); pstmt.setInt(1, item.getSubStatus().getValue()); pstmt.setInt(2, item.getAskStatus().getValue()); pstmt.setInt(3, item.getRecvStatus().getValue()); pstmt.setString(4, item.getNickname()); pstmt.setLong(5, rosterID); pstmt.executeUpdate(); // Close now the statement (do not wait to be GC'ed) pstmt.close(); // Delete old group list pstmt = con.prepareStatement(DELETE_ROSTER_ITEM_GROUPS); pstmt.setLong(1, rosterID); pstmt.executeUpdate(); insertGroups(rosterID, item.getGroups().iterator(), con); } catch (SQLException e) { Log.error(LocaleUtils.getLocalizedString("admin.error"), e); } finally { try { if (pstmt != null) { pstmt.close(); } } catch (Exception e) { Log.error(e); } try { if (con != null) { con.close(); } } catch (Exception e) { Log.error(e); } } } /** * Delete the roster item with the given itemJID for the user (optional operation).<p> * * If you don't want roster items deleted through wildfire, throw * UnsupportedOperationException. * * @param username the long ID of the user/chatbot that owns the roster item * @param rosterItemID The roster item to delete */ public void deleteItem(String username, long rosterItemID) { // Only try to remove the user if they exist in the roster already: Connection con = null; PreparedStatement pstmt = null; try { con = DbConnectionManager.getConnection(); // Remove roster groups pstmt = con.prepareStatement(DELETE_ROSTER_ITEM_GROUPS); pstmt.setLong(1, rosterItemID); pstmt.executeUpdate(); // Close now the statement (do not wait to be GC'ed) pstmt.close(); // Remove roster pstmt = con.prepareStatement(DELETE_ROSTER_ITEM); pstmt.setLong(1, rosterItemID); pstmt.executeUpdate(); } catch (SQLException e) { Log.error(LocaleUtils.getLocalizedString("admin.error"), e); } finally { try { if (pstmt != null) { pstmt.close(); } } catch (Exception e) { Log.error(e); } try { if (con != null) { con.close(); } } catch (Exception e) { Log.error(e); } } } /** * Returns an iterator on the usernames whose roster includes the specified JID. * * @param jid the jid that the rosters should include. * @return an iterator on the usernames whose roster includes the specified JID. */ public Iterator<String> getUsernames(String jid) { List<String> answer = new ArrayList<String>(); Connection con = null; PreparedStatement pstmt = null; try { con = DbConnectionManager.getConnection(); pstmt = con.prepareStatement(LOAD_USERNAMES); pstmt.setString(1, jid); ResultSet rs = pstmt.executeQuery(); while (rs.next()) { answer.add(rs.getString(1)); } } catch (SQLException e) { Log.error(LocaleUtils.getLocalizedString("admin.error"), e); } finally { try { if (pstmt != null) { pstmt.close(); } } catch (Exception e) { Log.error(e); } try { if (con != null) { con.close(); } } catch (Exception e) { Log.error(e); } } return answer.iterator(); } /** * Obtain a count of the number of roster items available for the given user. * * @param username the username of the user/chatbot that owns the roster items * @return The number of roster items available for the user */ public int getItemCount(String username) { int count = 0; Connection con = null; PreparedStatement pstmt = null; try { con = DbConnectionManager.getConnection(); pstmt = con.prepareStatement(COUNT_ROSTER_ITEMS); pstmt.setString(1, username); ResultSet rs = pstmt.executeQuery(); if (rs.next()) { count = rs.getInt(1); } } catch (SQLException e) { Log.error(LocaleUtils.getLocalizedString("admin.error"), e); } finally { try { if (pstmt != null) { pstmt.close(); } } catch (Exception e) { Log.error(e); } try { if (con != null) { con.close(); } } catch (Exception e) { Log.error(e); } } return count; } /** * Retrieve an iterator of RosterItems for the given user.<p> * * This method will commonly be called when a user logs in. The data will be cached * in memory when possible. However, some rosters may be very large so items may need * to be retrieved from the provider more frequently than usual for provider data. * * @param username the username of the user/chatbot that owns the roster items * @return An iterator of all RosterItems owned by the user */ public Iterator<RosterItem> getItems(String username) { LinkedList<RosterItem> itemList = new LinkedList<RosterItem>(); Connection con = null; PreparedStatement pstmt = null; try { // Load all the contacts in the roster con = DbConnectionManager.getConnection(); pstmt = con.prepareStatement(LOAD_ROSTER); pstmt.setString(1, username); ResultSet rs = pstmt.executeQuery(); while (rs.next()) { // Create a new RosterItem (ie. user contact) from the stored information RosterItem item = new RosterItem(rs.getLong(2), new JID(rs.getString(1)), RosterItem.SubType.getTypeFromInt(rs.getInt(3)), RosterItem.AskType.getTypeFromInt(rs.getInt(4)), RosterItem.RecvType.getTypeFromInt(rs.getInt(5)), rs.getString(6), null); // Add the loaded RosterItem (ie. user contact) to the result itemList.add(item); } // Close the statement and result set pstmt.close(); rs.close(); // Set null to pstmt to be sure that it's not closed twice. It seems that // Sybase driver is raising an error when trying to close an alrady closed statement. pstmt = null; // Load the groups for the loaded contact for (RosterItem item : itemList) { pstmt = con.prepareStatement(LOAD_ROSTER_ITEM_GROUPS); pstmt.setLong(1, item.getID()); rs = pstmt.executeQuery(); while (rs.next()) { item.getGroups().add(rs.getString(1)); } // Close the result set rs.close(); // Close the prepared statement pstmt.close(); } } catch (SQLException e) { Log.error(LocaleUtils.getLocalizedString("admin.error"), e); } finally { try { if (pstmt != null) { pstmt.close(); } } catch (Exception e) { Log.error(e); } try { if (con != null) { con.close(); } } catch (Exception e) { Log.error(e); } } return itemList.iterator(); } /** * Insert the groups into the given roster item. * * @param rosterID The roster ID of the item the groups belong to * @param iter An iterator over the group names to insert */ private void insertGroups(long rosterID, Iterator<String> iter, Connection con) throws SQLException { PreparedStatement pstmt = null; String groupName = null; try { pstmt = con.prepareStatement(CREATE_ROSTER_ITEM_GROUPS); pstmt.setLong(1, rosterID); for (int i = 0; iter.hasNext(); i++) { pstmt.setInt(2, i); groupName = iter.next(); pstmt.setString(3, groupName); try { pstmt.executeUpdate(); } catch (SQLException e) { Log.error(e); } } } finally { try { if (pstmt != null) { pstmt.close(); } } catch (Exception e) { Log.error("Error inserting group: " + groupName + " for rosterID: " + rosterID, e); } } } }