| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183 | using System;using System.Collections.Generic;using System.Diagnostics;using System.Globalization;using System.IO;using System.Threading;#if !NETSTANDARDusing Microsoft.Win32;#endifusing Microsoft.Win32.SafeHandles;using System.Runtime.InteropServices;using System.Reflection;#if !NETSTANDARDusing System.Security.Principal;using System.Security.AccessControl;#endifusing System.ComponentModel;using System.Security.Cryptography;namespace WinSCP{    internal class ExeSessionProcess : IDisposable    {        public event OutputDataReceivedEventHandler OutputDataReceived;        public bool HasExited { get { return _process.HasExited; } }        public int ExitCode { get { return _process.ExitCode; } }        public PipeStream StdOut { get; set; }        public Stream StdIn { get; set; }        public static ExeSessionProcess CreateForSession(Session session)        {            return new ExeSessionProcess(session, true, null);        }        public static ExeSessionProcess CreateForConsole(Session session, string additionalArguments)        {            return new ExeSessionProcess(session, false, additionalArguments);        }        private ExeSessionProcess(Session session, bool useXmlLog, string additionalArguments)        {            _session = session;            _logger = session.Logger;            _incompleteLine = string.Empty;            using (_logger.CreateCallstack())            {                string executablePath = GetExecutablePath();                _logger.WriteLine("EXE executable path resolved to {0}", executablePath);                string assemblyFilePath = _logger.GetAssemblyFilePath();                FileVersionInfo assemblyVersion = null;                if (assemblyFilePath != null)                {                    assemblyVersion = FileVersionInfo.GetVersionInfo(assemblyFilePath);                }                CheckVersion(executablePath, assemblyVersion);                string configSwitch;                if (_session.DefaultConfigurationInternal)                {                    configSwitch = "/ini=nul ";                }                else                {                    if (!string.IsNullOrEmpty(_session.IniFilePathInternal))                    {                        configSwitch = string.Format(CultureInfo.InvariantCulture, "/ini=\"{0}\" ", _session.IniFilePathInternal);                    }                    else                    {                        configSwitch = "";                    }                }                string logSwitch = null;                if (!string.IsNullOrEmpty(_session.SessionLogPath))                {                    logSwitch = string.Format(CultureInfo.InvariantCulture, "/log=\"{0}\" ", LogPathEscape(_session.SessionLogPath));                }                string xmlLogSwitch;                if (useXmlLog)                {                    xmlLogSwitch = string.Format(CultureInfo.InvariantCulture, "/xmllog=\"{0}\" /xmlgroups /xmllogrequired ", LogPathEscape(_session.XmlLogPath));                }                else                {                    xmlLogSwitch = "";                }                string logLevelSwitch = null;                if (_session.DebugLogLevel != 0)                {                    logLevelSwitch = string.Format(CultureInfo.InvariantCulture, "/loglevel={0} ", _session.DebugLogLevel);                }                string assemblyVersionStr =                    (assemblyVersion == null) ? "unk" :                    string.Format(CultureInfo.InvariantCulture, "{0}.{1}.{2} ", assemblyVersion.ProductMajorPart, assemblyVersion.ProductMinorPart, assemblyVersion.ProductBuildPart);                string assemblyVersionSwitch =                    string.Format(CultureInfo.InvariantCulture, "/dotnet={0} ", assemblyVersionStr);                string arguments =                    xmlLogSwitch + "/nointeractiveinput /stdout /stdin " + assemblyVersionSwitch +                    configSwitch + logSwitch + logLevelSwitch + _session.AdditionalExecutableArguments;                Tools.AddRawParameters(ref arguments, _session.RawConfiguration, "/rawconfig", false);                if (!string.IsNullOrEmpty(additionalArguments))                {                    arguments += " " + additionalArguments;                }                _process = new Process();                _process.StartInfo.FileName = executablePath;                _process.StartInfo.WorkingDirectory = Path.GetDirectoryName(executablePath);                _process.StartInfo.Arguments = arguments;                _process.StartInfo.UseShellExecute = false;                _process.Exited += ProcessExited;            }        }        private static string LogPathEscape(string path)        {            return Tools.ArgumentEscape(path).Replace("!", "!!");        }        public void Abort()        {            using (_logger.CreateCallstack())            {                lock (_lock)                {                    if ((_process != null) && !_process.HasExited)                    {                        _process.Kill();                    }                }            }        }        public void Start()        {            using (_logger.CreateCallstack())            {                InitializeConsole();                InitializeChild();            }        }        private void InitializeChild()        {            using (_logger.CreateCallstack())            {                // The /console is redundant for CreateForConsole                _process.StartInfo.Arguments += string.Format(CultureInfo.InvariantCulture, " /console /consoleinstance={0}", _instanceName);#if !NETSTANDARD                // When running under IIS in "impersonated" mode, the process starts, but does not do anything.                // Supposedly it "displayes" some invisible error message when starting and hangs.                // Running it "as the user" helps, eventhough it already runs as the user.                // These's probably some difference between "run as" and impersonations                if (!string.IsNullOrEmpty(_session.ExecutableProcessUserName))                {                    _logger.WriteLine("Will run process as {0}", _session.ExecutableProcessUserName);                    _process.StartInfo.UserName = _session.ExecutableProcessUserName;                    _process.StartInfo.Password = _session.ExecutableProcessPassword;                    // One of the hints for resolving C0000142 error (see below)                    // was setting this property, so that an environment is correctly loaded,                    // so DLLs can be found and loaded.                    _process.StartInfo.LoadUserProfile = true;                    // Without granting both window station and desktop access permissions,                    // WinSCP process aborts with C0000142 (DLL Initialization Failed) error,                    // when "running as user"                    _logger.WriteLine("Granting access to window station");                    try                    {                        IntPtr windowStation = UnsafeNativeMethods.GetProcessWindowStation();                        GrantAccess(windowStation, (int)WindowStationRights.AllAccess);                    }                    catch (Exception e)                    {                        throw _logger.WriteException(new SessionLocalException(_session, "Error granting access to window station", e));                    }                    _logger.WriteLine("Granting access to desktop");                    try                    {                        IntPtr desktop = UnsafeNativeMethods.GetThreadDesktop(UnsafeNativeMethods.GetCurrentThreadId());                        GrantAccess(desktop, (int)DesktopRights.AllAccess);                    }                    catch (Exception e)                    {                        throw _logger.WriteException(new SessionLocalException(_session, "Error granting access to desktop", e));                    }                }#endif                _logger.WriteLine("Starting \"{0}\" {1}", _process.StartInfo.FileName, _process.StartInfo.Arguments);                _process.Start();                _logger.WriteLine("Started process {0}", _process.Id);                _thread = new Thread(ProcessEvents)                {                    IsBackground = true                };                _thread.Start();            }        }        // Handles returned by GetProcessWindowStation and GetThreadDesktop should not be closed        internal class NoopSafeHandle : SafeHandle        {            public NoopSafeHandle(IntPtr handle) :                base(handle, false)            {            }            public override bool IsInvalid            {                get { return false; }            }            protected override bool ReleaseHandle()            {                return true;            }        }#if !NETSTANDARD        private void GrantAccess(IntPtr handle, int accessMask)        {            using (SafeHandle safeHandle = new NoopSafeHandle(handle))            {                GenericSecurity security =                    new GenericSecurity(false, ResourceType.WindowObject, safeHandle, AccessControlSections.Access);                security.AddAccessRule(                    new GenericAccessRule(new NTAccount(_session.ExecutableProcessUserName), accessMask, AccessControlType.Allow));                security.Persist(safeHandle, AccessControlSections.Access);            }        }#endif        private void ProcessExited(object sender, EventArgs e)        {            _logger.WriteLine("Process {0} exited with exit code {1}", _process.Id, _process.ExitCode);        }        private bool AbortedOrExited()        {            if (_abort)            {                _logger.WriteLine("Aborted");                return true;            }            else if (_process.HasExited)            {                _logger.WriteLine("Exited");                return true;            }            else            {                return false;            }        }        private void ProcessEvents()        {            using (_logger.CreateCallstack())            {                while (!AbortedOrExited())                {                    _logger.WriteLineLevel(1, "Waiting for request event");                    // Keep in sync with a delay in SessionLogReader.DoRead                    if (_requestEvent.WaitOne(100, false))                    {                        _logger.WriteLineLevel(1, "Got request event");                        ProcessEvent();                    }                    if (_logger.LogLevel >= 1)                    {                        _logger.WriteLine(string.Format(CultureInfo.InvariantCulture, "2nd generation collection count: {0}", GC.CollectionCount(2)));                        _logger.WriteLine(string.Format(CultureInfo.InvariantCulture, "Total memory allocated: {0}", GC.GetTotalMemory(false)));                    }                }            }        }        private void ProcessEvent()        {            using (_logger.CreateCallstack())            {                using (ConsoleCommStruct commStruct = AcquireCommStruct())                {                    switch (commStruct.Event)                    {                        case ConsoleEvent.Print:                            ProcessPrintEvent(commStruct.PrintEvent);                            break;                        case ConsoleEvent.Input:                            ProcessInputEvent(commStruct.InputEvent);                            break;                        case ConsoleEvent.Choice:                            ProcessChoiceEvent(commStruct.ChoiceEvent);                            break;                        case ConsoleEvent.Title:                            ProcessTitleEvent(commStruct.TitleEvent);                            break;                        case ConsoleEvent.Init:                            ProcessInitEvent(commStruct.InitEvent);                            break;                        case ConsoleEvent.Progress:                            ProcessProgressEvent(commStruct.ProgressEvent);                            break;                        case ConsoleEvent.TransferOut:                            ProcessTransferOutEvent(commStruct.TransferOutEvent);                            break;                        case ConsoleEvent.TransferIn:                            ProcessTransferInEvent(commStruct.TransferInEvent);                            break;                        default:                            throw _logger.WriteException(new NotImplementedException());                    }                }                _responseEvent.Set();            }        }        private void ProcessChoiceEvent(ConsoleChoiceEventStruct e)        {            using (_logger.CreateCallstack())            {                _logger.WriteLine(                    "Options: [{0}], Timer: [{1}], Timeouting: [{2}], Timeouted: [{3}], Break: [{4}]",                    e.Options, e.Timer, e.Timeouting, e.Timeouted, e.Break);                QueryReceivedEventArgs args = new QueryReceivedEventArgs                {                    Message = e.Message                };                _session.ProcessChoice(args);                if (args.SelectedAction == QueryReceivedEventArgs.Action.None)                {                    if (e.Timeouting)                    {                        Thread.Sleep((int)e.Timer);                        e.Result = e.Timeouted;                    }                    else                    {                        e.Result = e.Break;                    }                }                else if (args.SelectedAction == QueryReceivedEventArgs.Action.Continue)                {                    if (e.Timeouting)                    {                        Thread.Sleep((int)e.Timer);                        e.Result = e.Timeouted;                    }                    else                    {                        e.Result = e.Continue;                    }                }                else if (args.SelectedAction == QueryReceivedEventArgs.Action.Abort)                {                    e.Result = e.Break;                }                _logger.WriteLine("Options Result: [{0}]", e.Result);            }        }        private void ProcessTitleEvent(ConsoleTitleEventStruct e)        {            using (_logger.CreateCallstack())            {                _logger.WriteLine("Not-supported title event [{0}]", e.Title);            }        }        private void ProcessInputEvent(ConsoleInputEventStruct e)        {            using (_logger.CreateCallstack())            {                while (!AbortedOrExited())                {                    lock (_input)                    {                        if (_input.Count > 0)                        {                            e.Str = _input[0];                            e.Result = true;                            _input.RemoveAt(0);                            Print(false, false, _log[0] + "\n");                            _log.RemoveAt(0);                            return;                        }                    }                    _inputEvent.WaitOne(100, false);                }            }        }        private void Print(bool fromBeginning, bool error, string message)        {            if (fromBeginning && ((message.Length == 0) || (message[0] != '\n')))            {                _lastFromBeginning = message;                _logger.WriteLine("Buffered from-beginning message [{0}]", _lastFromBeginning);                OutputDataReceived?.Invoke(this, null);            }            else            {                if (!string.IsNullOrEmpty(_lastFromBeginning))                {                    AddToOutput(_lastFromBeginning, false);                    _lastFromBeginning = null;                }                if (fromBeginning && (message.Length > 0) && (message[0] == '\n'))                {                    AddToOutput("\n", false);                    _lastFromBeginning = message.Substring(1);                    _logger.WriteLine("Buffered from-beginning message [{0}]", _lastFromBeginning);                }                else                {                    AddToOutput(message, error);                }            }        }        private void AddToOutput(string message, bool error)        {            string[] lines = (_incompleteLine + message).Split(new[] { '\n' });            _incompleteLine = lines[lines.Length - 1];            for (int i = 0; i < lines.Length - 1; ++i)            {                OutputDataReceived?.Invoke(this, new OutputDataReceivedEventArgs(lines[i], error));            }        }        private void ProcessPrintEvent(ConsolePrintEventStruct e)        {            _logger.WriteLineLevel(1, string.Format(CultureInfo.CurrentCulture, "Print: {0}", e.Message));            Print(e.FromBeginning, e.Error, e.Message);        }        private void ProcessInitEvent(ConsoleInitEventStruct e)        {            using (_logger.CreateCallstack())            {                if (!e.UseStdErr ||                    (e.BinaryOutput != ConsoleInitEventStruct.StdInOut.Binary) ||                    (e.BinaryInput != ConsoleInitEventStruct.StdInOut.Binary))                {                    throw _logger.WriteException(new InvalidOperationException("Unexpected console interface options"));                }                e.InputType = 3; // pipe                e.OutputType = 3; // pipe                e.WantsProgress = _session.WantsProgress;            }        }        private void ProcessProgressEvent(ConsoleProgressEventStruct e)        {            using (_logger.CreateCallstack())            {                _logger.WriteLine(                    "File Name [{0}] - Directory [{1}] - Overall Progress [{2}] - File Progress [{3}] - CPS [{4}]",                    e.FileName, e.Directory, e.OverallProgress, e.FileProgress, e.CPS);                if (!_cancel)                {                    FileTransferProgressEventArgs args = new FileTransferProgressEventArgs();                    switch (e.Operation)                    {                        case ConsoleProgressEventStruct.ProgressOperation.Copy:                            args.Operation = ProgressOperation.Transfer;                            break;                        default:                            throw _logger.WriteException(new ArgumentOutOfRangeException("Unknown progress operation", (Exception)null));                    }                    switch (e.Side)                    {                        case ConsoleProgressEventStruct.ProgressSide.Local:                            args.Side = ProgressSide.Local;                            break;                        case ConsoleProgressEventStruct.ProgressSide.Remote:                            args.Side = ProgressSide.Remote;                            break;                        default:                            throw _logger.WriteException(new ArgumentOutOfRangeException("Unknown progress side", (Exception)null));                    }                    args.FileName = e.FileName;                    args.Directory = e.Directory;                    args.OverallProgress = ((double)e.OverallProgress) / 100;                    args.FileProgress = ((double)e.FileProgress) / 100;                    args.CPS = (int)e.CPS;                    args.Cancel = false;                    _session.ProcessProgress(args);                }                if (_cancel)                {                    e.Cancel = true;                }            }        }        private void ProcessTransferOutEvent(ConsoleTransferEventStruct e)        {            using (_logger.CreateCallstack())            {                _logger.WriteLine("Len [{0}]", e.Len);                if (StdOut == null)                {                    throw _logger.WriteException(new InvalidOperationException("Unexpected data"));                }                int len = (int)e.Len;                if (len > 0)                {                    StdOut.WriteInternal(e.Data, 0, len);                    _logger.WriteLine("Data written to the buffer");                }                else                {                    StdOut.CloseWrite();                    _logger.WriteLine("Data buffer closed");                }            }        }        private void ProcessTransferInEvent(ConsoleTransferEventStruct e)        {            using (_logger.CreateCallstack())            {                _logger.WriteLine("Len [{0}]", e.Len);                if (StdIn == null)                {                    throw _logger.WriteException(new InvalidOperationException("Unexpected data request"));                }                try                {                    int len = (int)e.Len;                    len = StdIn.Read(e.Data, 0, len);                    _logger.WriteLine("{0} bytes read", len);                    e.Len = (UIntPtr)len;                }                catch (Exception ex)                {                    _logger.WriteLine("Error reading data stream");                    _logger.WriteException(ex);                    e.Error = true;                }            }        }        private void InitializeConsole()        {            using (_logger.CreateCallstack())            {                int attempts = 0;                Random random = new Random();                int process = Process.GetCurrentProcess().Id;                do                {                    if (attempts > MaxAttempts)                    {                        throw _logger.WriteException(new SessionLocalException(_session, "Cannot find unique name for event object."));                    }                    int instanceNumber = random.Next(1000);                    _instanceName = string.Format(CultureInfo.InvariantCulture, "_{0}_{1}_{2}", process, GetHashCode(), instanceNumber);                    _logger.WriteLine("Trying event {0}", _instanceName);                    if (!TryCreateEvent(ConsoleEventRequest + _instanceName, out _requestEvent))                    {                        _logger.WriteLine("Event {0} is not unique", _instanceName);                        _requestEvent.Close();                        _requestEvent = null;                    }                    else                    {                        _logger.WriteLine("Event {0} is unique", _instanceName);                        _responseEvent = CreateEvent(ConsoleEventResponse + _instanceName);                        _cancelEvent = CreateEvent(ConsoleEventCancel + _instanceName);                        string fileMappingName = ConsoleMapping + _instanceName;                        _fileMapping = CreateFileMapping(fileMappingName);                        if (Marshal.GetLastWin32Error() == UnsafeNativeMethods.ERROR_ALREADY_EXISTS)                        {                            throw _logger.WriteException(new SessionLocalException(_session, string.Format(CultureInfo.InvariantCulture, "File mapping {0} already exists", fileMappingName)));                        }                        if (_fileMapping.IsInvalid)                        {                            throw _logger.WriteException(new SessionLocalException(_session, string.Format(CultureInfo.InvariantCulture, "Cannot create file mapping {0}", fileMappingName)));                        }                    }                    ++attempts;                }                while (_requestEvent == null);                using (ConsoleCommStruct commStruct = AcquireCommStruct())                {                    commStruct.InitHeader();                }                if (_session.GuardProcessWithJobInternal)                {                    string jobName = ConsoleJob + _instanceName;                    _job = new Job(_logger, jobName);                }            }        }        private SafeFileHandle CreateFileMapping(string fileMappingName)        {            unsafe            {                IntPtr securityAttributesPtr = IntPtr.Zero;#if !NETSTANDARD                // We use the EventWaitHandleSecurity only to generate the descriptor binary form                // that does not differ for object types, so we abuse the existing "event handle" implementation,                // not to have to create the file mapping SecurityAttributes via P/Invoke.                // .NET 4 supports MemoryMappedFile and MemoryMappedFileSecurity natively already                EventWaitHandleSecurity security = CreateSecurity((EventWaitHandleRights)FileMappingRights.AllAccess);                if (security != null)                {                    SecurityAttributes securityAttributes = new SecurityAttributes();                    securityAttributes.nLength = (uint)Marshal.SizeOf(securityAttributes);                    byte[] descriptorBinaryForm = security.GetSecurityDescriptorBinaryForm();                    byte * buffer = stackalloc byte[descriptorBinaryForm.Length];                    for (int i = 0; i < descriptorBinaryForm.Length; i++)                    {                        buffer[i] = descriptorBinaryForm[i];                    }                    securityAttributes.lpSecurityDescriptor = (IntPtr)buffer;                    int length = Marshal.SizeOf(typeof(SecurityAttributes));                    securityAttributesPtr = Marshal.AllocHGlobal(length);                    Marshal.StructureToPtr(securityAttributes, securityAttributesPtr, false);                }#endif                return                    UnsafeNativeMethods.CreateFileMapping(                        new SafeFileHandle(new IntPtr(-1), true), securityAttributesPtr, FileMapProtection.PageReadWrite, 0,                        ConsoleCommStruct.Size, fileMappingName);            }        }        private ConsoleCommStruct AcquireCommStruct()        {            return new ConsoleCommStruct(_session, _fileMapping);        }        private bool TryCreateEvent(string name, out EventWaitHandle ev)        {            _logger.WriteLine("Creating event {0}", name);            string securityDesc;#if !NETSTANDARD            EventWaitHandleSecurity security = CreateSecurity(EventWaitHandleRights.FullControl);            ev = new EventWaitHandle(false, EventResetMode.AutoReset, name, out bool createdNew, security);            securityDesc = (security != null ? security.GetSecurityDescriptorSddlForm(AccessControlSections.All) : "none");#else            ev = new EventWaitHandle(false, EventResetMode.AutoReset, name, out bool createdNew);            securityDesc = "not impl";#endif            _logger.WriteLine(                "Created event {0} with handle {1} with security {2}, new {3}",                name, ev.SafeWaitHandle.DangerousGetHandle(), securityDesc, createdNew);            return createdNew;        }#if !NETSTANDARD        private EventWaitHandleSecurity CreateSecurity(EventWaitHandleRights eventRights)        {            EventWaitHandleSecurity security = null;            // When "running as user", we have to grant the target user permissions to the objects (events and file mapping) explicitly            if (!string.IsNullOrEmpty(_session.ExecutableProcessUserName))            {                security = new EventWaitHandleSecurity();                IdentityReference si;                try                {                    si = new NTAccount(_session.ExecutableProcessUserName);                }                catch (Exception e)                {                    throw _logger.WriteException(new SessionLocalException(_session, string.Format(CultureInfo.CurrentCulture, "Error resolving account {0}", _session.ExecutableProcessUserName), e));                }                EventWaitHandleAccessRule rule =                    new EventWaitHandleAccessRule(                        si, eventRights, AccessControlType.Allow);                security.AddAccessRule(rule);            }            return security;        }#endif        private EventWaitHandle CreateEvent(string name)        {            if (!TryCreateEvent(name, out EventWaitHandle ev))            {                throw _logger.WriteException(new SessionLocalException(_session, string.Format(CultureInfo.InvariantCulture, "Event {0} already exists", name)));            }            return ev;        }        private void TestEventClosed(string name)        {            if (_session.TestHandlesClosedInternal)            {                _logger.WriteLine("Testing that event {0} is closed", name);                if (TryCreateEvent(name, out EventWaitHandle ev))                {                    ev.Close();                }                else                {                    _logger.WriteLine("Exception: Event {0} was not closed yet", name);                }            }        }        private void AddInput(string str, string log)        {            Type structType = typeof(ConsoleInputEventStruct);            FieldInfo strField = structType.GetField("Str");            object[] attributes = strField.GetCustomAttributes(typeof(MarshalAsAttribute), false);            if (attributes.Length != 1)            {                throw _logger.WriteException(new InvalidOperationException("MarshalAs attribute not found for ConsoleInputEventStruct.Str"));            }            MarshalAsAttribute marshalAsAttribute = (MarshalAsAttribute)attributes[0];            if (marshalAsAttribute.SizeConst <= str.Length)            {                throw _logger.WriteException(                    new SessionLocalException(                        _session,                        string.Format(CultureInfo.CurrentCulture, "Input [{0}] is too long ({1} limit)", str, marshalAsAttribute.SizeConst)));            }            lock (_input)            {                _input.Add(str);                _log.Add(log);                _inputEvent.Set();            }        }        public void ExecuteCommand(string command, string log)        {            using (_logger.CreateCallstack())            {                _cancel = false;                AddInput(command, log);            }        }        public void Close()        {            using (_logger.CreateCallstack())            {                int timeout;                #if DEBUG                // in debug build, we expect the winscp.exe to run in tracing mode, being very slow                timeout = 10000;                #else                timeout = 2000;                #endif                _logger.WriteLine("Waiting for process to exit ({0} ms)", timeout);                if (!_process.WaitForExit(timeout))                {                    _logger.WriteLine("Killing process");                    _process.Kill();                }            }        }        public void Dispose()        {            using (_logger.CreateCallstack())            {                lock (_lock)                {                    if (_session.TestHandlesClosedInternal)                    {                        _logger.WriteLine("Will test that handles are closed");                    }                    _abort = true;                    if (_thread != null)                    {                        _thread.Join();                        _thread = null;                    }                    if (_process != null)                    {                        _process.Dispose();                        _process = null;                    }                    if (_requestEvent != null)                    {                        _requestEvent.Close();                        TestEventClosed(ConsoleEventRequest + _instanceName);                    }                    if (_responseEvent != null)                    {                        _responseEvent.Close();                        TestEventClosed(ConsoleEventResponse + _instanceName);                    }                    if (_cancelEvent != null)                    {                        _cancelEvent.Close();                        TestEventClosed(ConsoleEventCancel + _instanceName);                    }                    if (_fileMapping != null)                    {                        _fileMapping.Dispose();                        _fileMapping = null;                        if (_session.TestHandlesClosedInternal)                        {                            _logger.WriteLine("Testing that file mapping is closed");                            string fileMappingName = ConsoleMapping + _instanceName;                            SafeFileHandle fileMapping = CreateFileMapping(fileMappingName);                            if (Marshal.GetLastWin32Error() == UnsafeNativeMethods.ERROR_ALREADY_EXISTS)                            {                                _logger.WriteLine("Exception: File mapping {0} was not closed yet", fileMappingName);                            }                            if (!fileMapping.IsInvalid)                            {                                fileMapping.Dispose();                            }                        }                    }                    if (_inputEvent != null)                    {                        _inputEvent.Close();                        _inputEvent = null;                    }                    if (_job != null)                    {                        _job.Dispose();                        _job = null;                    }                }            }        }        private string GetExecutablePath()        {            using (_logger.CreateCallstack())            {                string executablePath;                if (!string.IsNullOrEmpty(_session.ExecutablePath))                {                    executablePath = _session.ExecutablePath;                    if (!File.Exists(executablePath))                    {                        throw _logger.WriteException(new SessionLocalException(_session, string.Format(CultureInfo.CurrentCulture, "{0} does not exists.", executablePath)));                    }                }                else                {                    if (!TryFindExecutableInPath(GetAssemblyPath(), out executablePath) &&                        !TryFindExecutableInPath(GetEntryAssemblyPath(), out executablePath) &&#if !NETSTANDARD                        !TryFindExecutableInPath(GetInstallationPath(RegistryHive.CurrentUser), out executablePath) &&                        !TryFindExecutableInPath(GetInstallationPath(RegistryHive.LocalMachine), out executablePath) &&#endif                        !TryFindExecutableInPath(GetDefaultInstallationPath(), out executablePath))                    {                        string entryAssemblyDesc = string.Empty;                        Assembly entryAssembly = Assembly.GetEntryAssembly();                        if (entryAssembly != null)                        {                            entryAssemblyDesc = $", nor the entry assembly {entryAssembly.GetName().Name} ({GetEntryAssemblyPath()})";                        }                        throw _logger.WriteException(                            new SessionLocalException(_session,                                string.Format(CultureInfo.CurrentCulture,                                    "The {0} executable was not found at location of the assembly {1} ({2}){3}, nor in an installation path. You may use Session.ExecutablePath property to explicitly set path to {0}.",                                    ExeExecutableFileName, Assembly.GetExecutingAssembly().GetName().Name, GetAssemblyPath(), entryAssemblyDesc)));                    }                }                return executablePath;            }        }        private static string GetDefaultInstallationPath()        {            string programFiles;            if (IntPtr.Size == 8)            {                programFiles = Environment.GetFolderPath(Environment.SpecialFolder.ProgramFilesX86);            }            else            {                programFiles = Environment.GetFolderPath(Environment.SpecialFolder.ProgramFiles);            }            return Path.Combine(programFiles, "WinSCP");        }#if !NETSTANDARD        private static string GetInstallationPath(RegistryHive hive)        {            RegistryKey baseKey = RegistryKey.OpenBaseKey(hive, RegistryView.Registry32);            RegistryKey key = baseKey.OpenSubKey(@"Software\Microsoft\Windows\CurrentVersion\Uninstall\winscp3_is1");            string result = (key != null) ? (string)key.GetValue("Inno Setup: App Path") : null;            return result;        }#endif        private bool TryFindExecutableInPath(string path, out string result)        {            if (string.IsNullOrEmpty(path))            {                result = null;            }            else            {                string executablePath = Path.Combine(path, ExeExecutableFileName);                if (File.Exists(executablePath))                {                    result = executablePath;                    _logger.WriteLine("Executable found in {0}", executablePath);                }                else                {                    result = null;                    _logger.WriteLine("Executable not found in {0}", executablePath);                }            }            return (result != null);        }        private string GetAssemblyPath()        {            return DoGetAssemblyPath(_logger.GetAssemblyFilePath());        }        private string GetEntryAssemblyPath()        {            return DoGetAssemblyPath(_logger.GetEntryAssemblyFilePath());        }        private static string DoGetAssemblyPath(string codeBasePath)        {            string path = null;            if (!string.IsNullOrEmpty(codeBasePath))            {                path = Path.GetDirectoryName(codeBasePath);                Debug.Assert(path != null);            }            return path;        }        [DllImport("version.dll", CharSet = CharSet.Auto, SetLastError = true, BestFitMapping = false)]        public static extern int GetFileVersionInfoSize(string lptstrFilename, out int handle);        private void CheckVersion(string exePath, FileVersionInfo assemblyVersion)        {            using (_logger.CreateCallstack())            {                if (assemblyVersion == null)                {                    _logger.WriteLine("Assembly version not known, cannot check version");                }                else if (assemblyVersion.ProductVersion == AssemblyConstants.UndefinedProductVersion)                {                    _logger.WriteLine("Undefined assembly version, cannot check version");                }                else                {                    FileVersionInfo version = FileVersionInfo.GetVersionInfo(exePath);                    _logger.WriteLine("Version of {0} is {1}, product {2} version is {3}", exePath, version.FileVersion, version.ProductName, version.ProductVersion);                    Exception accessException = null;                    try                    {                        using (File.OpenRead(exePath))                        {                        }                        long size = new FileInfo(exePath).Length;                        _logger.WriteLine($"Size of the executable file is {size}");                        int verInfoSize = GetFileVersionInfoSize(exePath, out int handle);                        if (verInfoSize == 0)                        {                            throw new Exception($"Cannot retrieve {exePath} version info", new Win32Exception());                        }                        else                        {                            _logger.WriteLine($"Size of the executable file version info is {verInfoSize}");                        }                    }                    catch (Exception e)                    {                        _logger.WriteLine("Accessing executable file failed");                        _logger.WriteException(e);                        accessException = e;                    }                    if (_session.DisableVersionCheck)                    {                        _logger.WriteLine("Version check disabled (not recommended)");                    }                    else if (assemblyVersion.ProductVersion != version.ProductVersion)                    {                        try                        {                            using (SHA256 SHA256 = SHA256.Create())                            using (FileStream stream = File.OpenRead(exePath))                            {                                string sha256 = Convert.ToBase64String(SHA256.ComputeHash(stream));                                _logger.WriteLine($"SHA-256 of the executable file is {sha256}");                            }                        }                        catch (Exception e)                        {                            _logger.WriteLine("Calculating SHA-256 of the executable file failed");                            _logger.WriteException(e);                        }                        string message;                        if (string.IsNullOrEmpty(version.ProductVersion) && (accessException != null))                        {                            message = $"Cannot use {exePath}";                        }                        else                        {                            message =                                $"The version of {exePath} ({version.ProductVersion}) does not match " +                                $"version of this assembly {_logger.GetAssemblyFilePath()} ({assemblyVersion.ProductVersion}).";                        }                        throw _logger.WriteException(new SessionLocalException(_session, message, accessException));                    }                }            }        }        public void WriteStatus()        {            string executablePath = GetExecutablePath();            _logger.WriteLine("{0} - exists [{1}]", executablePath, File.Exists(executablePath));        }        public void RequestCallstack()        {            using (_logger.CreateCallstack())            {                lock (_lock)                {                    if (_process == null)                    {                        _logger.WriteLine("Process is closed already");                    }                    else                    {                        try                        {                            string eventName = string.Format(CultureInfo.InvariantCulture, "WinSCPCallstack{0}", _process.Id);                            using (EventWaitHandle ev = EventWaitHandle.OpenExisting(eventName))                            {                                _logger.WriteLine("Setting event {0}", eventName);                                ev.Set();                                string callstackFileName = string.Format(CultureInfo.InvariantCulture, "{0}.txt", eventName);                                string callstackPath = Path.Combine(Path.GetTempPath(), callstackFileName);                                int timeout = 2000;                                while (!File.Exists(callstackPath))                                {                                    if (timeout < 0)                                    {                                        string message = string.Format(CultureInfo.CurrentCulture, "Timeout waiting for callstack file {0} to be created ", callstackPath);                                        throw new TimeoutException(message);                                    }                                    int step = 50;                                    timeout -= 50;                                    Thread.Sleep(step);                                }                                _logger.WriteLine("Callstack file {0} has been created", callstackPath);                                // allow writting to be finished                                Thread.Sleep(100);                                _logger.WriteLine(File.ReadAllText(callstackPath));                                File.Delete(callstackPath);                            }                        }                        catch (Exception e)                        {                            _logger.WriteException(e);                        }                    }                }            }        }        public void Cancel()        {            _cancel = true;        }        private const int MaxAttempts = 10;        private const string ConsoleMapping = "WinSCPConsoleMapping";        private const string ConsoleEventRequest = "WinSCPConsoleEventRequest";        private const string ConsoleEventResponse = "WinSCPConsoleEventResponse";        private const string ConsoleEventCancel = "WinSCPConsoleEventCancel";        private const string ConsoleJob = "WinSCPConsoleJob";        private const string ExeExecutableFileName = "winscp.exe";        private Process _process;        private readonly object _lock = new object();        private readonly Logger _logger;        private readonly Session _session;        private EventWaitHandle _requestEvent;        private EventWaitHandle _responseEvent;        private EventWaitHandle _cancelEvent;        private SafeFileHandle _fileMapping;        private string _instanceName;        private Thread _thread;        private bool _abort;        private string _lastFromBeginning;        private string _incompleteLine;        private readonly List<string> _input = new List<string>();        private readonly List<string> _log = new List<string>();        private AutoResetEvent _inputEvent = new AutoResetEvent(false);        private Job _job;        private bool _cancel;    }}
 |