/** * '$Author$' * '$Date$' * '$Revision$' * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ package edu.ucsb.nceas.metacat; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import java.util.Vector; import org.apache.log4j.Logger; import edu.ucsb.nceas.metacat.database.DBConnection; import edu.ucsb.nceas.metacat.database.DBConnectionPool; import edu.ucsb.nceas.metacat.properties.PropertyService; import edu.ucsb.nceas.metacat.util.DocumentUtil; import edu.ucsb.nceas.metacat.util.MetacatUtil; import edu.ucsb.nceas.utilities.PropertyNotFoundException; import edu.ucsb.nceas.utilities.triple.Triple; import edu.ucsb.nceas.utilities.triple.TripleCollection; public class RelationHandler //implements Runnable { private DBConnection connection = null; private String docid = null; private String docType = null; private static Logger logMetacat = Logger.getLogger(RelationHandler.class); TripleCollection tripleForPackage = null; /** * Constructor for this class. finds all of the relations to a single xml * document and writes them to the database. This includes transitive * relations. * @param docid the ID of the XML document to index. * @param doctype the doctype of this document * @param conn the db connection * @param list the triple list */ public RelationHandler(String docid, String doctype, DBConnection conn, TripleCollection list) throws McdbException, SQLException, AccessionNumberException { this.connection = conn; this.docid = docid; this.docType = doctype; tripleForPackage = list; createRelations(); } /** * insert the relations specified in the triples into xml_relation table */ private void createRelations() throws McdbException, SQLException, AccessionNumberException { String packagetype = docType; String subject = null; String subjectParentId = null; String subDoctype = null; String relationship = null; String relationshipParentId = null; String object = null; String objDoctype = null; PreparedStatement tstmt = null; // to insert each relation into xml_relation logMetacat.info("Running relation handler!"); // first delete the relations for this package document if any deleteRelations(docid); //get the vetor of triples Vector tripleList= new Vector(); //get vector form tripleCollection if (tripleForPackage != null) { tripleList =tripleForPackage.getCollection(); } if (tripleList != null && tripleList.size()>0) { tstmt = connection.prepareStatement("INSERT INTO xml_relation (" + "docid,packagetype,subject,subdoctype," + "relationship, object, objdoctype) " + "VALUES (?, ?, ?, ?, ?, ?, ?)"); // go through tripe list for (int i= 0; i<tripleList.size(); i++) { //increase usage count connection.increaseUsageCount(1); // Get the triple Triple triple = (Triple)tripleList.elementAt(i); logMetacat.info("Info from triple: "); logMetacat.info("subject from triple:"+triple.getSubject()); logMetacat.info("relationship from triple:"+triple.getRelationship()); logMetacat.info("object from triple: "+triple.getObject()); //subject = (new DocumentIdentifier(triple.getSubject())).getIdentifier(); subject = DocumentUtil.getDocIdFromString(triple.getSubject()); relationship = triple.getRelationship(); //object = (new DocumentIdentifier(triple.getObject())).getIdentifier(); object = DocumentUtil.getDocIdFromString(triple.getObject()); if (subject != null && relationship != null && object != null) { //put the new relation into xml_relation logMetacat.info("Insert into xml_relation table"); tstmt.setString(1, docid); logMetacat.info("Insert docid into xml_relation table" + docid); tstmt.setString(2, packagetype); tstmt.setString(3, subject); logMetacat.info("Insert subject into xml_relation table" + subject); tstmt.setString(4, subDoctype); tstmt.setString(5, relationship); logMetacat.info("Insert relationship into xml_relation table" + relationship); tstmt.setString(6, object); logMetacat.info("Insert object into xml_relation table" + object); tstmt.setString(7, objDoctype); tstmt.execute(); }//if }//for }//if if ( tstmt != null ) { tstmt.close(); } } /** * Deletes all of the relations with a docid of 'docid'. * @param docid the docid of the package which relations to delete. */ public void deleteRelations(String docid) throws SQLException { try { PreparedStatement pstmt = connection.prepareStatement( "DELETE FROM xml_relation " + "WHERE docid = ?"); pstmt.setString(1, docid); //increase usage count connection.increaseUsageCount(1); pstmt.execute(); pstmt.close(); } catch(SQLException e) { logMetacat.error("error in RelationHandler.deleteRelations(): " + e.getMessage()); throw e; } } /** * Get the access file id for a package * @param docid the document identifier of the package * @return the document identifier of the access file for that package */ public static String getAccessFileIDWithRevision(String docid) throws SQLException { String aclid = null; int rev; PreparedStatement pstmt = null; DBConnection dbConn = null; int serialNumber = -1; StringBuffer sql = new StringBuffer(); sql .append("SELECT docid, rev FROM xml_documents WHERE docid in (SELECT subject "); sql.append("FROM xml_relation WHERE docid = ? "); sql.append(" AND ("); Vector accessdoctypes; try { accessdoctypes = MetacatUtil.getOptionList(PropertyService .getProperty("xml.accessdoctype")); } catch (PropertyNotFoundException pnfe) { throw new SQLException("Could not find access doctype: " + pnfe.getMessage()); } for (int i = 0; i < accessdoctypes.size(); i++) { String atype = (String) accessdoctypes.elementAt(i); sql.append("doctype='").append(atype).append("'"); if (i < accessdoctypes.size() - 1) { sql.append(" OR "); } } sql.append("))"); //System.out.println("new sql script: " + sql.toString()); try { dbConn = DBConnectionPool.getDBConnection("RelationHandler.getAccessFileID"); serialNumber = dbConn.getCheckOutSerialNumber(); pstmt = dbConn.prepareStatement(sql.toString()); pstmt.setString(1, docid); pstmt.execute(); ResultSet rs = pstmt.getResultSet(); boolean hasRow = rs.next(); if (hasRow) { aclid = rs.getString(1); rev = rs.getInt(2); String sep = "."; try { sep = PropertyService.getProperty("document.accNumSeparator"); } catch (PropertyNotFoundException pnfe) { logMetacat.error("Could not find account separator. Setting to '.': " + pnfe.getMessage()); } aclid = sep + rev; } pstmt.close(); }//try finally { try { pstmt.close(); }//try finally { DBConnectionPool.returnDBConnection(dbConn, serialNumber); }//finally }//finally logMetacat.info("The access docid get from xml_relation is: " + aclid); return aclid; } }