/* * P4.Net * Copyright (c) 2007 Shawn Hladky Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ using System; using System.Configuration; using System.Web; using System.Web.Security; using Microsoft.Win32; using System.IO; using P4API; using System.Text; using System.Collections.Generic; namespace P4HttpHandler { public class P4HttpHandler: IHttpHandler { private static Dictionary<string, string> _mimeTypeCache = new Dictionary<string, string>(); public bool IsReusable { get { return true; } } public void ProcessRequest(HttpContext context) { // Don't allow this response to be cached by the browser. context.Response.Cache.SetCacheability(HttpCacheability.NoCache); context.Response.Cache.SetNoStore(); context.Response.Cache.SetExpires(DateTime.MinValue); // Create a new p4 connection, and set the appropriate properties P4Connection p4 = new P4Connection(); AppSettingsReader appSettings = new AppSettingsReader(); p4.Port = (string) appSettings.GetValue("P4PORT", typeof(string)); p4.User = (string) appSettings.GetValue("P4USER", typeof(string)); p4.Client = (string) appSettings.GetValue("P4CLIENT", typeof(string)); p4.Password = (string) appSettings.GetValue("P4PASSWD", typeof(string)); try { p4.Connect(); //Figure out the clientPath for the file string clientPath = string.Format("//{0}/{1}", p4.Client, context.Request.AppRelativeCurrentExecutionFilePath.Substring(2)); if (!clientPath.EndsWith("/")) { // We have a path to a file // find the MIME type and set it string ext = Path.GetExtension(clientPath); string mimeType = getMimeType(ext); context.Response.ContentType = mimeType; //stream the results ... will throw an exception if the path isn't found try { p4.PrintStream(context.Response.OutputStream, clientPath); context.Response.OutputStream.Flush(); } catch (P4API.Exceptions.FileNotFound) { context.Response.StatusCode = 404; } } else { // we have a directory... let's look for a default "index" file and redirect // My Rule for a default page is: // :: "index.htm" or "index.html" in the current directory (case insensitive) // // I don't rely on the Perforce server to be case insensitive, so I will run an fstat for // all files in the directory and see if there are any "index.htm*" files P4RecordSet rs = p4.Run("fstat", "-Op", clientPath + "*"); foreach (P4Record r in rs) { if (r["depotFile"].ToLower().EndsWith("index.html") || r["depotFile"].ToLower().EndsWith("index.htm")) { // the -Op switch means client file will be //<clientname>/<clientpath> clientPath = r["clientFile"]; break; } } if (clientPath.EndsWith("/")) { // clientPath not updated, means we can't find a default page // For now, just 404... in the future we could allow directory browsing // (which we be a lot bigger than a sample application ;-) context.Response.StatusCode = 404; } else { // redirect to the index page string redirect = "~" + clientPath.Substring(p4.Client.Length + 2); context.Response.Redirect(redirect, false); } } } catch (Exception e) { // unhandled exception... send a 500 to the browser System.Diagnostics.Trace.WriteLine(e.StackTrace); context.Response.StatusCode = 500; } finally { p4.Disconnect(); context.Response.End(); } } /// <summary> /// Get's the mime type from the registry. This is a DUMB way to do it... why isn't there an API? /// </summary> /// <param name="sExtension"></param> /// <returns></returns> private static string getMimeType(string sExtension) { string extension = sExtension.ToLower(); if (_mimeTypeCache.ContainsKey(extension)) { return _mimeTypeCache[extension]; } RegistryKey key = Registry.ClassesRoot.OpenSubKey(sExtension); if (key != null) { string mimetype = key.GetValue("Content Type") as string; if (mimetype != null) { lock (_mimeTypeCache) { _mimeTypeCache.Add(extension, mimetype); } return mimetype; } } return "application/unknown"; } #region debugHelper private void debugMe(HttpContext context) { string html = @" <!DOCTYPE HTML PUBLIC '-//W3C//DTD HTML 3.2 Final//EN'> <HTML><HEAD><TITLE>Test</TITLE></HEAD> <BODY BGCOLOR='#FFFFFF' TEXT='#000000' LINK='#FF0000' VLINK='#800000' ALINK='#FF00FF' BACKGROUND='?'> <BR>ApplicationPath: {0}</BR> <BR>Path: {1}</BR> <BR>FilePath: {2}</BR> <BR>ContentType: {3}</BR> <BR>AppRelativeCurrentExecutionFilePath: {4}</BR> <BR>Browser: {5}</BR> <BR>PathInfo: {6}</BR> <BR>QueryString: {7}</BR> <BR>RawUrl: {8}</BR> <BR>Url: {9}</BR> <BR>Extension: {10}</BR> <BR>My Mime: {11}</BR> </BODY></HTML> "; string ext = Path.GetExtension(context.Request.AppRelativeCurrentExecutionFilePath); string mimeType = getMimeType(ext); string sHtml = string.Format(html, context.Request.ApplicationPath, context.Request.Path, context.Request.FilePath, context.Request.ContentType, context.Request.AppRelativeCurrentExecutionFilePath, context.Request.Browser, context.Request.PathInfo, context.Request.QueryString, context.Request.RawUrl, context.Request.Url, ext, mimeType ); context.Response.Write(sHtml); } #endregion } }