2014-08-13 10:39:27 +01:00
|
|
|
//
|
|
|
|
// MacProxy.cs
|
|
|
|
//
|
|
|
|
// Author: Jeffrey Stedfast <jeff@xamarin.com>
|
|
|
|
//
|
|
|
|
// Copyright (c) 2012-2014 Xamarin Inc.
|
|
|
|
//
|
|
|
|
// 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;
|
2016-02-22 11:00:01 -05:00
|
|
|
using System.Net;
|
2014-08-13 10:39:27 +01:00
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
using System.Threading;
|
2017-04-10 11:41:01 +00:00
|
|
|
using ObjCRuntime;
|
2014-08-13 10:39:27 +01:00
|
|
|
|
2016-02-22 11:00:01 -05:00
|
|
|
namespace Mono.Net
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
2017-04-10 11:41:01 +00:00
|
|
|
internal class CFType {
|
|
|
|
[DllImport (CFObject.CoreFoundationLibrary, EntryPoint="CFGetTypeID")]
|
|
|
|
public static extern IntPtr GetTypeID (IntPtr typeRef);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal class CFObject : IDisposable, INativeObject
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
|
|
|
public const string CoreFoundationLibrary = "/System/Library/Frameworks/CoreFoundation.framework/CoreFoundation";
|
|
|
|
const string SystemLibrary = "/usr/lib/libSystem.dylib";
|
|
|
|
|
|
|
|
[DllImport (SystemLibrary)]
|
|
|
|
public static extern IntPtr dlopen (string path, int mode);
|
|
|
|
|
|
|
|
[DllImport (SystemLibrary)]
|
2017-04-10 11:41:01 +00:00
|
|
|
static extern IntPtr dlsym (IntPtr handle, string symbol);
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
[DllImport (SystemLibrary)]
|
|
|
|
public static extern void dlclose (IntPtr handle);
|
|
|
|
|
|
|
|
public static IntPtr GetIndirect (IntPtr handle, string symbol)
|
|
|
|
{
|
|
|
|
return dlsym (handle, symbol);
|
|
|
|
}
|
|
|
|
|
2017-04-10 11:41:01 +00:00
|
|
|
public static CFString GetStringConstant (IntPtr handle, string symbol)
|
|
|
|
{
|
|
|
|
var indirect = dlsym (handle, symbol);
|
|
|
|
if (indirect == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
var actual = Marshal.ReadIntPtr (indirect);
|
|
|
|
if (actual == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
return new CFString (actual, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static IntPtr GetIntPtr (IntPtr handle, string symbol)
|
|
|
|
{
|
|
|
|
var indirect = dlsym (handle, symbol);
|
|
|
|
if (indirect == IntPtr.Zero)
|
|
|
|
return IntPtr.Zero;
|
|
|
|
return Marshal.ReadIntPtr (indirect);
|
|
|
|
}
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
public static IntPtr GetCFObjectHandle (IntPtr handle, string symbol)
|
|
|
|
{
|
|
|
|
var indirect = dlsym (handle, symbol);
|
|
|
|
if (indirect == IntPtr.Zero)
|
|
|
|
return IntPtr.Zero;
|
|
|
|
|
|
|
|
return Marshal.ReadIntPtr (indirect);
|
|
|
|
}
|
|
|
|
|
|
|
|
public CFObject (IntPtr handle, bool own)
|
|
|
|
{
|
|
|
|
Handle = handle;
|
|
|
|
|
|
|
|
if (!own)
|
|
|
|
Retain ();
|
|
|
|
}
|
|
|
|
|
|
|
|
~CFObject ()
|
|
|
|
{
|
|
|
|
Dispose (false);
|
|
|
|
}
|
|
|
|
|
|
|
|
public IntPtr Handle { get; private set; }
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
2017-04-10 11:41:01 +00:00
|
|
|
internal extern static IntPtr CFRetain (IntPtr handle);
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
void Retain ()
|
|
|
|
{
|
|
|
|
CFRetain (Handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
2017-04-10 11:41:01 +00:00
|
|
|
internal extern static void CFRelease (IntPtr handle);
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
void Release ()
|
|
|
|
{
|
|
|
|
CFRelease (Handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected virtual void Dispose (bool disposing)
|
|
|
|
{
|
|
|
|
if (Handle != IntPtr.Zero) {
|
|
|
|
Release ();
|
|
|
|
Handle = IntPtr.Zero;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Dispose ()
|
|
|
|
{
|
|
|
|
Dispose (true);
|
|
|
|
GC.SuppressFinalize (this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
internal class CFArray : CFObject
|
|
|
|
{
|
|
|
|
public CFArray (IntPtr handle, bool own) : base (handle, own) { }
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFArrayCreate (IntPtr allocator, IntPtr values, /* CFIndex */ IntPtr numValues, IntPtr callbacks);
|
|
|
|
static readonly IntPtr kCFTypeArrayCallbacks;
|
|
|
|
|
|
|
|
static CFArray ()
|
|
|
|
{
|
|
|
|
var handle = dlopen (CoreFoundationLibrary, 0);
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
return;
|
|
|
|
|
|
|
|
try {
|
|
|
|
kCFTypeArrayCallbacks = GetIndirect (handle, "kCFTypeArrayCallBacks");
|
|
|
|
} finally {
|
|
|
|
dlclose (handle);
|
|
|
|
}
|
|
|
|
}
|
2017-04-10 11:41:01 +00:00
|
|
|
|
|
|
|
public static CFArray FromNativeObjects (params INativeObject[] values)
|
|
|
|
{
|
|
|
|
return new CFArray (Create (values), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static unsafe IntPtr Create (params IntPtr[] values)
|
|
|
|
{
|
|
|
|
if (values == null)
|
|
|
|
throw new ArgumentNullException ("values");
|
|
|
|
fixed (IntPtr* pv = values) {
|
|
|
|
return CFArrayCreate (IntPtr.Zero, (IntPtr) pv, (IntPtr)values.Length, kCFTypeArrayCallbacks);
|
|
|
|
}
|
|
|
|
}
|
2014-08-13 10:39:27 +01:00
|
|
|
|
2017-04-10 11:41:01 +00:00
|
|
|
internal static unsafe CFArray CreateArray (params IntPtr[] values)
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
|
|
|
if (values == null)
|
|
|
|
throw new ArgumentNullException ("values");
|
|
|
|
|
|
|
|
fixed (IntPtr *pv = values) {
|
|
|
|
IntPtr handle = CFArrayCreate (IntPtr.Zero, (IntPtr) pv, (IntPtr) values.Length, kCFTypeArrayCallbacks);
|
|
|
|
|
|
|
|
return new CFArray (handle, false);
|
|
|
|
}
|
|
|
|
}
|
2017-04-10 11:41:01 +00:00
|
|
|
|
|
|
|
public static CFArray CreateArray (params INativeObject[] values)
|
|
|
|
{
|
|
|
|
return new CFArray (Create (values), true);
|
|
|
|
}
|
2014-08-13 10:39:27 +01:00
|
|
|
|
2017-04-10 11:41:01 +00:00
|
|
|
public static IntPtr Create (params INativeObject[] values)
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
|
|
|
if (values == null)
|
|
|
|
throw new ArgumentNullException ("values");
|
|
|
|
IntPtr[] _values = new IntPtr [values.Length];
|
2017-04-10 11:41:01 +00:00
|
|
|
for (int i = 0; i < _values.Length; ++i)
|
|
|
|
_values [i] = values [i].Handle;
|
2014-08-13 10:39:27 +01:00
|
|
|
return Create (_values);
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static /* CFIndex */ IntPtr CFArrayGetCount (IntPtr handle);
|
|
|
|
|
|
|
|
public int Count {
|
|
|
|
get { return (int) CFArrayGetCount (Handle); }
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFArrayGetValueAtIndex (IntPtr handle, /* CFIndex */ IntPtr index);
|
|
|
|
|
|
|
|
public IntPtr this[int index] {
|
|
|
|
get {
|
|
|
|
return CFArrayGetValueAtIndex (Handle, (IntPtr) index);
|
|
|
|
}
|
|
|
|
}
|
2017-04-10 11:41:01 +00:00
|
|
|
|
|
|
|
static public T [] ArrayFromHandle<T> (IntPtr handle, Func<IntPtr, T> creation) where T : class, INativeObject
|
|
|
|
{
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
var c = CFArrayGetCount (handle);
|
|
|
|
T [] ret = new T [(int)c];
|
|
|
|
|
|
|
|
for (uint i = 0; i < (uint)c; i++) {
|
|
|
|
ret [i] = creation (CFArrayGetValueAtIndex (handle, (IntPtr)i));
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
internal class CFNumber : CFObject
|
|
|
|
{
|
|
|
|
public CFNumber (IntPtr handle, bool own) : base (handle, own) { }
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
[return: MarshalAs (UnmanagedType.I1)]
|
|
|
|
extern static bool CFNumberGetValue (IntPtr handle, /* CFNumberType */ IntPtr type, [MarshalAs (UnmanagedType.I1)] out bool value);
|
|
|
|
|
|
|
|
public static bool AsBool (IntPtr handle)
|
|
|
|
{
|
|
|
|
bool value;
|
|
|
|
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
CFNumberGetValue (handle, (IntPtr) 1, out value);
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static implicit operator bool (CFNumber number)
|
|
|
|
{
|
|
|
|
return AsBool (number.Handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
[return: MarshalAs (UnmanagedType.I1)]
|
|
|
|
extern static bool CFNumberGetValue (IntPtr handle, /* CFNumberType */ IntPtr type, out int value);
|
|
|
|
|
|
|
|
public static int AsInt32 (IntPtr handle)
|
|
|
|
{
|
|
|
|
int value;
|
|
|
|
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
// 9 == kCFNumberIntType == C int
|
|
|
|
CFNumberGetValue (handle, (IntPtr) 9, out value);
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
2017-04-10 11:41:01 +00:00
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFNumberCreate (IntPtr allocator, IntPtr theType, IntPtr valuePtr);
|
|
|
|
|
|
|
|
public static CFNumber FromInt32 (int number)
|
|
|
|
{
|
|
|
|
// 9 == kCFNumberIntType == C int
|
|
|
|
return new CFNumber (CFNumberCreate (IntPtr.Zero, (IntPtr)9, (IntPtr)number), true);
|
|
|
|
}
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
public static implicit operator int (CFNumber number)
|
|
|
|
{
|
|
|
|
return AsInt32 (number.Handle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
internal struct CFRange {
|
|
|
|
public IntPtr Location, Length;
|
|
|
|
|
|
|
|
public CFRange (int loc, int len)
|
|
|
|
{
|
|
|
|
Location = (IntPtr) loc;
|
|
|
|
Length = (IntPtr) len;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-13 10:44:36 +00:00
|
|
|
internal struct CFStreamClientContext {
|
|
|
|
public IntPtr Version;
|
|
|
|
public IntPtr Info;
|
|
|
|
public IntPtr Retain;
|
|
|
|
public IntPtr Release;
|
|
|
|
public IntPtr CopyDescription;
|
|
|
|
}
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
internal class CFString : CFObject
|
|
|
|
{
|
|
|
|
string str;
|
|
|
|
|
|
|
|
public CFString (IntPtr handle, bool own) : base (handle, own) { }
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFStringCreateWithCharacters (IntPtr alloc, IntPtr chars, /* CFIndex */ IntPtr length);
|
|
|
|
|
|
|
|
public static CFString Create (string value)
|
|
|
|
{
|
|
|
|
IntPtr handle;
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
fixed (char *ptr = value) {
|
|
|
|
handle = CFStringCreateWithCharacters (IntPtr.Zero, (IntPtr) ptr, (IntPtr) value.Length);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return new CFString (handle, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static /* CFIndex */ IntPtr CFStringGetLength (IntPtr handle);
|
|
|
|
|
|
|
|
public int Length {
|
|
|
|
get {
|
|
|
|
if (str != null)
|
|
|
|
return str.Length;
|
|
|
|
|
|
|
|
return (int) CFStringGetLength (Handle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFStringGetCharactersPtr (IntPtr handle);
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFStringGetCharacters (IntPtr handle, CFRange range, IntPtr buffer);
|
|
|
|
|
|
|
|
public static string AsString (IntPtr handle)
|
|
|
|
{
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
int len = (int) CFStringGetLength (handle);
|
|
|
|
|
|
|
|
if (len == 0)
|
|
|
|
return string.Empty;
|
|
|
|
|
|
|
|
IntPtr chars = CFStringGetCharactersPtr (handle);
|
|
|
|
IntPtr buffer = IntPtr.Zero;
|
|
|
|
|
|
|
|
if (chars == IntPtr.Zero) {
|
|
|
|
CFRange range = new CFRange (0, len);
|
|
|
|
buffer = Marshal.AllocHGlobal (len * 2);
|
|
|
|
CFStringGetCharacters (handle, range, buffer);
|
|
|
|
chars = buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
string str;
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
str = new string ((char *) chars, 0, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (buffer != IntPtr.Zero)
|
|
|
|
Marshal.FreeHGlobal (buffer);
|
|
|
|
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
public override string ToString ()
|
|
|
|
{
|
|
|
|
if (str == null)
|
|
|
|
str = AsString (Handle);
|
|
|
|
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static implicit operator string (CFString str)
|
|
|
|
{
|
|
|
|
return str.ToString ();
|
|
|
|
}
|
|
|
|
|
|
|
|
public static implicit operator CFString (string str)
|
|
|
|
{
|
|
|
|
return Create (str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-10 11:41:01 +00:00
|
|
|
|
|
|
|
internal class CFData : CFObject
|
|
|
|
{
|
|
|
|
public CFData (IntPtr handle, bool own) : base (handle, own) { }
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static /* CFDataRef */ IntPtr CFDataCreate (/* CFAllocatorRef */ IntPtr allocator, /* UInt8* */ IntPtr bytes, /* CFIndex */ IntPtr length);
|
|
|
|
public unsafe static CFData FromData (byte [] buffer)
|
|
|
|
{
|
|
|
|
fixed (byte* p = buffer)
|
|
|
|
{
|
|
|
|
return FromData ((IntPtr)p, (IntPtr)buffer.Length);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static CFData FromData (IntPtr buffer, IntPtr length)
|
|
|
|
{
|
|
|
|
return new CFData (CFDataCreate (IntPtr.Zero, buffer, length), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public IntPtr Length {
|
|
|
|
get { return CFDataGetLength (Handle); }
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static /* CFIndex */ IntPtr CFDataGetLength (/* CFDataRef */ IntPtr theData);
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static /* UInt8* */ IntPtr CFDataGetBytePtr (/* CFDataRef */ IntPtr theData);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Exposes a read-only pointer to the underlying storage.
|
|
|
|
*/
|
|
|
|
public IntPtr Bytes {
|
|
|
|
get { return CFDataGetBytePtr (Handle); }
|
|
|
|
}
|
|
|
|
|
|
|
|
public byte this [long idx] {
|
|
|
|
get {
|
|
|
|
if (idx < 0 || (ulong) idx > (ulong) Length)
|
|
|
|
throw new ArgumentException ("idx");
|
|
|
|
return Marshal.ReadByte (new IntPtr (Bytes.ToInt64 () + idx));
|
|
|
|
}
|
|
|
|
|
|
|
|
set {
|
|
|
|
throw new NotImplementedException ("NSData arrays can not be modified, use an NSMutableData instead");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
internal class CFDictionary : CFObject
|
|
|
|
{
|
2017-04-10 11:41:01 +00:00
|
|
|
static readonly IntPtr KeyCallbacks;
|
|
|
|
static readonly IntPtr ValueCallbacks;
|
|
|
|
|
|
|
|
static CFDictionary ()
|
|
|
|
{
|
|
|
|
var handle = dlopen (CoreFoundationLibrary, 0);
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
return;
|
|
|
|
|
|
|
|
try {
|
|
|
|
KeyCallbacks = GetIndirect (handle, "kCFTypeDictionaryKeyCallBacks");
|
|
|
|
ValueCallbacks = GetIndirect (handle, "kCFTypeDictionaryValueCallBacks");
|
|
|
|
} finally {
|
|
|
|
dlclose (handle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
public CFDictionary (IntPtr handle, bool own) : base (handle, own) { }
|
|
|
|
|
2017-04-10 11:41:01 +00:00
|
|
|
public static CFDictionary FromObjectAndKey (IntPtr obj, IntPtr key)
|
|
|
|
{
|
|
|
|
return new CFDictionary (CFDictionaryCreate (IntPtr.Zero, new IntPtr[] { key }, new IntPtr [] { obj }, (IntPtr)1, KeyCallbacks, ValueCallbacks), true);
|
|
|
|
}
|
2017-04-19 15:52:01 +00:00
|
|
|
|
|
|
|
public static CFDictionary FromKeysAndObjects (IList<Tuple<IntPtr,IntPtr>> items)
|
|
|
|
{
|
|
|
|
var keys = new IntPtr [items.Count];
|
|
|
|
var values = new IntPtr [items.Count];
|
|
|
|
for (int i = 0; i < items.Count; i++) {
|
|
|
|
keys [i] = items [i].Item1;
|
|
|
|
values [i] = items [i].Item2;
|
|
|
|
}
|
|
|
|
return new CFDictionary (CFDictionaryCreate (IntPtr.Zero, keys, values, (IntPtr)items.Count, KeyCallbacks, ValueCallbacks), true);
|
|
|
|
}
|
|
|
|
|
2017-04-10 11:41:01 +00:00
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFDictionaryCreate (IntPtr allocator, IntPtr[] keys, IntPtr[] vals, IntPtr len, IntPtr keyCallbacks, IntPtr valCallbacks);
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFDictionaryGetValue (IntPtr handle, IntPtr key);
|
|
|
|
|
2017-04-10 11:41:01 +00:00
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFDictionaryCreateCopy (IntPtr allocator, IntPtr handle);
|
|
|
|
|
|
|
|
public CFDictionary Copy ()
|
|
|
|
{
|
|
|
|
return new CFDictionary (CFDictionaryCreateCopy (IntPtr.Zero, Handle), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public CFMutableDictionary MutableCopy ()
|
|
|
|
{
|
|
|
|
return new CFMutableDictionary (CFDictionaryCreateMutableCopy (IntPtr.Zero, IntPtr.Zero, Handle), true);
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFDictionaryCreateMutableCopy (IntPtr allocator, IntPtr capacity, IntPtr theDict);
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
public IntPtr GetValue (IntPtr key)
|
|
|
|
{
|
|
|
|
return CFDictionaryGetValue (Handle, key);
|
|
|
|
}
|
|
|
|
|
|
|
|
public IntPtr this[IntPtr key] {
|
|
|
|
get {
|
|
|
|
return GetValue (key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-04-10 11:41:01 +00:00
|
|
|
|
|
|
|
internal class CFMutableDictionary : CFDictionary
|
|
|
|
{
|
|
|
|
public CFMutableDictionary (IntPtr handle, bool own) : base (handle, own) { }
|
|
|
|
|
|
|
|
public void SetValue (IntPtr key, IntPtr val)
|
|
|
|
{
|
|
|
|
CFDictionarySetValue (Handle, key, val);
|
|
|
|
}
|
|
|
|
|
2017-04-19 15:52:01 +00:00
|
|
|
public static CFMutableDictionary Create ()
|
|
|
|
{
|
|
|
|
var handle = CFDictionaryCreateMutable (IntPtr.Zero, IntPtr.Zero, IntPtr.Zero, IntPtr.Zero);
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
throw new InvalidOperationException ();
|
|
|
|
return new CFMutableDictionary (handle, true);
|
|
|
|
}
|
|
|
|
|
2017-04-10 11:41:01 +00:00
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static void CFDictionarySetValue (IntPtr handle, IntPtr key, IntPtr val);
|
2017-04-19 15:52:01 +00:00
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFDictionaryCreateMutable (IntPtr allocator, IntPtr capacity, IntPtr keyCallback, IntPtr valueCallbacks);
|
|
|
|
|
2017-04-10 11:41:01 +00:00
|
|
|
}
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
internal class CFUrl : CFObject
|
|
|
|
{
|
|
|
|
public CFUrl (IntPtr handle, bool own) : base (handle, own) { }
|
|
|
|
|
|
|
|
[DllImport (CoreFoundationLibrary)]
|
|
|
|
extern static IntPtr CFURLCreateWithString (IntPtr allocator, IntPtr str, IntPtr baseURL);
|
|
|
|
|
|
|
|
public static CFUrl Create (string absolute)
|
|
|
|
{
|
|
|
|
if (string.IsNullOrEmpty (absolute))
|
|
|
|
return null;
|
|
|
|
|
|
|
|
CFString str = CFString.Create (absolute);
|
|
|
|
IntPtr handle = CFURLCreateWithString (IntPtr.Zero, str.Handle, IntPtr.Zero);
|
|
|
|
str.Dispose ();
|
|
|
|
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return new CFUrl (handle, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-13 10:44:36 +00:00
|
|
|
internal class CFRunLoop : CFObject
|
|
|
|
{
|
|
|
|
[DllImport (CFObject.CoreFoundationLibrary)]
|
|
|
|
static extern void CFRunLoopAddSource (IntPtr rl, IntPtr source, IntPtr mode);
|
|
|
|
|
|
|
|
[DllImport (CFObject.CoreFoundationLibrary)]
|
|
|
|
static extern void CFRunLoopRemoveSource (IntPtr rl, IntPtr source, IntPtr mode);
|
|
|
|
|
|
|
|
[DllImport (CFObject.CoreFoundationLibrary)]
|
|
|
|
static extern int CFRunLoopRunInMode (IntPtr mode, double seconds, bool returnAfterSourceHandled);
|
|
|
|
|
|
|
|
[DllImport (CFObject.CoreFoundationLibrary)]
|
|
|
|
static extern IntPtr CFRunLoopGetCurrent ();
|
|
|
|
|
|
|
|
[DllImport (CFObject.CoreFoundationLibrary)]
|
|
|
|
static extern void CFRunLoopStop (IntPtr rl);
|
|
|
|
|
|
|
|
public CFRunLoop (IntPtr handle, bool own): base (handle, own)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
public static CFRunLoop CurrentRunLoop {
|
|
|
|
get { return new CFRunLoop (CFRunLoopGetCurrent (), false); }
|
|
|
|
}
|
|
|
|
|
|
|
|
public void AddSource (IntPtr source, CFString mode)
|
|
|
|
{
|
|
|
|
CFRunLoopAddSource (Handle, source, mode.Handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void RemoveSource (IntPtr source, CFString mode)
|
|
|
|
{
|
|
|
|
CFRunLoopRemoveSource (Handle, source, mode.Handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
public int RunInMode (CFString mode, double seconds, bool returnAfterSourceHandled)
|
|
|
|
{
|
|
|
|
return CFRunLoopRunInMode (mode.Handle, seconds, returnAfterSourceHandled);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Stop ()
|
|
|
|
{
|
|
|
|
CFRunLoopStop (Handle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
internal enum CFProxyType {
|
|
|
|
None,
|
|
|
|
AutoConfigurationUrl,
|
|
|
|
AutoConfigurationJavaScript,
|
|
|
|
FTP,
|
|
|
|
HTTP,
|
|
|
|
HTTPS,
|
|
|
|
SOCKS
|
|
|
|
}
|
|
|
|
|
|
|
|
internal class CFProxy {
|
|
|
|
//static IntPtr kCFProxyAutoConfigurationHTTPResponseKey;
|
|
|
|
static IntPtr kCFProxyAutoConfigurationJavaScriptKey;
|
|
|
|
static IntPtr kCFProxyAutoConfigurationURLKey;
|
|
|
|
static IntPtr kCFProxyHostNameKey;
|
|
|
|
static IntPtr kCFProxyPasswordKey;
|
|
|
|
static IntPtr kCFProxyPortNumberKey;
|
|
|
|
static IntPtr kCFProxyTypeKey;
|
|
|
|
static IntPtr kCFProxyUsernameKey;
|
|
|
|
|
|
|
|
//static IntPtr kCFProxyTypeNone;
|
|
|
|
static IntPtr kCFProxyTypeAutoConfigurationURL;
|
|
|
|
static IntPtr kCFProxyTypeAutoConfigurationJavaScript;
|
|
|
|
static IntPtr kCFProxyTypeFTP;
|
|
|
|
static IntPtr kCFProxyTypeHTTP;
|
|
|
|
static IntPtr kCFProxyTypeHTTPS;
|
|
|
|
static IntPtr kCFProxyTypeSOCKS;
|
|
|
|
|
|
|
|
static CFProxy ()
|
|
|
|
{
|
|
|
|
IntPtr handle = CFObject.dlopen (CFNetwork.CFNetworkLibrary, 0);
|
|
|
|
|
|
|
|
//kCFProxyAutoConfigurationHTTPResponseKey = CFObject.GetCFObjectHandle (handle, "kCFProxyAutoConfigurationHTTPResponseKey");
|
|
|
|
kCFProxyAutoConfigurationJavaScriptKey = CFObject.GetCFObjectHandle (handle, "kCFProxyAutoConfigurationJavaScriptKey");
|
|
|
|
kCFProxyAutoConfigurationURLKey = CFObject.GetCFObjectHandle (handle, "kCFProxyAutoConfigurationURLKey");
|
|
|
|
kCFProxyHostNameKey = CFObject.GetCFObjectHandle (handle, "kCFProxyHostNameKey");
|
|
|
|
kCFProxyPasswordKey = CFObject.GetCFObjectHandle (handle, "kCFProxyPasswordKey");
|
|
|
|
kCFProxyPortNumberKey = CFObject.GetCFObjectHandle (handle, "kCFProxyPortNumberKey");
|
|
|
|
kCFProxyTypeKey = CFObject.GetCFObjectHandle (handle, "kCFProxyTypeKey");
|
|
|
|
kCFProxyUsernameKey = CFObject.GetCFObjectHandle (handle, "kCFProxyUsernameKey");
|
|
|
|
|
|
|
|
//kCFProxyTypeNone = CFObject.GetCFObjectHandle (handle, "kCFProxyTypeNone");
|
|
|
|
kCFProxyTypeAutoConfigurationURL = CFObject.GetCFObjectHandle (handle, "kCFProxyTypeAutoConfigurationURL");
|
|
|
|
kCFProxyTypeAutoConfigurationJavaScript = CFObject.GetCFObjectHandle (handle, "kCFProxyTypeAutoConfigurationJavaScript");
|
|
|
|
kCFProxyTypeFTP = CFObject.GetCFObjectHandle (handle, "kCFProxyTypeFTP");
|
|
|
|
kCFProxyTypeHTTP = CFObject.GetCFObjectHandle (handle, "kCFProxyTypeHTTP");
|
|
|
|
kCFProxyTypeHTTPS = CFObject.GetCFObjectHandle (handle, "kCFProxyTypeHTTPS");
|
|
|
|
kCFProxyTypeSOCKS = CFObject.GetCFObjectHandle (handle, "kCFProxyTypeSOCKS");
|
|
|
|
|
|
|
|
CFObject.dlclose (handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
CFDictionary settings;
|
|
|
|
|
|
|
|
internal CFProxy (CFDictionary settings)
|
|
|
|
{
|
|
|
|
this.settings = settings;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CFProxyType CFProxyTypeToEnum (IntPtr type)
|
|
|
|
{
|
|
|
|
if (type == kCFProxyTypeAutoConfigurationJavaScript)
|
|
|
|
return CFProxyType.AutoConfigurationJavaScript;
|
|
|
|
|
|
|
|
if (type == kCFProxyTypeAutoConfigurationURL)
|
|
|
|
return CFProxyType.AutoConfigurationUrl;
|
|
|
|
|
|
|
|
if (type == kCFProxyTypeFTP)
|
|
|
|
return CFProxyType.FTP;
|
|
|
|
|
|
|
|
if (type == kCFProxyTypeHTTP)
|
|
|
|
return CFProxyType.HTTP;
|
|
|
|
|
|
|
|
if (type == kCFProxyTypeHTTPS)
|
|
|
|
return CFProxyType.HTTPS;
|
|
|
|
|
|
|
|
if (type == kCFProxyTypeSOCKS)
|
|
|
|
return CFProxyType.SOCKS;
|
|
|
|
|
|
|
|
return CFProxyType.None;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if false
|
|
|
|
// AFAICT these get used with CFNetworkExecuteProxyAutoConfiguration*()
|
|
|
|
|
|
|
|
// TODO: bind CFHTTPMessage so we can return the proper type here.
|
|
|
|
public IntPtr AutoConfigurationHTTPResponse {
|
|
|
|
get { return settings[kCFProxyAutoConfigurationHTTPResponseKey]; }
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
public IntPtr AutoConfigurationJavaScript {
|
|
|
|
get {
|
|
|
|
if (kCFProxyAutoConfigurationJavaScriptKey == IntPtr.Zero)
|
|
|
|
return IntPtr.Zero;
|
|
|
|
|
|
|
|
return settings[kCFProxyAutoConfigurationJavaScriptKey];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public IntPtr AutoConfigurationUrl {
|
|
|
|
get {
|
|
|
|
if (kCFProxyAutoConfigurationURLKey == IntPtr.Zero)
|
|
|
|
return IntPtr.Zero;
|
|
|
|
|
|
|
|
return settings[kCFProxyAutoConfigurationURLKey];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public string HostName {
|
|
|
|
get {
|
|
|
|
if (kCFProxyHostNameKey == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return CFString.AsString (settings[kCFProxyHostNameKey]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public string Password {
|
|
|
|
get {
|
|
|
|
if (kCFProxyPasswordKey == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return CFString.AsString (settings[kCFProxyPasswordKey]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public int Port {
|
|
|
|
get {
|
|
|
|
if (kCFProxyPortNumberKey == IntPtr.Zero)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return CFNumber.AsInt32 (settings[kCFProxyPortNumberKey]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public CFProxyType ProxyType {
|
|
|
|
get {
|
|
|
|
if (kCFProxyTypeKey == IntPtr.Zero)
|
|
|
|
return CFProxyType.None;
|
|
|
|
|
|
|
|
return CFProxyTypeToEnum (settings[kCFProxyTypeKey]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public string Username {
|
|
|
|
get {
|
|
|
|
if (kCFProxyUsernameKey == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return CFString.AsString (settings[kCFProxyUsernameKey]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
internal class CFProxySettings {
|
|
|
|
static IntPtr kCFNetworkProxiesHTTPEnable;
|
|
|
|
static IntPtr kCFNetworkProxiesHTTPPort;
|
|
|
|
static IntPtr kCFNetworkProxiesHTTPProxy;
|
|
|
|
static IntPtr kCFNetworkProxiesProxyAutoConfigEnable;
|
|
|
|
static IntPtr kCFNetworkProxiesProxyAutoConfigJavaScript;
|
|
|
|
static IntPtr kCFNetworkProxiesProxyAutoConfigURLString;
|
|
|
|
|
|
|
|
static CFProxySettings ()
|
|
|
|
{
|
|
|
|
IntPtr handle = CFObject.dlopen (CFNetwork.CFNetworkLibrary, 0);
|
|
|
|
|
|
|
|
kCFNetworkProxiesHTTPEnable = CFObject.GetCFObjectHandle (handle, "kCFNetworkProxiesHTTPEnable");
|
|
|
|
kCFNetworkProxiesHTTPPort = CFObject.GetCFObjectHandle (handle, "kCFNetworkProxiesHTTPPort");
|
|
|
|
kCFNetworkProxiesHTTPProxy = CFObject.GetCFObjectHandle (handle, "kCFNetworkProxiesHTTPProxy");
|
|
|
|
kCFNetworkProxiesProxyAutoConfigEnable = CFObject.GetCFObjectHandle (handle, "kCFNetworkProxiesProxyAutoConfigEnable");
|
|
|
|
kCFNetworkProxiesProxyAutoConfigJavaScript = CFObject.GetCFObjectHandle (handle, "kCFNetworkProxiesProxyAutoConfigJavaScript");
|
|
|
|
kCFNetworkProxiesProxyAutoConfigURLString = CFObject.GetCFObjectHandle (handle, "kCFNetworkProxiesProxyAutoConfigURLString");
|
|
|
|
|
|
|
|
CFObject.dlclose (handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
CFDictionary settings;
|
|
|
|
|
|
|
|
public CFProxySettings (CFDictionary settings)
|
|
|
|
{
|
|
|
|
this.settings = settings;
|
|
|
|
}
|
|
|
|
|
|
|
|
public CFDictionary Dictionary {
|
|
|
|
get { return settings; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool HTTPEnable {
|
|
|
|
get {
|
|
|
|
if (kCFNetworkProxiesHTTPEnable == IntPtr.Zero)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return CFNumber.AsBool (settings[kCFNetworkProxiesHTTPEnable]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public int HTTPPort {
|
|
|
|
get {
|
|
|
|
if (kCFNetworkProxiesHTTPPort == IntPtr.Zero)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return CFNumber.AsInt32 (settings[kCFNetworkProxiesHTTPPort]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public string HTTPProxy {
|
|
|
|
get {
|
|
|
|
if (kCFNetworkProxiesHTTPProxy == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return CFString.AsString (settings[kCFNetworkProxiesHTTPProxy]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool ProxyAutoConfigEnable {
|
|
|
|
get {
|
|
|
|
if (kCFNetworkProxiesProxyAutoConfigEnable == IntPtr.Zero)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return CFNumber.AsBool (settings[kCFNetworkProxiesProxyAutoConfigEnable]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public string ProxyAutoConfigJavaScript {
|
|
|
|
get {
|
|
|
|
if (kCFNetworkProxiesProxyAutoConfigJavaScript == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return CFString.AsString (settings[kCFNetworkProxiesProxyAutoConfigJavaScript]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public string ProxyAutoConfigURLString {
|
|
|
|
get {
|
|
|
|
if (kCFNetworkProxiesProxyAutoConfigURLString == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return CFString.AsString (settings[kCFNetworkProxiesProxyAutoConfigURLString]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
internal static class CFNetwork {
|
|
|
|
#if !MONOTOUCH
|
|
|
|
public const string CFNetworkLibrary = "/System/Library/Frameworks/CoreServices.framework/Frameworks/CFNetwork.framework/CFNetwork";
|
|
|
|
#else
|
|
|
|
public const string CFNetworkLibrary = "/System/Library/Frameworks/CFNetwork.framework/CFNetwork";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
[DllImport (CFNetworkLibrary, EntryPoint = "CFNetworkCopyProxiesForAutoConfigurationScript")]
|
|
|
|
// CFArrayRef CFNetworkCopyProxiesForAutoConfigurationScript (CFStringRef proxyAutoConfigurationScript, CFURLRef targetURL, CFErrorRef* error);
|
|
|
|
extern static IntPtr CFNetworkCopyProxiesForAutoConfigurationScriptSequential (IntPtr proxyAutoConfigurationScript, IntPtr targetURL, out IntPtr error);
|
|
|
|
|
2015-01-13 10:44:36 +00:00
|
|
|
[DllImport (CFNetworkLibrary)]
|
|
|
|
extern static IntPtr CFNetworkExecuteProxyAutoConfigurationURL (IntPtr proxyAutoConfigURL, IntPtr targetURL, CFProxyAutoConfigurationResultCallback cb, ref CFStreamClientContext clientContext);
|
|
|
|
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
class GetProxyData : IDisposable {
|
|
|
|
public IntPtr script;
|
|
|
|
public IntPtr targetUri;
|
|
|
|
public IntPtr error;
|
|
|
|
public IntPtr result;
|
|
|
|
public ManualResetEvent evt = new ManualResetEvent (false);
|
|
|
|
|
|
|
|
public void Dispose ()
|
|
|
|
{
|
|
|
|
evt.Close ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static object lock_obj = new object ();
|
|
|
|
static Queue<GetProxyData> get_proxy_queue;
|
|
|
|
static AutoResetEvent proxy_event;
|
|
|
|
|
|
|
|
static void CFNetworkCopyProxiesForAutoConfigurationScriptThread ()
|
|
|
|
{
|
|
|
|
GetProxyData data;
|
|
|
|
var data_left = true;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
proxy_event.WaitOne ();
|
|
|
|
|
|
|
|
do {
|
|
|
|
lock (lock_obj) {
|
|
|
|
if (get_proxy_queue.Count == 0)
|
|
|
|
break;
|
|
|
|
data = get_proxy_queue.Dequeue ();
|
|
|
|
data_left = get_proxy_queue.Count > 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
data.result = CFNetworkCopyProxiesForAutoConfigurationScriptSequential (data.script, data.targetUri, out data.error);
|
|
|
|
data.evt.Set ();
|
|
|
|
} while (data_left);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static IntPtr CFNetworkCopyProxiesForAutoConfigurationScript (IntPtr proxyAutoConfigurationScript, IntPtr targetURL, out IntPtr error)
|
|
|
|
{
|
|
|
|
// This method must only be called on only one thread during an application's life time.
|
|
|
|
// Note that it's not enough to use a lock to make calls sequential across different threads,
|
|
|
|
// it has to be one thread. Also note that that thread can't be the main thread, because the
|
|
|
|
// main thread might be blocking waiting for this network request to finish.
|
|
|
|
// Another possibility would be to use JavaScriptCore to execute this piece of
|
|
|
|
// javascript ourselves, but unfortunately it's not available before iOS7.
|
|
|
|
// See bug #7923 comment #21+.
|
|
|
|
|
|
|
|
using (var data = new GetProxyData ()) {
|
|
|
|
data.script = proxyAutoConfigurationScript;
|
|
|
|
data.targetUri = targetURL;
|
|
|
|
|
|
|
|
lock (lock_obj) {
|
|
|
|
if (get_proxy_queue == null) {
|
|
|
|
get_proxy_queue = new Queue<GetProxyData> ();
|
|
|
|
proxy_event = new AutoResetEvent (false);
|
|
|
|
new Thread (CFNetworkCopyProxiesForAutoConfigurationScriptThread) {
|
|
|
|
IsBackground = true,
|
|
|
|
}.Start ();
|
|
|
|
}
|
|
|
|
get_proxy_queue.Enqueue (data);
|
|
|
|
proxy_event.Set ();
|
|
|
|
}
|
|
|
|
|
|
|
|
data.evt.WaitOne ();
|
|
|
|
|
|
|
|
error = data.error;
|
|
|
|
|
|
|
|
return data.result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static CFArray CopyProxiesForAutoConfigurationScript (IntPtr proxyAutoConfigurationScript, CFUrl targetURL)
|
|
|
|
{
|
|
|
|
IntPtr err = IntPtr.Zero;
|
|
|
|
IntPtr native = CFNetworkCopyProxiesForAutoConfigurationScript (proxyAutoConfigurationScript, targetURL.Handle, out err);
|
|
|
|
|
|
|
|
if (native == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return new CFArray (native, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static CFProxy[] GetProxiesForAutoConfigurationScript (IntPtr proxyAutoConfigurationScript, CFUrl targetURL)
|
|
|
|
{
|
|
|
|
if (proxyAutoConfigurationScript == IntPtr.Zero)
|
|
|
|
throw new ArgumentNullException ("proxyAutoConfigurationScript");
|
|
|
|
|
|
|
|
if (targetURL == null)
|
|
|
|
throw new ArgumentNullException ("targetURL");
|
|
|
|
|
|
|
|
CFArray array = CopyProxiesForAutoConfigurationScript (proxyAutoConfigurationScript, targetURL);
|
|
|
|
|
|
|
|
if (array == null)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
CFProxy[] proxies = new CFProxy [array.Count];
|
|
|
|
for (int i = 0; i < proxies.Length; i++) {
|
|
|
|
CFDictionary dict = new CFDictionary (array[i], false);
|
|
|
|
proxies[i] = new CFProxy (dict);
|
|
|
|
}
|
|
|
|
|
|
|
|
array.Dispose ();
|
|
|
|
|
|
|
|
return proxies;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static CFProxy[] GetProxiesForAutoConfigurationScript (IntPtr proxyAutoConfigurationScript, Uri targetUri)
|
|
|
|
{
|
|
|
|
if (proxyAutoConfigurationScript == IntPtr.Zero)
|
|
|
|
throw new ArgumentNullException ("proxyAutoConfigurationScript");
|
|
|
|
|
|
|
|
if (targetUri == null)
|
|
|
|
throw new ArgumentNullException ("targetUri");
|
|
|
|
|
|
|
|
CFUrl targetURL = CFUrl.Create (targetUri.AbsoluteUri);
|
|
|
|
CFProxy[] proxies = GetProxiesForAutoConfigurationScript (proxyAutoConfigurationScript, targetURL);
|
|
|
|
targetURL.Dispose ();
|
|
|
|
|
|
|
|
return proxies;
|
|
|
|
}
|
2015-01-13 10:44:36 +00:00
|
|
|
|
|
|
|
delegate void CFProxyAutoConfigurationResultCallback (IntPtr client, IntPtr proxyList, IntPtr error);
|
|
|
|
|
|
|
|
public static CFProxy[] ExecuteProxyAutoConfigurationURL (IntPtr proxyAutoConfigURL, Uri targetURL)
|
|
|
|
{
|
|
|
|
CFUrl url = CFUrl.Create (targetURL.AbsoluteUri);
|
|
|
|
if (url == null)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
CFProxy[] proxies = null;
|
|
|
|
|
|
|
|
var runLoop = CFRunLoop.CurrentRunLoop;
|
|
|
|
|
|
|
|
// Callback that will be called after executing the configuration script
|
|
|
|
CFProxyAutoConfigurationResultCallback cb = delegate (IntPtr client, IntPtr proxyList, IntPtr error) {
|
|
|
|
if (proxyList != IntPtr.Zero) {
|
|
|
|
var array = new CFArray (proxyList, false);
|
|
|
|
proxies = new CFProxy [array.Count];
|
|
|
|
for (int i = 0; i < proxies.Length; i++) {
|
|
|
|
CFDictionary dict = new CFDictionary (array[i], false);
|
|
|
|
proxies[i] = new CFProxy (dict);
|
|
|
|
}
|
|
|
|
array.Dispose ();
|
|
|
|
}
|
|
|
|
runLoop.Stop ();
|
|
|
|
};
|
|
|
|
|
|
|
|
var clientContext = new CFStreamClientContext ();
|
|
|
|
var loopSource = CFNetworkExecuteProxyAutoConfigurationURL (proxyAutoConfigURL, url.Handle, cb, ref clientContext);
|
|
|
|
|
|
|
|
// Create a private mode
|
|
|
|
var mode = CFString.Create ("Mono.MacProxy");
|
|
|
|
|
|
|
|
runLoop.AddSource (loopSource, mode);
|
|
|
|
runLoop.RunInMode (mode, double.MaxValue, false);
|
|
|
|
runLoop.RemoveSource (loopSource, mode);
|
|
|
|
|
|
|
|
return proxies;
|
|
|
|
}
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
[DllImport (CFNetworkLibrary)]
|
|
|
|
// CFArrayRef CFNetworkCopyProxiesForURL (CFURLRef url, CFDictionaryRef proxySettings);
|
|
|
|
extern static IntPtr CFNetworkCopyProxiesForURL (IntPtr url, IntPtr proxySettings);
|
|
|
|
|
|
|
|
static CFArray CopyProxiesForURL (CFUrl url, CFDictionary proxySettings)
|
|
|
|
{
|
|
|
|
IntPtr native = CFNetworkCopyProxiesForURL (url.Handle, proxySettings != null ? proxySettings.Handle : IntPtr.Zero);
|
|
|
|
|
|
|
|
if (native == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
return new CFArray (native, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static CFProxy[] GetProxiesForURL (CFUrl url, CFProxySettings proxySettings)
|
|
|
|
{
|
|
|
|
if (url == null || url.Handle == IntPtr.Zero)
|
|
|
|
throw new ArgumentNullException ("url");
|
|
|
|
|
|
|
|
if (proxySettings == null)
|
|
|
|
proxySettings = GetSystemProxySettings ();
|
|
|
|
|
|
|
|
CFArray array = CopyProxiesForURL (url, proxySettings.Dictionary);
|
|
|
|
|
|
|
|
if (array == null)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
CFProxy[] proxies = new CFProxy [array.Count];
|
|
|
|
for (int i = 0; i < proxies.Length; i++) {
|
|
|
|
CFDictionary dict = new CFDictionary (array[i], false);
|
|
|
|
proxies[i] = new CFProxy (dict);
|
|
|
|
}
|
|
|
|
|
|
|
|
array.Dispose ();
|
|
|
|
|
|
|
|
return proxies;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static CFProxy[] GetProxiesForUri (Uri uri, CFProxySettings proxySettings)
|
|
|
|
{
|
|
|
|
if (uri == null)
|
|
|
|
throw new ArgumentNullException ("uri");
|
|
|
|
|
|
|
|
CFUrl url = CFUrl.Create (uri.AbsoluteUri);
|
|
|
|
if (url == null)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
CFProxy[] proxies = GetProxiesForURL (url, proxySettings);
|
|
|
|
url.Dispose ();
|
|
|
|
|
|
|
|
return proxies;
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CFNetworkLibrary)]
|
|
|
|
// CFDictionaryRef CFNetworkCopySystemProxySettings (void);
|
|
|
|
extern static IntPtr CFNetworkCopySystemProxySettings ();
|
|
|
|
|
|
|
|
public static CFProxySettings GetSystemProxySettings ()
|
|
|
|
{
|
|
|
|
IntPtr native = CFNetworkCopySystemProxySettings ();
|
|
|
|
|
|
|
|
if (native == IntPtr.Zero)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
var dict = new CFDictionary (native, true);
|
|
|
|
|
|
|
|
return new CFProxySettings (dict);
|
|
|
|
}
|
|
|
|
|
|
|
|
class CFWebProxy : IWebProxy {
|
|
|
|
ICredentials credentials;
|
|
|
|
bool userSpecified;
|
|
|
|
|
|
|
|
public CFWebProxy ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
public ICredentials Credentials {
|
|
|
|
get { return credentials; }
|
|
|
|
set {
|
|
|
|
userSpecified = true;
|
|
|
|
credentials = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static Uri GetProxyUri (CFProxy proxy, out NetworkCredential credentials)
|
|
|
|
{
|
|
|
|
string protocol;
|
|
|
|
|
|
|
|
switch (proxy.ProxyType) {
|
|
|
|
case CFProxyType.FTP:
|
|
|
|
protocol = "ftp://";
|
|
|
|
break;
|
|
|
|
case CFProxyType.HTTP:
|
|
|
|
case CFProxyType.HTTPS:
|
|
|
|
protocol = "http://";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
credentials = null;
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
string username = proxy.Username;
|
|
|
|
string password = proxy.Password;
|
|
|
|
string hostname = proxy.HostName;
|
|
|
|
int port = proxy.Port;
|
|
|
|
string uri;
|
|
|
|
|
|
|
|
if (username != null)
|
|
|
|
credentials = new NetworkCredential (username, password);
|
|
|
|
else
|
|
|
|
credentials = null;
|
|
|
|
|
|
|
|
uri = protocol + hostname + (port != 0 ? ':' + port.ToString () : string.Empty);
|
|
|
|
|
|
|
|
return new Uri (uri, UriKind.Absolute);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Uri GetProxyUriFromScript (IntPtr script, Uri targetUri, out NetworkCredential credentials)
|
|
|
|
{
|
|
|
|
CFProxy[] proxies = CFNetwork.GetProxiesForAutoConfigurationScript (script, targetUri);
|
2015-01-13 10:44:36 +00:00
|
|
|
return SelectProxy (proxies, targetUri, out credentials);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Uri ExecuteProxyAutoConfigurationURL (IntPtr proxyAutoConfigURL, Uri targetUri, out NetworkCredential credentials)
|
|
|
|
{
|
|
|
|
CFProxy[] proxies = CFNetwork.ExecuteProxyAutoConfigurationURL (proxyAutoConfigURL, targetUri);
|
|
|
|
return SelectProxy (proxies, targetUri, out credentials);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static Uri SelectProxy (CFProxy[] proxies, Uri targetUri, out NetworkCredential credentials)
|
|
|
|
{
|
2014-08-13 10:39:27 +01:00
|
|
|
if (proxies == null) {
|
|
|
|
credentials = null;
|
|
|
|
return targetUri;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < proxies.Length; i++) {
|
|
|
|
switch (proxies[i].ProxyType) {
|
|
|
|
case CFProxyType.HTTPS:
|
|
|
|
case CFProxyType.HTTP:
|
|
|
|
case CFProxyType.FTP:
|
|
|
|
// create a Uri based on the hostname/port/etc info
|
|
|
|
return GetProxyUri (proxies[i], out credentials);
|
|
|
|
case CFProxyType.SOCKS:
|
|
|
|
default:
|
|
|
|
// unsupported proxy type, try the next one
|
|
|
|
break;
|
|
|
|
case CFProxyType.None:
|
|
|
|
// no proxy should be used
|
|
|
|
credentials = null;
|
|
|
|
return targetUri;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
credentials = null;
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Uri GetProxy (Uri targetUri)
|
|
|
|
{
|
|
|
|
NetworkCredential credentials = null;
|
|
|
|
Uri proxy = null;
|
|
|
|
|
|
|
|
if (targetUri == null)
|
|
|
|
throw new ArgumentNullException ("targetUri");
|
|
|
|
|
|
|
|
try {
|
|
|
|
CFProxySettings settings = CFNetwork.GetSystemProxySettings ();
|
|
|
|
CFProxy[] proxies = CFNetwork.GetProxiesForUri (targetUri, settings);
|
|
|
|
|
|
|
|
if (proxies != null) {
|
|
|
|
for (int i = 0; i < proxies.Length && proxy == null; i++) {
|
|
|
|
switch (proxies[i].ProxyType) {
|
|
|
|
case CFProxyType.AutoConfigurationJavaScript:
|
|
|
|
proxy = GetProxyUriFromScript (proxies[i].AutoConfigurationJavaScript, targetUri, out credentials);
|
|
|
|
break;
|
|
|
|
case CFProxyType.AutoConfigurationUrl:
|
2015-01-13 10:44:36 +00:00
|
|
|
proxy = ExecuteProxyAutoConfigurationURL (proxies[i].AutoConfigurationUrl, targetUri, out credentials);
|
2014-08-13 10:39:27 +01:00
|
|
|
break;
|
|
|
|
case CFProxyType.HTTPS:
|
|
|
|
case CFProxyType.HTTP:
|
|
|
|
case CFProxyType.FTP:
|
|
|
|
// create a Uri based on the hostname/port/etc info
|
|
|
|
proxy = GetProxyUri (proxies[i], out credentials);
|
|
|
|
break;
|
|
|
|
case CFProxyType.SOCKS:
|
|
|
|
// unsupported proxy type, try the next one
|
|
|
|
break;
|
|
|
|
case CFProxyType.None:
|
|
|
|
// no proxy should be used
|
|
|
|
proxy = targetUri;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (proxy == null) {
|
|
|
|
// no supported proxies for this Uri, fall back to trying to connect to targetUri directly
|
|
|
|
proxy = targetUri;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
proxy = targetUri;
|
|
|
|
}
|
|
|
|
} catch {
|
|
|
|
// ignore errors while retrieving proxy data
|
|
|
|
proxy = targetUri;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!userSpecified)
|
|
|
|
this.credentials = credentials;
|
|
|
|
|
|
|
|
return proxy;
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool IsBypassed (Uri targetUri)
|
|
|
|
{
|
|
|
|
if (targetUri == null)
|
|
|
|
throw new ArgumentNullException ("targetUri");
|
|
|
|
|
|
|
|
return GetProxy (targetUri) == targetUri;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static IWebProxy GetDefaultProxy ()
|
|
|
|
{
|
|
|
|
return new CFWebProxy ();
|
|
|
|
}
|
|
|
|
}
|
2017-04-10 11:41:01 +00:00
|
|
|
|
|
|
|
class CFBoolean : INativeObject, IDisposable {
|
|
|
|
IntPtr handle;
|
|
|
|
|
|
|
|
public static readonly CFBoolean True;
|
|
|
|
public static readonly CFBoolean False;
|
|
|
|
|
|
|
|
static CFBoolean ()
|
|
|
|
{
|
|
|
|
var handle = CFObject.dlopen (CFObject.CoreFoundationLibrary, 0);
|
|
|
|
if (handle == IntPtr.Zero)
|
|
|
|
return;
|
|
|
|
try {
|
|
|
|
True = new CFBoolean (CFObject.GetCFObjectHandle (handle, "kCFBooleanTrue"), false);
|
|
|
|
False = new CFBoolean (CFObject.GetCFObjectHandle (handle, "kCFBooleanFalse"), false);
|
|
|
|
}
|
|
|
|
finally {
|
|
|
|
CFObject.dlclose (handle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
internal CFBoolean (IntPtr handle, bool owns)
|
|
|
|
{
|
|
|
|
this.handle = handle;
|
|
|
|
if (!owns)
|
|
|
|
CFObject.CFRetain (handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
~CFBoolean ()
|
|
|
|
{
|
|
|
|
Dispose (false);
|
|
|
|
}
|
|
|
|
|
|
|
|
public IntPtr Handle {
|
|
|
|
get {
|
|
|
|
return handle;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Dispose ()
|
|
|
|
{
|
|
|
|
Dispose (true);
|
|
|
|
GC.SuppressFinalize (this);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected virtual void Dispose (bool disposing)
|
|
|
|
{
|
|
|
|
if (handle != IntPtr.Zero){
|
|
|
|
CFObject.CFRelease (handle);
|
|
|
|
handle = IntPtr.Zero;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static implicit operator bool (CFBoolean value)
|
|
|
|
{
|
|
|
|
return value.Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static explicit operator CFBoolean (bool value)
|
|
|
|
{
|
|
|
|
return FromBoolean (value);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static CFBoolean FromBoolean (bool value)
|
|
|
|
{
|
|
|
|
return value ? True : False;
|
|
|
|
}
|
|
|
|
|
|
|
|
[DllImport (CFObject.CoreFoundationLibrary)]
|
|
|
|
[return: MarshalAs (UnmanagedType.I1)]
|
|
|
|
extern static /* Boolean */ bool CFBooleanGetValue (/* CFBooleanRef */ IntPtr boolean);
|
|
|
|
|
|
|
|
public bool Value {
|
|
|
|
get {return CFBooleanGetValue (handle);}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static bool GetValue (IntPtr boolean)
|
|
|
|
{
|
|
|
|
return CFBooleanGetValue (boolean);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|