WP7 및 F #에서 비동기 POST가 실패 함


87

let! read = from.AsyncRead bufF #에서 수행 하면 TCP 소켓이 죽을 때까지 차단되고 반환되지 않습니다. 왜? 그리고 어떻게 고치나요?

코드 :

module StreamUtil

open System.IO

/// copy from 'from' stream to 'toStream'
let (|>>) (from : Stream) (toStream : Stream) =
  let buf = Array.zeroCreate<byte> 1024
  let rec doBlock () =
    async {
      let! read = from.AsyncRead buf
      if read <= 0 then
        toStream.Flush()
        return ()
      else
        do! toStream.AsyncWrite(buf, 0, read)
        return! doBlock () }
  doBlock ()

이 코드에서 호출됩니다.

use fs = new FileStream(targPath, FileMode.CreateNew, FileAccess.ReadWrite)
do! req.InputStream |>> fs

Windows Phone 7.1 에뮬레이터에서 다음 코드를 사용하여 HTTP를 통해 요청했습니다.

public void Send()
{
    var b = new UriBuilder(_imageService.BaseUrl) {Path = "/images"};

    var req = WebRequest.CreateHttp(b.Uri);
    req.ContentType = "image/jpeg";
    req.Method = "POST";
    var imgLen = SelectedImage.ImageStream.Length;
    req.Headers[HttpRequestHeader.ContentLength] = imgLen.ToString(CultureInfo.InvariantCulture);
    req.Accept = "application/json";
    req.BeginGetRequestStream(RequestReady, new ReqState(req, imgLen));
}

void RequestReady(IAsyncResult ar)
{
    var state = (ReqState)ar.AsyncState;
    var req = state.Request;

    var reqStream = req.EndGetRequestStream(ar);

    SmartDispatcher.BeginInvoke(() =>
        {
            using (var sw = new StreamWriter(reqStream))
            using (var br = new BinaryReader(SelectedVoucher.ImageStream))
            {
                var readBytes = br.ReadBytes(state.ImgLen);

                // tried both 2
                sw.Write(readBytes);
                //sw.Write(Convert.ToBase64String(readBytes));
                sw.Flush();
                sw.Close();
            }
            req.BeginGetResponse(ResponseReady, req);
        });
}

// WHY IS IT YOU ARE NOT CALLED???
void ResponseReady(IAsyncResult ar)
{
    try
    {
        var request = (HttpWebRequest)ar.AsyncState;
        var response = request.EndGetResponse(ar);

        SmartDispatcher.BeginInvoke(() =>
            {
                var rdr = new StreamReader(response.GetResponseStream());
                var msg = rdr.ReadToEnd();

                var imageLocation = response.Headers["Location"];

                Debug.WriteLine(msg);
                Debug.WriteLine(imageLocation);
            });
    }
    catch (WebException ex)
    {
        Debug.WriteLine(ex.ToString());
    }
    catch (Exception ex)
    {
        Debug.WriteLine(ex.ToString());
    }
}

실패했습니다. ResponseReady콜백에 도달되지 않습니다.

한편이 코드는 훌륭하게 작동합니다.

open System
open System.Net.Http // WebAPI nuget

let sync aw = Async.RunSynchronously aw

let postC<'a> (c : HttpClient) (r : Uri) (cont : HttpContent) =
  let response = sync <| Async.AwaitTask( c.PostAsync(r, cont) )
  let struc:'a = sync <| deserialize<'a> response
  response, struc

let withContent<'a> (fVerb : (HttpClient -> Uri -> HttpContent -> _ * 'a))=
  let c = new HttpClient()
  fVerb c

[<Test>]
let ``POST /images 201 + Location header`` () =
  let post = withContent<MyImage> postC
  let bytes = IO.File.ReadAllBytes("sample.jpg")
  let hash = SHA1.Create().ComputeHash(bytes) |> Convert.ToBase64String
  let pic = new ByteArrayContent(bytes)
  pic.Headers.Add("Content-Type", "image/jpeg")
  pic.Headers.Add("X-SHA1-Hash", hash)
  let resp, ri = (resource "/images", pic) ||> post

  resp.StatusCode =? Code.Created
  ri.sha1 =? hash
  mustHaveHeaders resp

Fiddler2가 WP7과 함께 작동하도록 할 수 없습니다.

편집 : 야크에 오신 것을 환영합니다. 나는 더 푸른 목초지로 옮겼습니다.)


9
If from.AsyncRead블록은 원격 서버가 바이트를 보내지 않음을 의미합니다.
qehgt

나는 스트림이 제대로 닫히지 않는다는 문제에서 벗어 났지만 여전히 수신 측과 WP에서 40 바이트 크기의 파일을 얻을 수 있습니다. , 그래서 디버그하는 것은 정말 고통 스럽습니다. 내가 도착하면 완전한 솔루션을 게시 할 것입니다.
Henrik

나는이 질문을 잊지 않았다. 지금해야 할 일이 많으며 곧 수정 사항을 게시 할 것입니다.
Henrik

5
피들러가 모바일 장치에서 작동하도록하려는 경우 유용 할 수 있습니다. diaryofaninja.com/blog/2010/11/09/…
Alpha

2
TCP 소켓에서 무한한 양의 바이트를 요청할 때 예상되는 동작이 아닙니까?
jyoung

답변:


1

BufferStream INput 출력을 보내고 사용하기 전에 바이트를 넣어야합니다.

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.