我正在编写一个用于教育目的的超小型web服务器。
对于下面的代码,如果我请求一个包含图像的html页面,我将无法在浏览器中看到该图像。我做错了什么?
static void Main(string[] args)
{
TcpListener listener = new TcpListener(9999);
listener.Start();
while (true)
{
TcpClient client = listener.AcceptTcpClient();
string request = GetRequest(client.GetStream(),
client.ReceiveBufferSize);
WriteOutput(request, client.GetStream());
client.Close();
}
}
static void WriteOutput(string request, NetworkStream output)
{
try
{
string[] reqs = request.Split(' ');
WriteOutputHelper(output, reqs[1].Substring(1));
}
catch (Exception)
{
WriteOutputHelper(output, "404.html");
}
}
private static void WriteOutputHelper(NetworkStream output, string file)
{
byte[] statusLine = (new System.Text.ASCIIEncoding()).
GetBytes(GetStatusLine(file) + "rnrn");
output.Write(statusLine, 0, statusLine.Length);
byte[] ContentType =
(new System.Text.ASCIIEncoding()).GetBytes(GetContentType(file) +
"rnrn");
output.Write(ContentType, 0, ContentType.Length);
byte[] response = System.IO.File.ReadAllBytes("C:\" + file);
output.Write(response, 0, response.Length);
output.Flush();
}
static string GetContentType(string fileName)
{
string i = "<META http-equiv="Content-Type" content="";
if ((fileName.IndexOf(".htm") > -1) || (fileName.IndexOf(".html") > -1))
i = i + "text/html";
else if (fileName.IndexOf(".jpg") > -1)
i = i + "image/jpeg";
i = i + ";">";
return i;
}
static string GetStatusLine(string fileName)
{
string i = "HTTP/1.0 ";
if (fileName.IndexOf("404") > -1)
return i + "404 Not Found";
else if (fileName.IndexOf("jpg") > -1)
return i + "302 Found";
return i + "200 OK";
}
static string GetRequest(NetworkStream reqStream,int bufSize)
{
byte[] bytesFrom = new byte[10025];
reqStream.Read(bytesFrom, 0, bufSize);
string request = System.Text.Encoding.ASCII.GetString(bytesFrom);
return request;
}
编辑:
static void imageTest(NetworkStream output)
{
byte[] fileContent = System.IO.File.ReadAllBytes("C:\sachin.jpg");
string statusLine = "HTTP/1.0 200 OK" + System.Environment.NewLine;
string contentType = "Content-type: image/jpeg" + System.Environment.NewLine;
string contentLength = "Content-length: " + fileContent.Length + System.Environment.NewLine;
System.Text.UnicodeEncoding coding = new UnicodeEncoding();
byte[] headers = coding.GetBytes(statusLine + contentType + contentLength);
output.Write(headers, 0, headers.Length);
output.Write(fileContent, 0, fileContent.Length);
output.Flush();
}
对于上面的代码,我在fiddler中得到了这个错误。
服务器未返回格式正确的HTTP标头。HTTP标头应使用CRLFCRLF终止。这些都是用LFLF终止的。
我使用Unicode编码是因为我想将字符串转换为字节,而我只知道使用编码。
JPG的响应必须只是HTTP头,然后是JPEG的内容,而不是它周围的任何HTML。
类似的东西
HTTP/1.0 200 OK
Content-type: image/jpeg
Content-length: XXXXX
RAWJPEGDATA
用Jpeg中的字节数填充XXXXX,只需直接输出原始Jpeg数据,无需任何编码。
使用Fiddler或Firebug来帮助调试——它们显示了发送的确切请求/响应。
看起来您正在发送jpeg文件的302 Found
状态,该状态用于重定向。您需要像发送HTML文件一样发送200 OK
。
我认为问题出在WriteOutputHelper
和GetContentTypeHelper
方法中。
标头不应该有rnrn
,一个就足够了,而且GetContentTypeHelper方法应该返回一个标头,如:
Content-type: image/jpeg
不是用于(X)html内容的html <meta>
元素,而不是HTTP标头。
你有没有想过使用cassini,它是以utildev为项目标题的开源软件。你当然可以写自己的,但你永远不可能涵盖所有的基础。一个关于问题的猜测是,你的lite web服务器不支持所有的mime类型。