Распаковать данные в C # - PullRequest
0 голосов
/ 23 февраля 2012

Я новичок в C #, я пишу простой веб-сервис.Он берет zip-файл и распаковывает его в файловой системе.В коде C # это:

[WebMethod]
public String SetZip(string device_id, string file)
{
    if (device_id == null || device_id.Length == 0)
    {
        return "10;no auth data";
    }

    StringBuilder output = new StringBuilder();

    if (direcory == null)
    {
        return output.ToString();
    }

    string dirname = "c:\\temp\\" + direcory + "\\";

    if (!System.IO.Directory.Exists(dirname))
    {
        System.IO.Directory.CreateDirectory(dirname);
    }

    string filename = dirname + "file1.txt";

    string text = UnZipStr(Convert.FromBase64String(file));

    File.WriteAllText(filename, text);

    output.AppendLine("0;done");

    return output.ToString();
}

public static string UnZipStr(byte[] input)
{
    using (MemoryStream memstream = new MemoryStream())
    {
        using (MemoryStream inputStream = new MemoryStream(input))
        {
            using (DeflateStream gzip =
              new DeflateStream(inputStream, CompressionMode.Decompress))
            {
                using (StreamReader reader =
                  new StreamReader(gzip, System.Text.Encoding.UTF8))
                {
                    return reader.ReadToEnd();
                }
            }
        }
    }
}

И отправка почтовых данных из кода Java:

    void callService(byte[] xmlData) {
    SoapObject request = new SoapObject(NAMESPACE, METHOD_NAME);
    request.addProperty("device_id", AGENT);

    Deflater deflater = new Deflater();
    deflater.setInput(xmlData);
    deflater.finish();

     ByteArrayOutputStream baos = new ByteArrayOutputStream();
     byte[] buf = new byte[1024];
     while (!deflater.finished()) {
         int byteCount = deflater.deflate(buf);
         baos.write(buf, 0, byteCount);
     }
     deflater.end();

     byte[] compressedBytes = baos.toByteArray();

    request.addPropertyIfValue("file", org.kobjects.base64.Base64.encode(compressedBytes));...}

В коде C # у меня есть исключение при чтении данных из StreamReader

SoapFault - faultcode: 'soap:Server' faultstring:     'System.Web.Services.Protocols.SoapException: ---> InvalidDataException: Block length does    not correspond to the complement.
System.IO.Compression.Inflater.DecodeUncompressedBlock(Boolean& end_of_block)
System.IO.Compression.Inflater.Decode()
System.IO.Compression.Inflater.Inflate(Byte[] bytes, Int32 offset, Int32 length)
System.IO.Compression.DeflateStream.Read(Byte[] array, Int32 offset, Int32 count)
System.IO.StreamReader.ReadBuffer()
System.IO.StreamReader.ReadToEnd()
Service.UnZipStr(Byte[] input) в c:\inetpub\wwwroot\WebSite\App_Code\Service.cs: at 94
Service.SetZip(String device_id, String file) в c:\inetpub\wwwroot\WebSite\App_Code    \Service.cs: at 73

Что я делаю не так?

Ответы [ 2 ]

0 голосов
/ 23 февраля 2012

Похоже, @ Джони Салонен дал ответ на проблему с дефляцией, поэтому я хочу немного рассказать об архитектуре. Чтобы изолировать проблему, вы должны разделить две проблемы. Во-первых, вам нужно сбросить заархивированный файл. Тогда вам нужно спустить его. Затем сфокусируйтесь на проблемной области отдельно. Вы всегда можете «обсудить» две проблемы позже.

Кстати, полезно иметь сырой zip-файл во многих случаях. Это становится сетью безопасности, когда что-то не работает, как запланировано, особенно. если у вас есть только один шанс захватить файл.

0 голосов
/ 23 февраля 2012

Редактировать: Я копался и есть способ генерировать данные формата DEFLATE из Java, которые распознает C #: вы должны использовать конструктор Deflater(int,boolean). Поэтому измените свой Java-код на:

Deflater deflater = new Deflater(Deflater.DEFAULT_COMPRESSION, true);
deflater.setInput(xmlData);
deflater.finish();

Причина в том, что по умолчанию дефлятор испускает заголовок и контрольную сумму ZLIB, чего C # DeflateStream не ожидает.


Кажется, что Java и C # не согласны с точным разнообразием алгоритма сжатия "deflate", но по моим тестам "gzip" должен работать:

Это сжимает xmlData в Java:

ByteArrayOutputStream baos = new ByteArrayOutputStream()
OutputStream out = new GZIPOutputStream(baos);
out.write(xmlData);
out.close();
byte[] compressedBytes = baos.toByteArray();

Это распаковка input в C #:

using (MemoryStream inputStream = new MemoryStream(input))
{
  using (GZipStream gzip = new GZipStream(inputStream, CompressionMode.Decompress))
  {
    using (StreamReader reader = new StreamReader(gzip, System.Text.Encoding.UTF8))
    {
      return reader.ReadToEnd();
    }
  }
}
...