我必须在客户端(C#)和服务器(PHP)文件结构上使用MD5哈希文件/文件夹。 (服务器土地是PHP,客户端土地是c#。)问题是当它们工作时它们不匹配。任何想法将不胜感激
以下是我的两个算法
C#
using System;
using System.IO;
using System.Security.Cryptography;
using System.Text;
namespace nofolder
{
public class classHasher
{
/**********
* recursive folder MD5 hash of a dir
*/
MD5 hashAlgo = null;
StringBuilder sb;
public classHasher()
{
hashAlgo = new MD5CryptoServiceProvider();
}
public string getHash(String path)
{
// get the file attributes for file or directory
if (File.Exists(path)) return getHashOverFile(path);
if (Directory.Exists(path)) return getHashOverFolder(path);
return "";
}
public string getHashOverFolder(String path)
{
sb = new StringBuilder();
getFolderContents(path);
return sb.ToString().GetHashCode().ToString();
}
public string getHashOverFile(String filename)
{
sb = new StringBuilder();
getFileHash(filename);
return sb.ToString().GetHashCode().ToString();
}
private void getFolderContents(string fold)
{
foreach (var d in Directory.GetDirectories(fold))
{
getFolderContents(d);
}
foreach (var f in Directory.GetFiles(fold))
{
getFileHash(f);
}
}
private void getFileHash(String f)
{
using (FileStream file = new FileStream(f, FileMode.Open, FileAccess.Read))
{
byte[] retVal = hashAlgo.ComputeHash(file);
file.Close();
foreach (var y in retVal)
{
sb.Append(y.ToString());
}
}
}
}
}
PHP
function include__md5_dir($dir){
/**********
* recursive folder MD5 hash of a dir
*/
if (!is_dir($dir)){
return md5_file($dir);
}
$filemd5s = array();
$d = dir($dir);
while (false !== ($entry = $d->read())){
if ($entry != '.' && $entry != '..'){
if (is_dir($dir.'/'.$entry)){
$filemd5s[] = include__md5_dir($dir.'/'.$entry);
}
else{
$filemd5s[] = md5_file($dir.'/'.$entry);
}
}
}
$d->close();
return md5(implode('', $filemd5s));
}
修改
我已决定 c#必须更改。 PHP很好,因为它。 100%工作的第一个代码获得赏金
答案 0 :(得分:1)
您的PHP代码正在汇编十六进制数字(根据md5_file()
文档)
您的C#代码正在组装非0填充的十进制数字
您需要y.ToString("x2")
格式化为十六进制。
另外,return sb.ToString().GetHashCode().ToString();
是非常错误的。不要拨打GetHashCode()
;这不是你想要的。
答案 1 :(得分:0)
我最终解决了这个问题,并为未来的后代提供了答案 - 这个解决方案的关键是照射linux和Windows使用的不同默认目录ORDERING 。这只在linux服务器(Cent OS6.3)和Windows 7 Client上进行了测试。
C#
public class classHasher
{
/**********
* recursive folder MD5 hash of a dir
*/
MD5 hashAlgo = null;
StringBuilder sb;
public classHasher()
{
hashAlgo = new MD5CryptoServiceProvider();
}
public string UltraHasher(String path)
{
/**********
* recursive folder MD5 hash of a dir
*/
if (!Directory.Exists(path))
{
return getHashOverFile(path);
}
List<string> filemd5s = new List<string>();
List<string> dir = new List<string>();
if (Directory.GetDirectories(path) != null) foreach (var d in Directory.GetDirectories(path))
{
dir.Add(d);
}
if (Directory.GetFiles(path) != null) foreach (var f in Directory.GetFiles(path))
{
dir.Add(f);
}
dir.Sort();
foreach (string entry in dir)
{
if (Directory.Exists(entry))
{
string rtn = UltraHasher(entry.ToString());
//Debug.WriteLine(" ULTRRAAHASHER:! " + entry.ToString() + ":" + rtn);
filemd5s.Add(rtn);
}
if (File.Exists(entry))
{
string rtn = getHashOverFile(entry.ToString());
//Debug.WriteLine(" FILEEEEHASHER:! " + entry.ToString() + ":" + rtn);
filemd5s.Add(rtn);
}
}
//Debug.WriteLine(" ULTRRAASUMMMM:! " + String.Join("", filemd5s.ToArray()));
string tosend = CalculateMD5Hash(String.Join("", filemd5s.ToArray()));
//Debug.WriteLine(" YEAHHAHHAHHAH:! " + tosend);
return tosend;
}
public string getHashOverFile(String filename)
{
sb = new StringBuilder();
getFileHash(filename);
return sb.ToString();
}
private void getFileHash(String f)
{
using (FileStream file = new FileStream(f, FileMode.Open, FileAccess.Read))
{
byte[] retVal = hashAlgo.ComputeHash(file);
file.Close();
foreach (var y in retVal)
{
sb.Append(y.ToString("x2"));
}
}
}
public string CalculateMD5Hash(string input)
{
byte[] inputBytes = System.Text.Encoding.ASCII.GetBytes(input);
byte[] hash = hashAlgo.ComputeHash(inputBytes);
StringBuilder sz = new StringBuilder();
for (int i = 0; i < hash.Length; i++)
{
sz.Append(hash[i].ToString("x2"));
}
return sz.ToString();
}
}
PHP
function md5_dir($dir){
/**********
* recursive folder MD5 hash of a dir
*/
if (!is_dir($dir)){
return md5_file($dir);
}
$filemd5s = array();
$bit = array();
$d = scandir($dir);
foreach($d as $entry){
if ($entry != '.' && $entry != '..'){
$bit[] = $entry;
}
}
asort($bit);
foreach($bit as $entry){
if (is_dir($dir.'/'.$entry)){
$sz = md5_dir($dir.'/'.$entry);
//echo "\n ULTRRAAHASHER:! ".$dir.'/'.$entry.":$sz";
$filemd5s[] = $sz;
}
else{
$sz = md5_file($dir.'/'.$entry);
$filemd5s[] = $sz;
//echo "\n FILEEEEHASHER:! ".$dir.'/'.$entry.":$sz";
}
}
//echo "\n ULTRRAASUMMMM:! ".implode('', $filemd5s)."";
//echo "\n YEAHHAHHAHHAH:! ".md5(implode('', $filemd5s))."";
return md5(implode('', $filemd5s));
}
这两个将遍历C #Windows和/或PHP linux文件夹,并返回所有目录的SAME哈希值(递归,因此它包括子目录)在Linuxland内部和Windowsland内部。