// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT License.
// See the LICENSE file in the project root for more information. 
#if HAS_REMOTING
using System.Linq.Expressions;
using System.Reflection;
using System.Runtime.Remoting.Lifetime;
namespace System.Reactive.Linq
{
    /// 
    /// Provides a set of static methods for exposing observable sequences through .NET Remoting.
    /// 
    public static partial class RemotingObservable
    {
        #region Remotable
        /// 
        /// Makes an observable sequence remotable, using an infinite lease for the  wrapping the source.
        /// 
        /// The type of the elements in the source sequence.
        /// Source sequence.
        /// The observable sequence that supports remote subscriptions.
        ///  is null.
        [Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Naming", "CA1704:IdentifiersShouldBeSpelledCorrectly", MessageId = "Remotable", Justification = "In honor of the .NET Remoting heroes.")]
        public static IObservable Remotable(this IObservable source)
        {
            if (source == null)
            {
                throw new ArgumentNullException(nameof(source));
            }
            return Remotable_(source);
        }
        /// 
        /// Makes an observable sequence remotable, using a controllable lease for the  wrapping the source.
        /// 
        /// The type of the elements in the source sequence.
        /// Source sequence.
        /// Lease object to control lifetime of the remotable sequence. Notice null is a supported value.
        /// The observable sequence that supports remote subscriptions.
        ///  is null.
        [Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Naming", "CA1704:IdentifiersShouldBeSpelledCorrectly", MessageId = "Remotable", Justification = "In honor of the .NET Remoting heroes.")]
        public static IObservable Remotable(this IObservable source, ILease lease)
        {
            if (source == null)
            {
                throw new ArgumentNullException(nameof(source));
            }
            return Remotable_(source, lease);
        }
        /// 
        /// Makes an observable sequence remotable, using an infinite lease for the  wrapping the source.
        /// 
        /// The type of the elements in the source sequence.
        /// Source sequence.
        /// The observable sequence that supports remote subscriptions.
        ///  is null.
        [Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Naming", "CA1704:IdentifiersShouldBeSpelledCorrectly", MessageId = "Remotable", Justification = "In honor of the .NET Remoting heroes.")]
        public static IQbservable Remotable(this IQbservable source)
        {
            if (source == null)
            {
                throw new ArgumentNullException(nameof(source));
            }
            return source.Provider.CreateQuery(
                Expression.Call(
                    null,
#if CRIPPLED_REFLECTION
                    InfoOf(() => RemotingObservable.Remotable(default(IQbservable))),
#else
                    ((MethodInfo)MethodBase.GetCurrentMethod()).MakeGenericMethod(typeof(TSource)),
#endif
                    source.Expression
                )
            );
        }
        /// 
        /// Makes an observable sequence remotable, using a controllable lease for the  wrapping the source.
        /// 
        /// The type of the elements in the source sequence.
        /// Source sequence.
        /// Lease object to control lifetime of the remotable sequence. Notice null is a supported value.
        /// The observable sequence that supports remote subscriptions.
        ///  is null.
        [Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Naming", "CA1704:IdentifiersShouldBeSpelledCorrectly", MessageId = "Remotable", Justification = "In honor of the .NET Remoting heroes.")]
        public static IQbservable Remotable(this IQbservable source, ILease lease)
        {
            if (source == null)
            {
                throw new ArgumentNullException(nameof(source));
            }
            return source.Provider.CreateQuery(
                Expression.Call(
                    null,
#if CRIPPLED_REFLECTION
                    InfoOf(() => RemotingObservable.Remotable(default(IQbservable), default(ILease))),
#else
                    ((MethodInfo)MethodBase.GetCurrentMethod()).MakeGenericMethod(typeof(TSource)),
#endif
                    source.Expression,
                    Expression.Constant(lease, typeof(ILease))
                )
            );
        }
#if CRIPPLED_REFLECTION
        internal static MethodInfo InfoOf(Expression> f)
        {
            return ((MethodCallExpression)f.Body).Method;
        }
#endif
        #endregion
    }
}
#endif