/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.forest.ape.server.persistence;
import java.io.File;
import java.io.IOException;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import org.apache.jute.Record;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.forest.ape.exception.ApeException;
import com.forest.ape.exception.ApeException.Code;
import com.forest.ape.server.DataTree;
import com.forest.ape.server.DataTree.ProcessTxnResult;
import com.forest.ape.server.ZooDefs.OpCode;
import com.forest.ape.server.persistence.TxnLog.TxnIterator;
import com.forest.ape.txn.CreateSessionTxn;
import com.forest.ape.txn.CreateTxn;
import com.forest.ape.txn.TxnHeader;
/**
* This is a helper class above the implementations of txnlog and snapshot
* classes
*/
public class FileTxnSnapLog {
// the direcotry containing the
// the transaction logs
File dataDir;
// the directory containing the
// the snapshot directory
File snapDir;
TxnLog txnLog;
SnapShot snapLog;
public final static int VERSION = 2;
public final static String version = "version-";
private static final Logger LOG = LoggerFactory.getLogger(FileTxnSnapLog.class);
/**
* This listener helps the external apis calling restore to gather
* information while the data is being restored.
*/
public interface PlayBackListener {
void onTxnLoaded(TxnHeader hdr, Record rec);
}
/**
* the constructor which takes the datadir and snapdir.
*
* @param dataDir
* the trasaction directory
* @param snapDir
* the snapshot directory
*/
public FileTxnSnapLog(File dataDir, File snapDir) throws IOException {
this.dataDir = new File(dataDir, version + VERSION);
this.snapDir = new File(snapDir, version + VERSION);
if (!this.dataDir.exists()) {
if (!this.dataDir.mkdirs()) {
throw new IOException("Unable to create data directory " + this.dataDir);
}
}
if (!this.snapDir.exists()) {
if (!this.snapDir.mkdirs()) {
throw new IOException("Unable to create snap directory " + this.snapDir);
}
}
txnLog = new FileTxnLog(this.dataDir);
snapLog = new FileSnap(this.snapDir);
}
/**
* get the datadir used by this filetxn snap log
*
* @return the data dir
*/
public File getDataDir() {
return this.dataDir;
}
/**
* get the snap dir used by this filetxn snap log
*
* @return the snap dir
*/
public File getSnapDir() {
return this.snapDir;
}
/**
* this function restores the server database after reading from the
* snapshots and transaction logs
*
* @param dt the datatree to be restored
* @param sessions the sessions to be restored
* @param listener the playback listener to run on the database restoration
* @return the highest zxid restored
* @throws IOException
*/
public long restore(DataTree dt, Map<Long, Integer> sessions/*, PlayBackListener listener*/)
throws IOException {
//��ȡ�������� ���з����л� ����ָ��������
snapLog.deserialize(dt, sessions);
FileTxnLog txnLog = new FileTxnLog(dataDir);
//�ӵ�ǰ���µ���һ��zxid��ʼ��ȡ������ʼ����û��snapshot�IJ���
TxnIterator itr = txnLog.read(dt.lastProcessedZxid + 1);
long highestZxid = dt.lastProcessedZxid;
TxnHeader hdr;
while (true) {
// iterator points to
// the first valid txn when initialized
hdr = itr.getHeader();
if (hdr == null) {
// empty logs
return dt.lastProcessedZxid;
}
if (hdr.getZxid() < highestZxid && highestZxid != 0) {
LOG.error(highestZxid + "(higestZxid) > " + hdr.getZxid() + "(next log) for type "
+ hdr.getType());
} else {
highestZxid = hdr.getZxid();
}
try {
processTransaction(hdr, dt, sessions, itr.getTxn());
} catch (ApeException.NoNodeException e) {
throw new IOException("Failed to process transaction type: " + hdr.getType()
+ " error: " + e.getMessage());
}
//listener.onTxnLoaded(hdr, itr.getTxn());
if (!itr.next())
break;
}
return highestZxid;
}
/**
* process the transaction on the datatree
*
* @param hdr
* the hdr of the transaction
* @param dt
* the datatree to apply transaction to
* @param sessions
* the sessions to be restored
* @param txn
* the transaction to be applied
*/
public void processTransaction(TxnHeader hdr, DataTree dt, Map<Long, Integer> sessions,
Record txn) throws ApeException.NoNodeException {
ProcessTxnResult rc;
switch (hdr.getType()) {
case OpCode.createSession:
sessions.put(hdr.getClientId(), ((CreateSessionTxn) txn).getTimeOut());
if (LOG.isTraceEnabled()) {
LOG.trace("playLog --- create session in log: " + Long.toHexString(hdr.getClientId())
+ " with timeout: " + ((CreateSessionTxn) txn).getTimeOut());
}
// give dataTree a chance to sync its lastProcessedZxid
rc = dt.processTxn(hdr, txn);
break;
case OpCode.closeSession:
sessions.remove(hdr.getClientId());
if (LOG.isTraceEnabled()) {
LOG.trace("playLog --- close session in log: " + Long.toHexString(hdr.getClientId()));
}
rc = dt.processTxn(hdr, txn);
break;
default:
rc = dt.processTxn(hdr, txn);
}
/**
* Snapshots are taken lazily. It can happen that the child znodes of a
* parent are created after the parent is serialized. Therefore, while
* replaying logs during restore, a create might fail because the node
* was already created.
*
* After seeing this failure, we should increment the cversion of the
* parent znode since the parent was serialized before its children.
*
* Note, such failures on DT should be seen only during restore.
*/
if (hdr.getType() == OpCode.create && rc.err == Code.NODEEXISTS.intValue()) {
LOG.debug("Adjusting parent cversion for Txn: " + hdr.getType() + " path:" + rc.path
+ " err: " + rc.err);
int lastSlash = rc.path.lastIndexOf('/');
String parentName = rc.path.substring(0, lastSlash);
CreateTxn cTxn = (CreateTxn) txn;
try {
dt.setCversionPzxid(parentName, cTxn.getParentCVersion(), hdr.getZxid());
} catch (ApeException.NoNodeException e) {
LOG.error("Failed to set parent cversion for: " + parentName, e);
throw e;
}
} else if (rc.err != Code.OK.intValue()) {
LOG.debug("Ignoring processTxn failure hdr: " + hdr.getType() + " : error: " + rc.err);
}
}
/**
* the last logged zxid on the transaction logs
*
* @return the last logged zxid
*/
public long getLastLoggedZxid() {
FileTxnLog txnLog = new FileTxnLog(dataDir);
return txnLog.getLastLoggedZxid();
}
/**
* save the datatree and the sessions into a snapshot
*
* @param dataTree
* the datatree to be serialized onto disk
* @param sessionsWithTimeouts
* the sesssion timeouts to be serialized onto disk
* @throws IOException
*/
public void save(DataTree dataTree, ConcurrentHashMap<Long, Integer> sessionsWithTimeouts)
throws IOException {
long lastZxid = dataTree.lastProcessedZxid;
LOG.info("Snapshotting: " + Long.toHexString(lastZxid));
File snapshot = new File(snapDir, Util.makeSnapshotName(lastZxid));
snapLog.serialize(dataTree, sessionsWithTimeouts, snapshot);
}
/**
* truncate the transaction logs the zxid specified
*
* @param zxid
* the zxid to truncate the logs to
* @return true if able to truncate the log, false if not
* @throws IOException
*/
public boolean truncateLog(long zxid) throws IOException {
FileTxnLog txnLog = new FileTxnLog(dataDir);
return txnLog.truncate(zxid);
}
/**
* the most recent snapshot in the snapshot directory
*
* @return the file that contains the most recent snapshot
* @throws IOException
*/
public File findMostRecentSnapshot() throws IOException {
FileSnap snaplog = new FileSnap(snapDir);
return snaplog.findMostRecentSnapshot();
}
/**
* the n most recent snapshots
*
* @param n
* the number of recent snapshots
* @return the list of n most recent snapshots, with the most recent in
* front
* @throws IOException
*/
public List<File> findNRecentSnapshots(int n) throws IOException {
FileSnap snaplog = new FileSnap(snapDir);
return snaplog.findNRecentSnapshots(n);
}
/**
* get the snapshot logs that are greater than the given zxid
*
* @param zxid
* the zxid that contains logs greater than zxid
* @return
*/
public File[] getSnapshotLogs(long zxid) {
return FileTxnLog.getLogFiles(dataDir.listFiles(), zxid);
}
/**
* append the request to the transaction logs
*
* @param si
* the request to be appended returns true iff something
* appended, otw false
* @throws IOException
*/
public boolean append(Request si) throws IOException {
return txnLog.append(si.hdr, si.txn);
}
/**
* commit the transaction of logs
*
* @throws IOException
*/
public void commit() throws IOException {
txnLog.commit();
}
/**
* roll the transaction logs
*
* @throws IOException
*/
public void rollLog() throws IOException {
txnLog.rollLog();
}
/**
* close the transaction log files
*
* @throws IOException
*/
public void close() throws IOException {
txnLog.close();
snapLog.close();
}
}