Silverlight получает ответ от обработчика - PullRequest
1 голос
/ 16 апреля 2011

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

private void UploadFileEx()
{
    Status = FileUploadStatus.Uploading;
    var temp = FileLength - BytesUploaded;

    var ub = new UriBuilder(_urlServer);
    var complete = temp <= ChunkSize;

    ub.Query = string.Format("{3}locationCode={4}&filename={0}&StartByte={1}&Complete={2}", RockDocumentName, BytesUploaded, complete,
                                string.IsNullOrEmpty(ub.Query) ? string.Empty : string.Format("{0}&", ub.Query.Remove(0, 1)), _locationCode);

    var webrequest = (HttpWebRequest) WebRequest.Create(ub.Uri);
    webrequest.Method = "POST";

    // Begins an asynchronous request for a Stream object to use to write data.
    // The asynchronous callback method uses the EndGetRequestStream method to return the actual stream.
    // This means
    webrequest.BeginGetRequestStream(WriteCallback, webrequest);
}

private void WriteCallback(IAsyncResult asynchronousResult)
{
    var webrequest = (HttpWebRequest) asynchronousResult.AsyncState;
    // End the operation.
    // Here we obtain the stream to write the request
    var requestStream = webrequest.EndGetRequestStream(asynchronousResult);

    var buffer = new Byte[4096];
    int bytesRead;
    var tempTotal = 0;

    Stream fileStream = File.OpenRead();

    fileStream.Position = BytesUploaded;
    while ((bytesRead = fileStream.Read(buffer, 0, buffer.Length)) != 0 && tempTotal + bytesRead < ChunkSize)
    {
        requestStream.Write(buffer, 0, bytesRead);
        requestStream.Flush();
        BytesUploaded += bytesRead;
        tempTotal += bytesRead;

        if (UploadProgressChanged != null)
        {
            var percent = (int) ((BytesUploaded/(double) FileLength)*100);
            var args = new UploadProgressChangedEventArgs(percent, bytesRead, BytesUploaded, FileLength,
                                                            RockDocumentName);
            Dispatcher.BeginInvoke(() => UploadProgressChanged(this, args));
        }
    }

    fileStream.Close();
    requestStream.Close();

    // Whenever the operation is ready call the ReadCallback method (when this method
    // is called we know that the chunk of file has arrived successfully to the server,
    // it's time to process the next chunk).
    webrequest.BeginGetResponse(ReadCallback, webrequest);
}

Я хотел бымой обработчик для возврата моего имени документа (я создаю новое имя файла для каждого документа в обработчике, потому что у нас есть используемый стандарт именования документов).Итак, я закодировал свой AsyncCallback, как я ожидал, что AsyncState вернет заголовок, который, как я думал, я установил в обработчике:

private void ReadCallback(IAsyncResult asynchronousResult)
{
    var documentName = ((System.Net.BrowserHttpWebRequest)asynchronousResult.AsyncState).Headers["documentname"].ToString();

    if (BytesUploaded < FileLength)
        UploadFileEx();
    else
    {
        Status = FileUploadStatus.Complete;
    }
}

Вот мой обработчик:

public void ProcessRequest(HttpContext context) {
      Context = context;
      UploadFile();
      //context.Request.Headers.Add("documentname", "testfilename.pdf");
      context.Response.Headers.Add("documentname", "testfilename.pdf"); 
}

Thisне работает, как я думал.Что я делаю неправильно?Я неправильно установил заголовок в моем обработчике?Есть ли другой способ для моего обработчика общаться с клиентским приложением Silverlight?

Любая помощь будет оценена.Заранее спасибо!

1 Ответ

1 голос
/ 18 апреля 2011

Вам может повезти больше, если ваш обработчик отправляет обратно имя документа как часть тела ответа, а не как заголовок.

Попробуйте этот код и посмотрите, что вы получите:

private void ReadCallback(IAsyncResult asynchronousResult)
{
    var webRequest = (HttpWebRequest) asynchronousResult.AsyncState;
    var webResponse = (HttpWebResponse) webRequest.EndGetResponse(asynchronousResult);
    var reader = new StreamReader(webResponse.GetResponseStream());
    var documentName = reader.ReadToEnd();
    reader.Close();


    if (BytesUploaded < FileLength)
        UploadFileEx();
    else
    {
        DocumentName = documentName;
        Status = FileUploadStatus.Complete;
    }
}

И в ProcessRequest:

Context.Response.Write(“Document Name”);
...