/*
* Copyright (C) 2012 The CyanogenMod Project
*
* Licensed 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.cyanogenmod.filemanager.commands.shell;
import com.cyanogenmod.filemanager.commands.AsyncResultListener;
import com.cyanogenmod.filemanager.commands.ChecksumExecutable;
import com.cyanogenmod.filemanager.commands.SIGNAL;
import com.cyanogenmod.filemanager.console.CommandNotFoundException;
import com.cyanogenmod.filemanager.console.ExecutionException;
import com.cyanogenmod.filemanager.console.InsufficientPermissionsException;
import java.io.File;
/**
* A class for calculate MD5 and SHA-1 checksums of a file system object.
*
* Partial results are returned in order (MD5 -> SHA1)
*
* {@link "http://unixhelp.ed.ac.uk/CGI/man-cgi?md5sum"}
* {@link "http://unixhelp.ed.ac.uk/CGI/man-cgi?sha1sum"}
* @see com.cyanogenmod.filemanager.commands.ChecksumExecutable.CHECKSUMS
*/
public class ChecksumCommand extends AsyncResultProgram implements ChecksumExecutable {
private static final String ID = "checksum"; //$NON-NLS-1$
private final String mName;
private final String[] mChecksums;
private int mChecksumsCounter;
private String mPartial;
/**
* Constructor of ChecksumCommand.
*
* @param src The source file
* @param asyncResultListener The partial result listener
* @throws InvalidCommandDefinitionException If the command has an invalid definition
*/
public ChecksumCommand(String src, AsyncResultListener asyncResultListener)
throws InvalidCommandDefinitionException {
super(ID, asyncResultListener, src);
this.mChecksums = new String[]{null, null};
this.mName = new File(src).getName();
}
/**
* {@inheritDoc}
*/
@Override
public void onStartParsePartialResult() {
this.mChecksums[0] = null;
this.mChecksums[1] = null;
this.mChecksumsCounter = 0;
this.mPartial = ""; //$NON-NLS-1$
}
/**
* {@inheritDoc}
*/
@Override
public void onEndParsePartialResult(boolean cancelled) {
// Send the last partial data
if (this.mPartial != null && this.mPartial.length() > 0) {
if (getAsyncResultListener() != null) {
String data = processPartialResult(this.mPartial);
if (data != null) {
getAsyncResultListener().onPartialResult(data);
}
}
}
this.mPartial = ""; //$NON-NLS-1$
}
/**
* {@inheritDoc}
*/
@Override
public void onParsePartialResult(final byte[] in) {
String partialIn = new String(in);
if (partialIn == null || partialIn.length() == 0) return;
boolean endsWithNewLine = partialIn.endsWith("\n"); //$NON-NLS-1$
String[] lines = partialIn.split("\n"); //$NON-NLS-1$
// Append the pending data to the first line
lines[0] = this.mPartial + lines[0];
// Return all the lines, except the last
for (int i = 0; i < lines.length - 1; i++) {
if (getAsyncResultListener() != null) {
String data = processPartialResult(lines[i]);
if (data != null) {
getAsyncResultListener().onPartialResult(data);
}
}
}
// Return the last line?
if (endsWithNewLine) {
if (getAsyncResultListener() != null) {
String data = processPartialResult(lines[lines.length - 1]);
if (data != null) {
getAsyncResultListener().onPartialResult(data);
}
}
this.mPartial = ""; //$NON-NLS-1$
} else {
// Save the partial for next calls
this.mPartial = lines[lines.length - 1];
}
}
/**
* {@inheritDoc}
*/
@Override
public void onParseErrorPartialResult(byte[] partialErr) {/**NON BLOCK**/}
/**
* {@inheritDoc}
*/
@Override
public SIGNAL onRequestEnd() {
try {
if (this.getProgramListener().getOutputStream() != null) {
this.getProgramListener().getOutputStream().flush();
}
} catch (Exception ex) {/**NON BLOCK**/}
try {
Thread.yield();
} catch (Exception ex) {/**NON BLOCK**/}
return null;
}
/**
* {@inheritDoc}
*/
@Override
public String[] getResult() {
return this.mChecksums;
}
/**
* {@inheritDoc}
*/
@Override
public String getChecksum(CHECKSUMS checksum) {
return getResult()[checksum.ordinal()];
}
/**
* Method that processes a line to determine if it's a valid partial result
*
* @param line The line to process
* @return String The processed line
*/
private String processPartialResult(String line) {
// MD5 and SHA-1 return both the digest and the name of the file
// 4c044b884cf2ff3839713da0e81dced19f099b09 boot.zip
int pos = line.indexOf(" "); //$NON-NLS-1$
if (line.endsWith(this.mName) && pos != -1) {
String digest = line.substring(0, pos).trim();
if (this.mChecksumsCounter < this.mChecksums.length) {
this.mChecksums[this.mChecksumsCounter] = digest;
}
this.mChecksumsCounter++;
return digest;
}
return null;
}
/**
* {@inheritDoc}
*/
@Override
public void checkExitCode(int exitCode)
throws InsufficientPermissionsException, CommandNotFoundException, ExecutionException {
//Ignore exit code 143 (cancelled)
//Ignore exit code 137 (kill -9)
if (exitCode != 0 && exitCode != 143 && exitCode != 137) {
throw new ExecutionException(
"exitcode != 0 && != 143 && != 137"); //$NON-NLS-1$
}
}
}