我正在编写一个用于教育目的的超小型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) + "\r\n\r\n");
output.Write(statusLine, 0, statusLine.Length);
byte[] ContentType =
(new System.Text.ASCIIEncoding()).GetBytes(GetContentType(file) +
"\r\n\r\n");
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编码,因为我想将字符串转换为字节,我只知道使用编码。
答案 0 :(得分:5)
JPG的响应必须只是HTTP标题,然后是JPEG的内容,而不是它周围的任何HTML。
像
这样的东西HTTP/1.0 200 OK
Content-type: image/jpeg
Content-length: XXXXX
RAWJPEGDATA
使用Jpeg中的字节数填写XXXXX,直接输出原始JPEG数据,无需任何编码。
使用Fiddler或Firebug帮助调试 - 它们显示正在发送的确切请求/响应。
答案 1 :(得分:5)
看起来您正在为jpeg文件发送302 Found
状态,这意味着重定向。您需要像对待HTML文件一样发送200 OK
。
答案 2 :(得分:2)
我认为问题出在WriteOutputHelper
和GetContentTypeHelper
方法中。
标题不应该有\r\n\r\n
,一个就足够了,GetContentTypeHelper方法也应该返回一个标题:
Content-type: image/jpeg
不是html <meta>
元素,它适用于(X)HTML内容,而不是HTTP标头。
答案 3 :(得分:0)
您是否考虑过使用cassini,它是utildev项目标题下的开源。你可以肯定写自己的,但你永远不能覆盖你的所有基础。关于你的精简版网络服务器不支持所有mime类型的问题。