228 lines
6.0 KiB
C#
228 lines
6.0 KiB
C#
|
//
|
||
|
// Utilities.cs:
|
||
|
//
|
||
|
// Author:
|
||
|
// Ankit Jain (jankit@novell.com)
|
||
|
//
|
||
|
// Copyright (c) 2009 Novell, Inc (http://www.novell.com)
|
||
|
//
|
||
|
// 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.Collections;
|
||
|
using System.Text;
|
||
|
using System.IO;
|
||
|
using System.Runtime.InteropServices;
|
||
|
|
||
|
namespace Mono.XBuild.Utilities {
|
||
|
internal static class MSBuildUtils {
|
||
|
|
||
|
public readonly static bool RunningOnMac;
|
||
|
public readonly static bool RunningOnWindows;
|
||
|
static Hashtable charsToEscape;
|
||
|
|
||
|
static MSBuildUtils ()
|
||
|
{
|
||
|
RunningOnWindows = Path.DirectorySeparatorChar == '\\';
|
||
|
RunningOnMac = !RunningOnWindows && IsRunningOnMac ();
|
||
|
|
||
|
charsToEscape = new Hashtable ();
|
||
|
|
||
|
charsToEscape.Add ('$', null);
|
||
|
charsToEscape.Add ('%', null);
|
||
|
charsToEscape.Add ('\'', null);
|
||
|
charsToEscape.Add ('(', null);
|
||
|
charsToEscape.Add (')', null);
|
||
|
charsToEscape.Add ('*', null);
|
||
|
charsToEscape.Add (';', null);
|
||
|
charsToEscape.Add ('?', null);
|
||
|
charsToEscape.Add ('@', null);
|
||
|
}
|
||
|
|
||
|
public static string Escape (string unescapedExpression)
|
||
|
{
|
||
|
StringBuilder sb = new StringBuilder ();
|
||
|
|
||
|
foreach (char c in unescapedExpression) {
|
||
|
if (charsToEscape.Contains (c))
|
||
|
sb.AppendFormat ("%{0:x2}", (int) c);
|
||
|
else
|
||
|
sb.Append (c);
|
||
|
}
|
||
|
|
||
|
return sb.ToString ();
|
||
|
}
|
||
|
|
||
|
// FIXME: add tests for this
|
||
|
internal static string Unescape (string escapedExpression)
|
||
|
{
|
||
|
StringBuilder sb = new StringBuilder ();
|
||
|
|
||
|
int i = 0;
|
||
|
while (i < escapedExpression.Length) {
|
||
|
sb.Append (Uri.HexUnescape (escapedExpression, ref i));
|
||
|
}
|
||
|
|
||
|
return sb.ToString ();
|
||
|
}
|
||
|
|
||
|
internal static string UnescapeFromXml (string text)
|
||
|
{
|
||
|
StringBuilder sb = new StringBuilder ();
|
||
|
for (int i = 0; i < text.Length; i++) {
|
||
|
char c1 = text[i];
|
||
|
if (c1 == '&') {
|
||
|
int end = text.IndexOf (';', i);
|
||
|
if (end == -1)
|
||
|
throw new FormatException ("Unterminated XML entity.");
|
||
|
string entity = text.Substring (i+1, end - i - 1);
|
||
|
switch (entity) {
|
||
|
case "lt":
|
||
|
sb.Append ('<');
|
||
|
break;
|
||
|
case "gt":
|
||
|
sb.Append ('>');
|
||
|
break;
|
||
|
case "amp":
|
||
|
sb.Append ('&');
|
||
|
break;
|
||
|
case "apos":
|
||
|
sb.Append ('\'');
|
||
|
break;
|
||
|
case "quot":
|
||
|
sb.Append ('"');
|
||
|
break;
|
||
|
default:
|
||
|
throw new FormatException ("Unrecognized XML entity '&" + entity + ";'.");
|
||
|
}
|
||
|
i = end;
|
||
|
} else
|
||
|
sb.Append (c1);
|
||
|
}
|
||
|
return sb.ToString ();
|
||
|
}
|
||
|
|
||
|
[DllImport ("libc")]
|
||
|
static extern int uname (IntPtr buf);
|
||
|
|
||
|
//From Managed.Windows.Forms/XplatUI
|
||
|
static bool IsRunningOnMac ()
|
||
|
{
|
||
|
IntPtr buf = IntPtr.Zero;
|
||
|
try {
|
||
|
buf = System.Runtime.InteropServices.Marshal.AllocHGlobal (8192);
|
||
|
// This is a hacktastic way of getting sysname from uname ()
|
||
|
if (uname (buf) == 0) {
|
||
|
string os = System.Runtime.InteropServices.Marshal.PtrToStringAnsi (buf);
|
||
|
if (os == "Darwin")
|
||
|
return true;
|
||
|
}
|
||
|
} catch {
|
||
|
} finally {
|
||
|
if (buf != IntPtr.Zero)
|
||
|
System.Runtime.InteropServices.Marshal.FreeHGlobal (buf);
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
internal static string FromMSBuildPath (string relPath)
|
||
|
{
|
||
|
string result = null;
|
||
|
FromMSBuildPath (String.Empty, relPath, out result);
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
internal static bool FromMSBuildPath (string basePath, string relPath, out string resultPath)
|
||
|
{
|
||
|
resultPath = relPath;
|
||
|
|
||
|
if (string.IsNullOrEmpty (relPath))
|
||
|
return false;
|
||
|
|
||
|
string path = relPath;
|
||
|
if (!RunningOnWindows)
|
||
|
path = path.Replace ("\\", "/");
|
||
|
|
||
|
path = Unescape (path);
|
||
|
|
||
|
if (char.IsLetter (path [0]) && path.Length > 1 && path[1] == ':') {
|
||
|
if (RunningOnWindows) {
|
||
|
resultPath = path; // Return the escaped value
|
||
|
return true;
|
||
|
} else
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if (basePath != null)
|
||
|
path = Path.Combine (basePath, path);
|
||
|
|
||
|
if (System.IO.File.Exists (path) || System.IO.Directory.Exists (path)){
|
||
|
resultPath = Path.GetFullPath (path);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if (Path.IsPathRooted (path) && !RunningOnWindows) {
|
||
|
|
||
|
// Windows paths are case-insensitive. When mapping an absolute path
|
||
|
// we can try to find the correct case for the path.
|
||
|
|
||
|
string[] names = path.Substring (1).Split ('/');
|
||
|
string part = "/";
|
||
|
|
||
|
for (int n=0; n<names.Length; n++) {
|
||
|
string[] entries;
|
||
|
|
||
|
if (names [n] == ".."){
|
||
|
if (part == "/")
|
||
|
return false; // Can go further back. It's not an existing file
|
||
|
part = Path.GetFullPath (part + "/..");
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
entries = Directory.GetFileSystemEntries (part);
|
||
|
|
||
|
string fpath = null;
|
||
|
foreach (string e in entries) {
|
||
|
if (string.Compare (Path.GetFileName (e), names[n], true) == 0) {
|
||
|
fpath = e;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (fpath == null) {
|
||
|
// Part of the path does not exist. Can't do any more checking.
|
||
|
part = Path.GetFullPath (part);
|
||
|
for (; n < names.Length; n++)
|
||
|
part += "/" + names[n];
|
||
|
resultPath = part;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
part = fpath;
|
||
|
}
|
||
|
resultPath = Path.GetFullPath (part);
|
||
|
} else {
|
||
|
resultPath = Path.GetFullPath (path);
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|