From f2db361681434689210d3a87b9fff623c89b44f0 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Wed, 13 Nov 2019 11:40:08 -0800 Subject: [PATCH 01/17] Squashed commit of the following: commit c36b2482c04070529cbb009b09ec60cea70da8bd Merge: 4d6e9754c c274b38dc Author: Dustin Howett Date: Fri Nov 8 14:16:44 2019 -0800 Merge remote-tracking branch 'github/master' into dev/duhowett/closeonexit commit 4d6e9754c99453b535c18ce9858efe312d0d958a Author: Dustin Howett Date: Wed Nov 6 15:49:38 2019 -0800 further hax commit 1950413962905b93ecc8a39bb1aa3f86d62c5aec Author: Dustin Howett Date: Wed Nov 6 07:30:32 2019 -0800 hacks: try to implement the state thing commit 22dd0294f289266f65e2757186cd725b5e9bf838 Merge: 5763be398 357e835f5 Author: Dustin Howett Date: Wed Nov 6 15:15:21 2019 -0800 Merge remote-tracking branch 'github/master' into HEAD commit 5763be3988c93f8e3937b234c51c76bad6095d2b Author: Dustin Howett Date: Tue Nov 5 17:47:22 2019 -0800 Spec: propose an evolution of closeOnExit and TerminalConnection --- .../TerminalConnection/AzureConnection.cpp | 33 ++++---------- .../TerminalConnection/AzureConnection.h | 7 +-- .../TerminalConnection/ConptyConnection.cpp | 43 +++++++++++++------ .../TerminalConnection/ConptyConnection.h | 11 +++-- .../TerminalConnection/EchoConnection.cpp | 11 ----- .../TerminalConnection/EchoConnection.h | 6 ++- .../ITerminalConnection.idl | 23 +++++++++- src/cascadia/TerminalControl/TermControl.cpp | 9 +++- src/cascadia/inc/cppwinrt_utils.h | 8 ++++ 9 files changed, 89 insertions(+), 62 deletions(-) diff --git a/src/cascadia/TerminalConnection/AzureConnection.cpp b/src/cascadia/TerminalConnection/AzureConnection.cpp index 2acda84576b..7066e3b02e4 100644 --- a/src/cascadia/TerminalConnection/AzureConnection.cpp +++ b/src/cascadia/TerminalConnection/AzureConnection.cpp @@ -74,28 +74,6 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _outputHandlers.remove(token); } - // Method description: - // - ascribes to the ITerminalConnection interface - // - registers a terminal-disconnected event handler - // Arguments: - // - the handler - // Return value: - // - the event token for the handler - winrt::event_token AzureConnection::TerminalDisconnected(Microsoft::Terminal::TerminalConnection::TerminalDisconnectedEventArgs const& handler) - { - return _disconnectHandlers.add(handler); - } - - // Method description: - // - ascribes to the ITerminalConnection interface - // - revokes a terminal-disconnected event handler - // Arguments: - // - the event token for the handler - void AzureConnection::TerminalDisconnected(winrt::event_token const& token) noexcept - { - _disconnectHandlers.remove(token); - } - // Method description: // - ascribes to the ITerminalConnection interface // - creates the output thread (where we will do the authentication and actually connect to Azure) @@ -112,6 +90,8 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation THROW_LAST_ERROR_IF_NULL(_hOutputThread); + _StateChangedHandlers(*this, ConnectionState::Connecting); + _connected = true; } @@ -271,6 +251,8 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation if (!_closing.exchange(true)) { + _StateChangedHandlers(*this, ConnectionState::Closing); + _canProceed.notify_all(); if (_state == State::TermConnected) { @@ -282,6 +264,8 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // Tear down our output thread WaitForSingleObject(_hOutputThread.get(), INFINITE); _hOutputThread.reset(); + + _StateChangedHandlers(*this, ConnectionState::Closed); } } @@ -359,6 +343,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // We are connected, continuously read from the websocket until its closed case State::TermConnected: { + _StateChangedHandlers(*this, ConnectionState::Connected); while (true) { // Read from websocket @@ -374,7 +359,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation if (!_closing.load()) { _state = State::NoConnect; - _disconnectHandlers(); + _StateChangedHandlers(*this, ConnectionState::Closed); return S_FALSE; } break; @@ -394,7 +379,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation case State::NoConnect: { _WriteStringWithNewline(RS_(L"AzureInternetOrServerIssue")); - _disconnectHandlers(); + _StateChangedHandlers(*this, ConnectionState::Failed); return E_FAIL; } } diff --git a/src/cascadia/TerminalConnection/AzureConnection.h b/src/cascadia/TerminalConnection/AzureConnection.h index 9467357fb42..5e577bab92d 100644 --- a/src/cascadia/TerminalConnection/AzureConnection.h +++ b/src/cascadia/TerminalConnection/AzureConnection.h @@ -11,6 +11,8 @@ #include #include +#include "../cascadia/inc/cppwinrt_utils.h" + namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { struct AzureConnection : AzureConnectionT @@ -20,16 +22,15 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation winrt::event_token TerminalOutput(TerminalConnection::TerminalOutputEventArgs const& handler); void TerminalOutput(winrt::event_token const& token) noexcept; - winrt::event_token TerminalDisconnected(TerminalConnection::TerminalDisconnectedEventArgs const& handler); - void TerminalDisconnected(winrt::event_token const& token) noexcept; void Start(); void WriteInput(hstring const& data); void Resize(uint32_t rows, uint32_t columns); void Close(); + UNTYPED_EVENT(StateChanged, StateChangedEventArgs); + private: winrt::event _outputHandlers; - winrt::event _disconnectHandlers; uint32_t _initialRows{}; uint32_t _initialCols{}; diff --git a/src/cascadia/TerminalConnection/ConptyConnection.cpp b/src/cascadia/TerminalConnection/ConptyConnection.cpp index b1ffd4d7b81..63acad34ccf 100644 --- a/src/cascadia/TerminalConnection/ConptyConnection.cpp +++ b/src/cascadia/TerminalConnection/ConptyConnection.cpp @@ -160,16 +160,6 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _outputHandlers.remove(token); } - winrt::event_token ConptyConnection::TerminalDisconnected(Microsoft::Terminal::TerminalConnection::TerminalDisconnectedEventArgs const& handler) - { - return _disconnectHandlers.add(handler); - } - - void ConptyConnection::TerminalDisconnected(winrt::event_token const& token) noexcept - { - _disconnectHandlers.remove(token); - } - void ConptyConnection::Start() try { @@ -213,7 +203,9 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // TODO GH#2563 - signal a transition into failed state here! LOG_HR(hr); - _disconnectHandlers(); + winrt::hstring failureText{ wil::str_printf(L"[failed to spawn `%ls': %8.08x]\r\n", _commandline.c_str(), (unsigned int)hr) }; + _outputHandlers(failureText); + _transitionToState(ConnectionState::Failed); } void ConptyConnection::_ClientTerminated() noexcept @@ -224,8 +216,28 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation return; } - // TODO GH#2563 - get the exit code from the process and see whether it was a failing one. - _disconnectHandlers(); + DWORD exitCode{ 0 }; + GetExitCodeProcess(_piClient.hProcess, &exitCode); + if (exitCode == 0) + { + _transitionToState(ConnectionState::Closed); + } + else + { + winrt::hstring failureText{ wil::str_printf(L"\r\n\x1b[1m[took a long walk off a short pier: %8.08llx]\r\n", (unsigned long long)exitCode) }; + _outputHandlers(failureText); + _transitionToState(ConnectionState::Failed); + } + } + + void ConptyConnection::_transitionToState(const ConnectionState state) noexcept + { + // only allow movement up the satte gradient + // old state must be < current. + if (_state.exchange(state) < state) + { + _StateChangedHandlers(*this, _state); + } } void ConptyConnection::WriteInput(hstring const& data) @@ -263,6 +275,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation if (!_closing.exchange(true)) { + _transitionToState(ConnectionState::Closing); _clientExitWait.reset(); // immediately stop waiting for the client to exit. _hPC.reset(); @@ -275,6 +288,8 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation LOG_LAST_ERROR_IF(WAIT_FAILED == WaitForSingleObject(_hOutputThread.get(), INFINITE)); _hOutputThread.reset(); + _transitionToState(ConnectionState::Closed); + // Wait for the client to terminate. LOG_LAST_ERROR_IF(WAIT_FAILED == WaitForSingleObject(_piClient.hProcess, INFINITE)); _piClient.reset(); @@ -298,7 +313,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation return 0; } - _disconnectHandlers(); + _transitionToState(ConnectionState::Failed); return (DWORD)-1; } diff --git a/src/cascadia/TerminalConnection/ConptyConnection.h b/src/cascadia/TerminalConnection/ConptyConnection.h index 7a0d5592758..484d8d03d52 100644 --- a/src/cascadia/TerminalConnection/ConptyConnection.h +++ b/src/cascadia/TerminalConnection/ConptyConnection.h @@ -1,9 +1,11 @@ -// Copyright (c) Microsoft Corporation. +// Copyright (c) Microsoft Corporation. // Licensed under the MIT license. #pragma once #include "ConptyConnection.g.h" +#include "../inc/cppwinrt_utils.h" + #include namespace wil @@ -20,8 +22,6 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation winrt::event_token TerminalOutput(TerminalConnection::TerminalOutputEventArgs const& handler); void TerminalOutput(winrt::event_token const& token) noexcept; - winrt::event_token TerminalDisconnected(TerminalConnection::TerminalDisconnectedEventArgs const& handler); - void TerminalDisconnected(winrt::event_token const& token) noexcept; void Start(); void WriteInput(hstring const& data); void Resize(uint32_t rows, uint32_t columns); @@ -29,12 +29,14 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation winrt::guid Guid() const noexcept; + UNTYPED_EVENT(StateChanged, StateChangedEventArgs); + private: HRESULT _LaunchAttachedClient() noexcept; void _ClientTerminated() noexcept; + void _transitionToState(const ConnectionState state) noexcept; winrt::event _outputHandlers; - winrt::event _disconnectHandlers; uint32_t _initialRows{}; uint32_t _initialCols{}; @@ -43,6 +45,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation hstring _startingTitle; guid _guid{}; // A unique session identifier for connected client + std::atomic _state{ ConnectionState::NotConnected }; bool _connected{}; std::atomic _closing{ false }; bool _recievedFirstByte{ false }; diff --git a/src/cascadia/TerminalConnection/EchoConnection.cpp b/src/cascadia/TerminalConnection/EchoConnection.cpp index 1df63ad8187..a75a1c12476 100644 --- a/src/cascadia/TerminalConnection/EchoConnection.cpp +++ b/src/cascadia/TerminalConnection/EchoConnection.cpp @@ -23,17 +23,6 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _outputHandlers.remove(token); } - winrt::event_token EchoConnection::TerminalDisconnected(TerminalConnection::TerminalDisconnectedEventArgs const& handler) - { - handler; - throw hresult_not_implemented(); - } - - void EchoConnection::TerminalDisconnected(winrt::event_token const& token) noexcept - { - token; - } - void EchoConnection::Start() { } diff --git a/src/cascadia/TerminalConnection/EchoConnection.h b/src/cascadia/TerminalConnection/EchoConnection.h index 0f60a56331a..3dfb3de948d 100644 --- a/src/cascadia/TerminalConnection/EchoConnection.h +++ b/src/cascadia/TerminalConnection/EchoConnection.h @@ -5,6 +5,8 @@ #include "EchoConnection.g.h" +#include "../cascadia/inc/cppwinrt_utils.h" + namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { struct EchoConnection : EchoConnectionT @@ -13,13 +15,13 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation winrt::event_token TerminalOutput(TerminalConnection::TerminalOutputEventArgs const& handler); void TerminalOutput(winrt::event_token const& token) noexcept; - winrt::event_token TerminalDisconnected(TerminalConnection::TerminalDisconnectedEventArgs const& handler); - void TerminalDisconnected(winrt::event_token const& token) noexcept; void Start(); void WriteInput(hstring const& data); void Resize(uint32_t rows, uint32_t columns); void Close(); + UNTYPED_EVENT(StateChanged, StateChangedEventArgs); + private: winrt::event _outputHandlers; }; diff --git a/src/cascadia/TerminalConnection/ITerminalConnection.idl b/src/cascadia/TerminalConnection/ITerminalConnection.idl index 679623629fe..1a2b87a7525 100644 --- a/src/cascadia/TerminalConnection/ITerminalConnection.idl +++ b/src/cascadia/TerminalConnection/ITerminalConnection.idl @@ -3,13 +3,32 @@ namespace Microsoft.Terminal.TerminalConnection { + enum ConnectionState + { + NotConnected = 0, + Connecting, + Connected, + Closing, + Closed, + Failed + }; + delegate void TerminalOutputEventArgs(String output); - delegate void TerminalDisconnectedEventArgs(); + delegate void StateChangedEventArgs(ITerminalConnection connection, ConnectionState newState); + + /* + [default_interface] runtimeclass StateChangedEventArgs + { + ConnectionState State { get; } + } + */ interface ITerminalConnection { event TerminalOutputEventArgs TerminalOutput; - event TerminalDisconnectedEventArgs TerminalDisconnected; + + event StateChangedEventArgs StateChanged; + //ConnectionState State { get; }; void Start(); void WriteInput(String data); diff --git a/src/cascadia/TerminalControl/TermControl.cpp b/src/cascadia/TerminalControl/TermControl.cpp index d2ca39067c8..d012d95dfc0 100644 --- a/src/cascadia/TerminalControl/TermControl.cpp +++ b/src/cascadia/TerminalControl/TermControl.cpp @@ -153,8 +153,13 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation // DON'T CALL _InitializeTerminal here - wait until the swap chain is loaded to do that. // Subscribe to the connection's disconnected event and call our connection closed handlers. - _connection.TerminalDisconnected([=]() { - _connectionClosedHandlers(); + _connection.StateChanged([=](auto&& /*s*/, auto&& v) { + auto ff = wil::str_printf(L"CONNECTION TRANSITIONED TO STATE %d\n", v); + OutputDebugStringW(ff.c_str()); + if (v == TerminalConnection::ConnectionState::Closed) + { + _connectionClosedHandlers(); + } }); } diff --git a/src/cascadia/inc/cppwinrt_utils.h b/src/cascadia/inc/cppwinrt_utils.h index 9329c1cdef8..88eec43372a 100644 --- a/src/cascadia/inc/cppwinrt_utils.h +++ b/src/cascadia/inc/cppwinrt_utils.h @@ -71,6 +71,14 @@ public: private: \ winrt::event> _##name##Handlers; +#define UNTYPED_EVENT(name, args) \ +public: \ + winrt::event_token name(args const& handler) { return _##name##Handlers.add(handler); } \ + void name(winrt::event_token const& token) noexcept { _##name##Handlers.remove(token); } \ + \ +private: \ + winrt::event _##name##Handlers; + // This is a helper macro for both declaring the signature and body of an event // which is exposed by one class, but actually handled entirely by one of the // class's members. This type of event could be considered "forwarded" or From f3a010b21bd3d54920f4faf0bad6be1d11456896 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Wed, 13 Nov 2019 17:54:13 -0800 Subject: [PATCH 02/17] Gotta admit, I don't love it. --- .../TerminalConnection/AzureConnection.cpp | 149 +++++++++--------- .../TerminalConnection/AzureConnection.h | 17 +- .../TerminalConnection/ConptyConnection.cpp | 93 +++++------ .../TerminalConnection/ConptyConnection.h | 15 +- .../TerminalConnection/EchoConnection.cpp | 12 +- .../TerminalConnection/EchoConnection.h | 8 +- .../ITerminalConnection.idl | 21 +-- .../Resources/en-US/Resources.resw | 11 +- src/cascadia/TerminalControl/TermControl.cpp | 6 +- src/cascadia/TerminalControl/TermControl.h | 1 + src/cascadia/inc/cppwinrt_utils.h | 8 +- 11 files changed, 166 insertions(+), 175 deletions(-) diff --git a/src/cascadia/TerminalConnection/AzureConnection.cpp b/src/cascadia/TerminalConnection/AzureConnection.cpp index 7066e3b02e4..55f7db17684 100644 --- a/src/cascadia/TerminalConnection/AzureConnection.cpp +++ b/src/cascadia/TerminalConnection/AzureConnection.cpp @@ -49,29 +49,23 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // - str: the string to write. void AzureConnection::_WriteStringWithNewline(const winrt::hstring& str) { - _outputHandlers(str + L"\r\n"); + _TerminalOutputHandlers(str + L"\r\n"); } - // Method description: - // - ascribes to the ITerminalConnection interface - // - registers an output event handler - // Arguments: - // - the handler - // Return value: - // - the event token for the handler - winrt::event_token AzureConnection::TerminalOutput(Microsoft::Terminal::TerminalConnection::TerminalOutputEventArgs const& handler) + bool AzureConnection::_transitionToState(const ConnectionState state) noexcept { - return _outputHandlers.add(handler); - } - - // Method description: - // - ascribes to the ITerminalConnection interface - // - revokes an output event handler - // Arguments: - // - the event token for the handler - void AzureConnection::TerminalOutput(winrt::event_token const& token) noexcept - { - _outputHandlers.remove(token); + { + std::lock_guard stateLock{ _commonMutex }; + // only allow movement up the state gradient + if (state < _connectionState) + { + return false; + } + _connectionState = state; + } + // Dispatch the event outside of lock. + _StateChangedHandlers(*this, nullptr); + return true; } // Method description: @@ -90,9 +84,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation THROW_LAST_ERROR_IF_NULL(_hOutputThread); - _StateChangedHandlers(*this, ConnectionState::Connecting); - - _connected = true; + _transitionToState(ConnectionState::Connecting); } // Method description: @@ -102,7 +94,8 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // the user's input void AzureConnection::WriteInput(hstring const& data) { - if (!_connected || _closing.load()) + // We read input while connected AND connecting. + if (_connectionState != ConnectionState::Connected && _connectionState != ConnectionState::Connecting) { return; } @@ -111,7 +104,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation switch (_state) { // The user has stored connection settings, let them choose one of them, create a new one or remove all stored ones - case State::AccessStored: + case AzureState::AccessStored: { const auto s = winrt::to_string(data); int storeNum = -1; @@ -152,7 +145,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation return; } // The user has multiple tenants in their Azure account, let them choose one of them - case State::TenantChoice: + case AzureState::TenantChoice: { int tenantNum = -1; try @@ -175,7 +168,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation return; } // User has the option to save their connection settings for future logins - case State::StoreTokens: + case AzureState::StoreTokens: { std::lock_guard lg{ _commonMutex }; if (data == RS_(L"AzureUserEntry_Yes")) @@ -198,7 +191,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation return; } // We are connected, send user's input over the websocket - case State::TermConnected: + case AzureState::TermConnected: { websocket_outgoing_message msg; const auto str = winrt::to_string(data); @@ -218,12 +211,12 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // - the new rows/cols values void AzureConnection::Resize(uint32_t rows, uint32_t columns) { - if (!_connected || !(_state == State::TermConnected)) + if (_connectionState != ConnectionState::Connected) { _initialRows = rows; _initialCols = columns; } - else if (!_closing.load()) + else // We only transition to Connected when we've established the websocket. { // Initialize client http_client terminalClient(_cloudShellUri); @@ -244,28 +237,24 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // - closes the websocket connection and the output thread void AzureConnection::Close() { - if (!_connected) + if (_transitionToState(ConnectionState::Closing)) { - return; - } - - if (!_closing.exchange(true)) - { - _StateChangedHandlers(*this, ConnectionState::Closing); - _canProceed.notify_all(); - if (_state == State::TermConnected) + if (_state == AzureState::TermConnected) { // Close the websocket connection auto closedTask = _cloudShellSocket.close(); closedTask.wait(); } - // Tear down our output thread - WaitForSingleObject(_hOutputThread.get(), INFINITE); - _hOutputThread.reset(); + if (_hOutputThread) + { + // Tear down our output thread + WaitForSingleObject(_hOutputThread.get(), INFINITE); + _hOutputThread.reset(); + } - _StateChangedHandlers(*this, ConnectionState::Closed); + _transitionToState(ConnectionState::Closed); } } @@ -304,46 +293,52 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { while (true) { + if (_connectionState >= ConnectionState::Closing) + { + // If we enter a new state while closing, just bail. + return S_FALSE; + } + try { switch (_state) { // Initial state, check if the user has any stored connection settings and allow them to login with those // or allow them to login with a different account or allow them to remove the saved settings - case State::AccessStored: + case AzureState::AccessStored: { RETURN_IF_FAILED(_AccessHelper()); break; } // User has no saved connection settings or has opted to login with a different account // Azure authentication happens here - case State::DeviceFlow: + case AzureState::DeviceFlow: { RETURN_IF_FAILED(_DeviceFlowHelper()); break; } // User has multiple tenants in their Azure account, they need to choose which one to connect to - case State::TenantChoice: + case AzureState::TenantChoice: { RETURN_IF_FAILED(_TenantChoiceHelper()); break; } // Ask the user if they want to save these connection settings for future logins - case State::StoreTokens: + case AzureState::StoreTokens: { RETURN_IF_FAILED(_StoreHelper()); break; } // Connect to Azure, we only get here once we have everything we need (tenantID, accessToken, refreshToken) - case State::TermConnecting: + case AzureState::TermConnecting: { RETURN_IF_FAILED(_ConnectHelper()); break; } // We are connected, continuously read from the websocket until its closed - case State::TermConnected: + case AzureState::TermConnected: { - _StateChangedHandlers(*this, ConnectionState::Connected); + _transitionToState(ConnectionState::Connected); while (true) { // Read from websocket @@ -355,15 +350,15 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation } catch (...) { - // Websocket has been closed - if (!_closing.load()) + // Websocket has been closed; consider it a graceful exit? + // This should result in our termination. + if (_transitionToState(ConnectionState::Closed)) { - _state = State::NoConnect; - _StateChangedHandlers(*this, ConnectionState::Closed); + // End the output thread. return S_FALSE; } - break; } + auto msg = msgT.get(); auto msgStringTask = msg.extract_string(); auto msgString = msgStringTask.get(); @@ -372,21 +367,21 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation const auto hstr = winrt::to_hstring(msgString); // Pass the output to our registered event handlers - _outputHandlers(hstr); + _TerminalOutputHandlers(hstr); } return S_OK; } - case State::NoConnect: + case AzureState::NoConnect: { _WriteStringWithNewline(RS_(L"AzureInternetOrServerIssue")); - _StateChangedHandlers(*this, ConnectionState::Failed); + _transitionToState(ConnectionState::Failed); return E_FAIL; } } } catch (...) { - _state = State::NoConnect; + _state = AzureState::NoConnect; } } } @@ -410,7 +405,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation catch (...) { // No credentials are stored, so start the device flow - _state = State::DeviceFlow; + _state = AzureState::DeviceFlow; return S_FALSE; } _maxStored = 0; @@ -443,7 +438,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _WriteStringWithNewline(RS_(L"AzureOldCredentialsFlushedMessage")); } // No valid up-to-date credentials were found, so start the device flow - _state = State::DeviceFlow; + _state = AzureState::DeviceFlow; return S_FALSE; } @@ -453,10 +448,10 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation std::unique_lock storedLock{ _commonMutex }; _canProceed.wait(storedLock, [=]() { - return (_storedNumber >= 0 && _storedNumber < _maxStored) || _removeOrNew.has_value() || _closing.load(); + return (_storedNumber >= 0 && _storedNumber < _maxStored) || _removeOrNew.has_value() || _connectionState >= ConnectionState::Closing; }); // User might have closed the tab while we waited for input - if (_closing.load()) + if (_connectionState >= ConnectionState::Closing) { return E_FAIL; } @@ -464,13 +459,13 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { // User wants to remove the stored settings _RemoveCredentials(); - _state = State::DeviceFlow; + _state = AzureState::DeviceFlow; return S_OK; } else if (_removeOrNew.has_value() && !_removeOrNew.value()) { // User wants to login with a different account - _state = State::DeviceFlow; + _state = AzureState::DeviceFlow; return S_OK; } // User wants to login with one of the saved connection settings @@ -499,7 +494,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation } // We have everything we need, so go ahead and connect - _state = State::TermConnecting; + _state = AzureState::TermConnecting; return S_OK; } @@ -556,11 +551,11 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _refreshToken = refreshResponse.at(L"refresh_token").as_string(); _expiry = std::stoi(refreshResponse.at(L"expires_on").as_string()); - _state = State::StoreTokens; + _state = AzureState::StoreTokens; } else { - _state = State::TenantChoice; + _state = AzureState::TenantChoice; } return S_OK; } @@ -587,10 +582,10 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // Use a lock to wait for the user to input a valid number std::unique_lock tenantNumberLock{ _commonMutex }; _canProceed.wait(tenantNumberLock, [=]() { - return (_tenantNumber >= 0 && _tenantNumber < _maxSize) || _closing.load(); + return (_tenantNumber >= 0 && _tenantNumber < _maxSize) || _connectionState >= ConnectionState::Closing; }); // User might have closed the tab while we waited for input - if (_closing.load()) + if (_connectionState >= ConnectionState::Closing) { return E_FAIL; } @@ -604,7 +599,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _refreshToken = refreshResponse.at(L"refresh_token").as_string(); _expiry = std::stoi(refreshResponse.at(L"expires_on").as_string()); - _state = State::StoreTokens; + _state = AzureState::StoreTokens; return S_OK; } CATCH_RETURN(); @@ -621,10 +616,10 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // Wait for user input std::unique_lock storeLock{ _commonMutex }; _canProceed.wait(storeLock, [=]() { - return _store.has_value() || _closing.load(); + return _store.has_value() || _connectionState >= ConnectionState::Closing; }); // User might have closed the tab while we waited for input - if (_closing.load()) + if (_connectionState >= ConnectionState::Closing) { return E_FAIL; } @@ -636,7 +631,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _WriteStringWithNewline(RS_(L"AzureTokensStored")); } - _state = State::TermConnecting; + _state = AzureState::TermConnecting; return S_OK; } @@ -668,13 +663,13 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation const auto shellType = L"bash"; _WriteStringWithNewline(RS_(L"AzureRequestingTerminal")); const auto socketUri = _GetTerminal(shellType); - _outputHandlers(L"\r\n"); + _TerminalOutputHandlers(L"\r\n"); // Step 8: connecting to said terminal const auto connReqTask = _cloudShellSocket.connect(socketUri); connReqTask.wait(); - _state = State::TermConnected; + _state = AzureState::TermConnected; return S_OK; } @@ -744,7 +739,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation for (int count = 0; count < expiresIn / pollInterval; count++) { // User might close the tab while we wait for them to authenticate, this case handles that - if (_closing.load()) + if (_connectionState >= ConnectionState::Closing) { throw "Tab closed."; } diff --git a/src/cascadia/TerminalConnection/AzureConnection.h b/src/cascadia/TerminalConnection/AzureConnection.h index 5e577bab92d..995726527c4 100644 --- a/src/cascadia/TerminalConnection/AzureConnection.h +++ b/src/cascadia/TerminalConnection/AzureConnection.h @@ -20,17 +20,18 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation static bool IsAzureConnectionAvailable(); AzureConnection(const uint32_t rows, const uint32_t cols); - winrt::event_token TerminalOutput(TerminalConnection::TerminalOutputEventArgs const& handler); - void TerminalOutput(winrt::event_token const& token) noexcept; void Start(); void WriteInput(hstring const& data); void Resize(uint32_t rows, uint32_t columns); void Close(); - UNTYPED_EVENT(StateChanged, StateChangedEventArgs); + ConnectionState State() const noexcept { return _connectionState; } + + WINRT_CALLBACK(TerminalOutput, TerminalOutputHandler); + TYPED_EVENT(StateChanged, ITerminalConnection, IInspectable); private: - winrt::event _outputHandlers; + bool _transitionToState(const ConnectionState state) noexcept; uint32_t _initialRows{}; uint32_t _initialCols{}; @@ -41,7 +42,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation std::condition_variable _canProceed; std::mutex _commonMutex; - enum class State + enum class AzureState { AccessStored, DeviceFlow, @@ -52,14 +53,12 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation NoConnect }; - State _state{ State::AccessStored }; + AzureState _state{ AzureState::AccessStored }; + std::atomic _connectionState{ ConnectionState::NotConnected }; std::optional _store; std::optional _removeOrNew; - bool _connected{}; - std::atomic _closing{ false }; - wil::unique_handle _hOutputThread; static DWORD WINAPI StaticOutputThreadProc(LPVOID lpParameter); diff --git a/src/cascadia/TerminalConnection/ConptyConnection.cpp b/src/cascadia/TerminalConnection/ConptyConnection.cpp index 63acad34ccf..23a4ec9fb14 100644 --- a/src/cascadia/TerminalConnection/ConptyConnection.cpp +++ b/src/cascadia/TerminalConnection/ConptyConnection.cpp @@ -11,6 +11,7 @@ #include "../../types/inc/Utils.hpp" #include "../../types/inc/Environment.hpp" #include "../../types/inc/UTF8OutPipeReader.hpp" +#include "LibraryResources.h" using namespace ::Microsoft::Console; @@ -150,16 +151,6 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation return _guid; } - winrt::event_token ConptyConnection::TerminalOutput(Microsoft::Terminal::TerminalConnection::TerminalOutputEventArgs const& handler) - { - return _outputHandlers.add(handler); - } - - void ConptyConnection::TerminalOutput(winrt::event_token const& token) noexcept - { - _outputHandlers.remove(token); - } - void ConptyConnection::Start() try { @@ -195,54 +186,64 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation SetThreadpoolWait(_clientExitWait.get(), _piClient.hProcess, nullptr); - _connected = true; + _transitionToState(ConnectionState::Connected); } catch (...) { const auto hr = wil::ResultFromCaughtException(); - // TODO GH#2563 - signal a transition into failed state here! - LOG_HR(hr); - winrt::hstring failureText{ wil::str_printf(L"[failed to spawn `%ls': %8.08x]\r\n", _commandline.c_str(), (unsigned int)hr) }; - _outputHandlers(failureText); + winrt::hstring failureText{ wil::str_printf(RS_(L"ProcessFailedToLaunch").c_str(), static_cast(hr), _commandline.c_str()) }; + _TerminalOutputHandlers(failureText); _transitionToState(ConnectionState::Failed); } void ConptyConnection::_ClientTerminated() noexcept { - if (_closing.load()) + if (_state >= ConnectionState::Closing) { - // This is okay, break out to kill the thread + // This termination was expected. return; } DWORD exitCode{ 0 }; GetExitCodeProcess(_piClient.hProcess, &exitCode); + if (exitCode == 0) { _transitionToState(ConnectionState::Closed); } else { - winrt::hstring failureText{ wil::str_printf(L"\r\n\x1b[1m[took a long walk off a short pier: %8.08llx]\r\n", (unsigned long long)exitCode) }; - _outputHandlers(failureText); + winrt::hstring failureText{ wil::str_printf(RS_(L"ProcessExitedUnexpectedly").c_str(), (unsigned int)exitCode) }; + _TerminalOutputHandlers(L"\r\n"); + _TerminalOutputHandlers(failureText); _transitionToState(ConnectionState::Failed); } + + // We don't need these any longer. + _piClient.reset(); + _hPC.reset(); } - void ConptyConnection::_transitionToState(const ConnectionState state) noexcept + bool ConptyConnection::_transitionToState(const ConnectionState state) noexcept { - // only allow movement up the satte gradient - // old state must be < current. - if (_state.exchange(state) < state) { - _StateChangedHandlers(*this, _state); + std::lock_guard stateLock{ _stateMutex }; + // only allow movement up the state gradient + if (state < _state) + { + return false; + } + _state = state; } + // Dispatch the event outside of lock. + _StateChangedHandlers(*this, nullptr); + return true; } void ConptyConnection::WriteInput(hstring const& data) { - if (!_connected || _closing.load()) + if (_state != ConnectionState::Connected) { return; } @@ -260,7 +261,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _initialRows = rows; _initialCols = columns; } - else if (!_closing.load()) + else if (_state == ConnectionState::Connected) { THROW_IF_FAILED(ConptyResizePseudoConsole(_hPC.get(), { Utils::ClampToShortMax(columns, 1), Utils::ClampToShortMax(rows, 1) })); } @@ -268,31 +269,31 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation void ConptyConnection::Close() { - if (!_connected) + if (_transitionToState(ConnectionState::Closing)) { - return; - } - - if (!_closing.exchange(true)) - { - _transitionToState(ConnectionState::Closing); _clientExitWait.reset(); // immediately stop waiting for the client to exit. - _hPC.reset(); + _hPC.reset(); // tear down the pseudoconsole (this is like clicking X on a console window) - _inPipe.reset(); + _inPipe.reset(); // break the pipes _outPipe.reset(); - // Tear down our output thread -- now that the output pipe was closed on the - // far side, we can run down our local reader. - LOG_LAST_ERROR_IF(WAIT_FAILED == WaitForSingleObject(_hOutputThread.get(), INFINITE)); - _hOutputThread.reset(); + if (_hOutputThread) + { + // Tear down our output thread -- now that the output pipe was closed on the + // far side, we can run down our local reader. + LOG_LAST_ERROR_IF(WAIT_FAILED == WaitForSingleObject(_hOutputThread.get(), INFINITE)); + _hOutputThread.reset(); + } + + if (_piClient.hProcess) + { + // Wait for the client to terminate (which it should do successfully) + LOG_LAST_ERROR_IF(WAIT_FAILED == WaitForSingleObject(_piClient.hProcess, INFINITE)); + _piClient.reset(); + } _transitionToState(ConnectionState::Closed); - - // Wait for the client to terminate. - LOG_LAST_ERROR_IF(WAIT_FAILED == WaitForSingleObject(_piClient.hProcess, INFINITE)); - _piClient.reset(); } } @@ -307,9 +308,9 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation HRESULT result = pipeReader.Read(strView); if (FAILED(result) || result == S_FALSE) { - if (_closing.load()) + if (_state >= ConnectionState::Closing) { - // This is okay, break out to kill the thread + // This termination was expected. return 0; } @@ -341,7 +342,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation auto hstr{ winrt::to_hstring(strView) }; // Pass the output to our registered event handlers - _outputHandlers(hstr); + _TerminalOutputHandlers(hstr); } return 0; diff --git a/src/cascadia/TerminalConnection/ConptyConnection.h b/src/cascadia/TerminalConnection/ConptyConnection.h index 484d8d03d52..75239451ec4 100644 --- a/src/cascadia/TerminalConnection/ConptyConnection.h +++ b/src/cascadia/TerminalConnection/ConptyConnection.h @@ -20,8 +20,6 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { ConptyConnection(const hstring& cmdline, const hstring& startingDirectory, const hstring& startingTitle, const uint32_t rows, const uint32_t cols, const guid& guid); - winrt::event_token TerminalOutput(TerminalConnection::TerminalOutputEventArgs const& handler); - void TerminalOutput(winrt::event_token const& token) noexcept; void Start(); void WriteInput(hstring const& data); void Resize(uint32_t rows, uint32_t columns); @@ -29,14 +27,15 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation winrt::guid Guid() const noexcept; - UNTYPED_EVENT(StateChanged, StateChangedEventArgs); + ConnectionState State() const noexcept { return _state; } + + WINRT_CALLBACK(TerminalOutput, TerminalOutputHandler); + TYPED_EVENT(StateChanged, ITerminalConnection, IInspectable); private: HRESULT _LaunchAttachedClient() noexcept; void _ClientTerminated() noexcept; - void _transitionToState(const ConnectionState state) noexcept; - - winrt::event _outputHandlers; + bool _transitionToState(const ConnectionState state) noexcept; uint32_t _initialRows{}; uint32_t _initialCols{}; @@ -46,8 +45,8 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation guid _guid{}; // A unique session identifier for connected client std::atomic _state{ ConnectionState::NotConnected }; - bool _connected{}; - std::atomic _closing{ false }; + mutable std::mutex _stateMutex; + bool _recievedFirstByte{ false }; std::chrono::high_resolution_clock::time_point _startTime{}; diff --git a/src/cascadia/TerminalConnection/EchoConnection.cpp b/src/cascadia/TerminalConnection/EchoConnection.cpp index a75a1c12476..1b1803c8581 100644 --- a/src/cascadia/TerminalConnection/EchoConnection.cpp +++ b/src/cascadia/TerminalConnection/EchoConnection.cpp @@ -13,16 +13,6 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { } - winrt::event_token EchoConnection::TerminalOutput(TerminalConnection::TerminalOutputEventArgs const& handler) - { - return _outputHandlers.add(handler); - } - - void EchoConnection::TerminalOutput(winrt::event_token const& token) noexcept - { - _outputHandlers.remove(token); - } - void EchoConnection::Start() { } @@ -45,7 +35,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation prettyPrint << wch; } } - _outputHandlers(prettyPrint.str()); + _TerminalOutputHandlers(prettyPrint.str()); } void EchoConnection::Resize(uint32_t rows, uint32_t columns) diff --git a/src/cascadia/TerminalConnection/EchoConnection.h b/src/cascadia/TerminalConnection/EchoConnection.h index 3dfb3de948d..0a985224cf8 100644 --- a/src/cascadia/TerminalConnection/EchoConnection.h +++ b/src/cascadia/TerminalConnection/EchoConnection.h @@ -13,17 +13,15 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { EchoConnection(); - winrt::event_token TerminalOutput(TerminalConnection::TerminalOutputEventArgs const& handler); - void TerminalOutput(winrt::event_token const& token) noexcept; void Start(); void WriteInput(hstring const& data); void Resize(uint32_t rows, uint32_t columns); void Close(); - UNTYPED_EVENT(StateChanged, StateChangedEventArgs); + ConnectionState State() const noexcept { return ConnectionState::Connected; } - private: - winrt::event _outputHandlers; + WINRT_CALLBACK(TerminalOutput, TerminalOutputHandler); + TYPED_EVENT(StateChanged, ITerminalConnection, IInspectable); }; } diff --git a/src/cascadia/TerminalConnection/ITerminalConnection.idl b/src/cascadia/TerminalConnection/ITerminalConnection.idl index 1a2b87a7525..398724cd882 100644 --- a/src/cascadia/TerminalConnection/ITerminalConnection.idl +++ b/src/cascadia/TerminalConnection/ITerminalConnection.idl @@ -13,27 +13,18 @@ namespace Microsoft.Terminal.TerminalConnection Failed }; - delegate void TerminalOutputEventArgs(String output); - delegate void StateChangedEventArgs(ITerminalConnection connection, ConnectionState newState); - - /* - [default_interface] runtimeclass StateChangedEventArgs - { - ConnectionState State { get; } - } - */ + delegate void TerminalOutputHandler(String output); interface ITerminalConnection { - event TerminalOutputEventArgs TerminalOutput; - - event StateChangedEventArgs StateChanged; - //ConnectionState State { get; }; - void Start(); void WriteInput(String data); void Resize(UInt32 rows, UInt32 columns); void Close(); - }; + event TerminalOutputHandler TerminalOutput; + + event Windows.Foundation.TypedEventHandler StateChanged; + ConnectionState State { get; }; + }; } diff --git a/src/cascadia/TerminalConnection/Resources/en-US/Resources.resw b/src/cascadia/TerminalConnection/Resources/en-US/Resources.resw index 2a4ae126e64..0fac3cbb453 100644 --- a/src/cascadia/TerminalConnection/Resources/en-US/Resources.resw +++ b/src/cascadia/TerminalConnection/Resources/en-US/Resources.resw @@ -205,4 +205,13 @@ n The user must enter this character to signify NO - \ No newline at end of file + + [process terminated with exit code %u] + The first argument (%u) is the (positive) error code of the process. + + + [error 0x%x when launching `%ls'] + The first argument is the hexadecimal error code. The second is the process name. +If this resource spans multiple lines, it will not be displayed properly. Yeah. + + diff --git a/src/cascadia/TerminalControl/TermControl.cpp b/src/cascadia/TerminalControl/TermControl.cpp index d012d95dfc0..b02ac8d102a 100644 --- a/src/cascadia/TerminalControl/TermControl.cpp +++ b/src/cascadia/TerminalControl/TermControl.cpp @@ -153,7 +153,8 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation // DON'T CALL _InitializeTerminal here - wait until the swap chain is loaded to do that. // Subscribe to the connection's disconnected event and call our connection closed handlers. - _connection.StateChanged([=](auto&& /*s*/, auto&& v) { + _connectionStateChangedRevoker = _connection.StateChanged(winrt::auto_revoke, [=](auto&& s, auto&& /*v*/) { + auto v = s.State(); auto ff = wil::str_printf(L"CONNECTION TRANSITIONED TO STATE %d\n", v); OutputDebugStringW(ff.c_str()); if (v == TerminalConnection::ConnectionState::Closed) @@ -1538,8 +1539,9 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation { if (!_closing.exchange(true)) { - // Stop accepting new output before we disconnect everything. + // Stop accepting new output and state changes before we disconnect everything. _connection.TerminalOutput(_connectionOutputEventToken); + _connectionStateChangedRevoker.revoke(); // Clear out the cursor timer, so it doesn't trigger again on us once we're destructed. if (auto localCursorTimer{ std::exchange(_cursorTimer, std::nullopt) }) diff --git a/src/cascadia/TerminalControl/TermControl.h b/src/cascadia/TerminalControl/TermControl.h index 2f6afbe990d..69cbae1e270 100644 --- a/src/cascadia/TerminalControl/TermControl.h +++ b/src/cascadia/TerminalControl/TermControl.h @@ -102,6 +102,7 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation Windows::UI::Xaml::Controls::SwapChainPanel _swapChainPanel; Windows::UI::Xaml::Controls::Primitives::ScrollBar _scrollBar; event_token _connectionOutputEventToken; + TerminalConnection::ITerminalConnection::StateChanged_revoker _connectionStateChangedRevoker; std::unique_ptr<::Microsoft::Terminal::Core::Terminal> _terminal; diff --git a/src/cascadia/inc/cppwinrt_utils.h b/src/cascadia/inc/cppwinrt_utils.h index 88eec43372a..fc12881ae08 100644 --- a/src/cascadia/inc/cppwinrt_utils.h +++ b/src/cascadia/inc/cppwinrt_utils.h @@ -71,7 +71,13 @@ public: private: \ winrt::event> _##name##Handlers; -#define UNTYPED_EVENT(name, args) \ +// This is a helper macro for both declaring the signature of a callback (nee event) and +// defining the body. Winrt callbacks need a method for adding a delegate to the +// callback and removing the delegate. This macro will both declare the method +// signatures and define them both for you, because they don't really vary from +// event to event. +// Use this in a class's header if you have a "delegate" type in your IDL. +#define WINRT_CALLBACK(name, args) \ public: \ winrt::event_token name(args const& handler) { return _##name##Handlers.add(handler); } \ void name(winrt::event_token const& token) noexcept { _##name##Handlers.remove(token); } \ From 5a11dcf1e8d59d73d34e48d9ce118ad27a3ac610 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Thu, 14 Nov 2019 17:19:43 -0800 Subject: [PATCH 03/17] Add a connection state helper --- .../TerminalConnection/AzureConnection.cpp | 36 ++++-------- .../TerminalConnection/AzureConnection.h | 9 +-- .../ConnectionStateHolder.h | 57 +++++++++++++++++++ .../TerminalConnection/ConptyConnection.cpp | 24 ++------ .../TerminalConnection/ConptyConnection.h | 10 +--- 5 files changed, 75 insertions(+), 61 deletions(-) create mode 100644 src/cascadia/TerminalConnection/ConnectionStateHolder.h diff --git a/src/cascadia/TerminalConnection/AzureConnection.cpp b/src/cascadia/TerminalConnection/AzureConnection.cpp index 55f7db17684..449f998290b 100644 --- a/src/cascadia/TerminalConnection/AzureConnection.cpp +++ b/src/cascadia/TerminalConnection/AzureConnection.cpp @@ -52,22 +52,6 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _TerminalOutputHandlers(str + L"\r\n"); } - bool AzureConnection::_transitionToState(const ConnectionState state) noexcept - { - { - std::lock_guard stateLock{ _commonMutex }; - // only allow movement up the state gradient - if (state < _connectionState) - { - return false; - } - _connectionState = state; - } - // Dispatch the event outside of lock. - _StateChangedHandlers(*this, nullptr); - return true; - } - // Method description: // - ascribes to the ITerminalConnection interface // - creates the output thread (where we will do the authentication and actually connect to Azure) @@ -95,7 +79,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation void AzureConnection::WriteInput(hstring const& data) { // We read input while connected AND connecting. - if (_connectionState != ConnectionState::Connected && _connectionState != ConnectionState::Connecting) + if (!_isStateOneOf(ConnectionState::Connected, ConnectionState::Connecting)) { return; } @@ -211,7 +195,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // - the new rows/cols values void AzureConnection::Resize(uint32_t rows, uint32_t columns) { - if (_connectionState != ConnectionState::Connected) + if (!_isConnected()) { _initialRows = rows; _initialCols = columns; @@ -293,7 +277,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { while (true) { - if (_connectionState >= ConnectionState::Closing) + if (_isStateAtOrBeyond(ConnectionState::Closing)) { // If we enter a new state while closing, just bail. return S_FALSE; @@ -448,10 +432,10 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation std::unique_lock storedLock{ _commonMutex }; _canProceed.wait(storedLock, [=]() { - return (_storedNumber >= 0 && _storedNumber < _maxStored) || _removeOrNew.has_value() || _connectionState >= ConnectionState::Closing; + return (_storedNumber >= 0 && _storedNumber < _maxStored) || _removeOrNew.has_value() || _isStateAtOrBeyond(ConnectionState::Closing); }); // User might have closed the tab while we waited for input - if (_connectionState >= ConnectionState::Closing) + if (_isStateAtOrBeyond(ConnectionState::Closing)) { return E_FAIL; } @@ -582,10 +566,10 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // Use a lock to wait for the user to input a valid number std::unique_lock tenantNumberLock{ _commonMutex }; _canProceed.wait(tenantNumberLock, [=]() { - return (_tenantNumber >= 0 && _tenantNumber < _maxSize) || _connectionState >= ConnectionState::Closing; + return (_tenantNumber >= 0 && _tenantNumber < _maxSize) || _isStateAtOrBeyond(ConnectionState::Closing); }); // User might have closed the tab while we waited for input - if (_connectionState >= ConnectionState::Closing) + if (_isStateAtOrBeyond(ConnectionState::Closing)) { return E_FAIL; } @@ -616,10 +600,10 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation // Wait for user input std::unique_lock storeLock{ _commonMutex }; _canProceed.wait(storeLock, [=]() { - return _store.has_value() || _connectionState >= ConnectionState::Closing; + return _store.has_value() || _isStateAtOrBeyond(ConnectionState::Closing); }); // User might have closed the tab while we waited for input - if (_connectionState >= ConnectionState::Closing) + if (_isStateAtOrBeyond(ConnectionState::Closing)) { return E_FAIL; } @@ -739,7 +723,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation for (int count = 0; count < expiresIn / pollInterval; count++) { // User might close the tab while we wait for them to authenticate, this case handles that - if (_connectionState >= ConnectionState::Closing) + if (_isStateAtOrBeyond(ConnectionState::Closing)) { throw "Tab closed."; } diff --git a/src/cascadia/TerminalConnection/AzureConnection.h b/src/cascadia/TerminalConnection/AzureConnection.h index 995726527c4..f2a590c87cf 100644 --- a/src/cascadia/TerminalConnection/AzureConnection.h +++ b/src/cascadia/TerminalConnection/AzureConnection.h @@ -12,10 +12,11 @@ #include #include "../cascadia/inc/cppwinrt_utils.h" +#include "ConnectionStateHolder.h" namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { - struct AzureConnection : AzureConnectionT + struct AzureConnection : AzureConnectionT, ConnectionStateHolder { static bool IsAzureConnectionAvailable(); AzureConnection(const uint32_t rows, const uint32_t cols); @@ -25,14 +26,9 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation void Resize(uint32_t rows, uint32_t columns); void Close(); - ConnectionState State() const noexcept { return _connectionState; } - WINRT_CALLBACK(TerminalOutput, TerminalOutputHandler); - TYPED_EVENT(StateChanged, ITerminalConnection, IInspectable); private: - bool _transitionToState(const ConnectionState state) noexcept; - uint32_t _initialRows{}; uint32_t _initialCols{}; int _storedNumber{ -1 }; @@ -54,7 +50,6 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation }; AzureState _state{ AzureState::AccessStored }; - std::atomic _connectionState{ ConnectionState::NotConnected }; std::optional _store; std::optional _removeOrNew; diff --git a/src/cascadia/TerminalConnection/ConnectionStateHolder.h b/src/cascadia/TerminalConnection/ConnectionStateHolder.h new file mode 100644 index 00000000000..a53e26c6511 --- /dev/null +++ b/src/cascadia/TerminalConnection/ConnectionStateHolder.h @@ -0,0 +1,57 @@ +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT license. + +#include "../inc/cppwinrt_utils.h" + +namespace winrt::Microsoft::Terminal::TerminalConnection::implementation +{ + template + struct ConnectionStateHolder + { + public: + ConnectionState State() const noexcept { return _connectionState; } + TYPED_EVENT(StateChanged, ITerminalConnection, winrt::Windows::Foundation::IInspectable); + + protected: + bool _transitionToState(const ConnectionState state) noexcept + { + { + std::lock_guard stateLock{ _stateMutex }; + // only allow movement up the state gradient + if (state < _connectionState) + { + return false; + } + _connectionState = state; + } + // Dispatch the event outside of lock. + _StateChangedHandlers(*static_cast(this), nullptr); + return true; + } + + template + bool _isStateOneOf(Args&&... args) const noexcept + { + static_assert((... && std::is_same::value), "all queried connection states must be from the ConnectionState enum"); + std::lock_guard stateLock{ _stateMutex }; + // dark magic: this is a C++17 fold expression that expands into + // state == 1 || state == 2 || state == 3 ... + return (... || (_connectionState == args)); + } + + bool _isStateAtOrBeyond(const ConnectionState state) const noexcept + { + std::lock_guard stateLock{ _stateMutex }; + return _connectionState >= state; + } + + bool _isConnected() const noexcept + { + return _isStateOneOf(ConnectionState::Connected); + } + + private: + std::atomic _connectionState{ ConnectionState::NotConnected }; + mutable std::mutex _stateMutex; + }; +} diff --git a/src/cascadia/TerminalConnection/ConptyConnection.cpp b/src/cascadia/TerminalConnection/ConptyConnection.cpp index 23a4ec9fb14..af95c04d61f 100644 --- a/src/cascadia/TerminalConnection/ConptyConnection.cpp +++ b/src/cascadia/TerminalConnection/ConptyConnection.cpp @@ -199,7 +199,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation void ConptyConnection::_ClientTerminated() noexcept { - if (_state >= ConnectionState::Closing) + if (_isStateAtOrBeyond(ConnectionState::Closing)) { // This termination was expected. return; @@ -225,25 +225,9 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _hPC.reset(); } - bool ConptyConnection::_transitionToState(const ConnectionState state) noexcept - { - { - std::lock_guard stateLock{ _stateMutex }; - // only allow movement up the state gradient - if (state < _state) - { - return false; - } - _state = state; - } - // Dispatch the event outside of lock. - _StateChangedHandlers(*this, nullptr); - return true; - } - void ConptyConnection::WriteInput(hstring const& data) { - if (_state != ConnectionState::Connected) + if (!_isConnected()) { return; } @@ -261,7 +245,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _initialRows = rows; _initialCols = columns; } - else if (_state == ConnectionState::Connected) + else if (_isConnected()) { THROW_IF_FAILED(ConptyResizePseudoConsole(_hPC.get(), { Utils::ClampToShortMax(columns, 1), Utils::ClampToShortMax(rows, 1) })); } @@ -308,7 +292,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation HRESULT result = pipeReader.Read(strView); if (FAILED(result) || result == S_FALSE) { - if (_state >= ConnectionState::Closing) + if (_isStateAtOrBeyond(ConnectionState::Closing)) { // This termination was expected. return 0; diff --git a/src/cascadia/TerminalConnection/ConptyConnection.h b/src/cascadia/TerminalConnection/ConptyConnection.h index 75239451ec4..014c9618804 100644 --- a/src/cascadia/TerminalConnection/ConptyConnection.h +++ b/src/cascadia/TerminalConnection/ConptyConnection.h @@ -4,6 +4,7 @@ #pragma once #include "ConptyConnection.g.h" +#include "ConnectionStateHolder.h" #include "../inc/cppwinrt_utils.h" #include @@ -16,7 +17,7 @@ namespace wil namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { - struct ConptyConnection : ConptyConnectionT + struct ConptyConnection : ConptyConnectionT, ConnectionStateHolder { ConptyConnection(const hstring& cmdline, const hstring& startingDirectory, const hstring& startingTitle, const uint32_t rows, const uint32_t cols, const guid& guid); @@ -27,15 +28,11 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation winrt::guid Guid() const noexcept; - ConnectionState State() const noexcept { return _state; } - WINRT_CALLBACK(TerminalOutput, TerminalOutputHandler); - TYPED_EVENT(StateChanged, ITerminalConnection, IInspectable); private: HRESULT _LaunchAttachedClient() noexcept; void _ClientTerminated() noexcept; - bool _transitionToState(const ConnectionState state) noexcept; uint32_t _initialRows{}; uint32_t _initialCols{}; @@ -44,9 +41,6 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation hstring _startingTitle; guid _guid{}; // A unique session identifier for connected client - std::atomic _state{ ConnectionState::NotConnected }; - mutable std::mutex _stateMutex; - bool _recievedFirstByte{ false }; std::chrono::high_resolution_clock::time_point _startTime{}; From 11b557f7bdb559824a79ea6b6d7ec745bedaa24e Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 18 Nov 2019 15:40:00 -0800 Subject: [PATCH 04/17] xyxy --- src/cascadia/TerminalApp/AppLogic.cpp | 7 +++ src/cascadia/TerminalApp/AppLogic.h | 1 + .../TerminalApp/AzureCloudShellGenerator.cpp | 1 - src/cascadia/TerminalApp/CascadiaSettings.cpp | 16 ++++++ src/cascadia/TerminalApp/CascadiaSettings.h | 2 + src/cascadia/TerminalApp/Pane.cpp | 49 +++++++++++----- src/cascadia/TerminalApp/Pane.h | 6 +- src/cascadia/TerminalApp/Profile.cpp | 49 +++++++++++++--- src/cascadia/TerminalApp/Profile.h | 14 ++++- src/cascadia/TerminalApp/Tab.cpp | 5 +- src/cascadia/TerminalApp/Tab.h | 2 +- src/cascadia/TerminalApp/TerminalPage.cpp | 2 +- .../TerminalConnection/ConptyConnection.cpp | 56 +++++++++++++++---- .../TerminalConnection/ConptyConnection.h | 1 + .../Resources/en-US/Resources.resw | 10 ++-- src/cascadia/TerminalControl/TermControl.cpp | 27 +++------ src/cascadia/TerminalControl/TermControl.h | 6 +- src/cascadia/TerminalControl/TermControl.idl | 8 ++- .../TerminalSettings/IControlSettings.idl | 1 - .../TerminalSettings/TerminalSettings.cpp | 11 ---- .../TerminalSettings/terminalsettings.h | 3 - 21 files changed, 188 insertions(+), 89 deletions(-) diff --git a/src/cascadia/TerminalApp/AppLogic.cpp b/src/cascadia/TerminalApp/AppLogic.cpp index db739eefce8..b3e8c34b093 100644 --- a/src/cascadia/TerminalApp/AppLogic.cpp +++ b/src/cascadia/TerminalApp/AppLogic.cpp @@ -592,6 +592,13 @@ namespace winrt::TerminalApp::implementation }); } + // Method Description: + // - Returns a pointer to the global shared settings. + [[nodiscard]] std::shared_ptr<::TerminalApp::CascadiaSettings> AppLogic::GetSettings() const noexcept + { + return _settings; + } + // Method Description: // - Update the current theme of the application. This will trigger our // RequestedThemeChanged event, to have our host change the theme of the diff --git a/src/cascadia/TerminalApp/AppLogic.h b/src/cascadia/TerminalApp/AppLogic.h index 6f90251ad05..06890380519 100644 --- a/src/cascadia/TerminalApp/AppLogic.h +++ b/src/cascadia/TerminalApp/AppLogic.h @@ -29,6 +29,7 @@ namespace winrt::TerminalApp::implementation void Create(); void LoadSettings(); + [[nodiscard]] std::shared_ptr<::TerminalApp::CascadiaSettings> GetSettings() const noexcept; Windows::Foundation::Point GetLaunchDimensions(uint32_t dpi); winrt::Windows::Foundation::Point GetLaunchInitialPositions(int32_t defaultInitialX, int32_t defaultInitialY); diff --git a/src/cascadia/TerminalApp/AzureCloudShellGenerator.cpp b/src/cascadia/TerminalApp/AzureCloudShellGenerator.cpp index 0414322ba06..b9e76dce7f0 100644 --- a/src/cascadia/TerminalApp/AzureCloudShellGenerator.cpp +++ b/src/cascadia/TerminalApp/AzureCloudShellGenerator.cpp @@ -39,7 +39,6 @@ std::vector AzureCloudShellGenerator::GenerateProfiles() azureCloudShellProfile.SetColorScheme({ L"Vintage" }); azureCloudShellProfile.SetAcrylicOpacity(0.6); azureCloudShellProfile.SetUseAcrylic(true); - azureCloudShellProfile.SetCloseOnExit(false); azureCloudShellProfile.SetConnectionType(AzureConnectionType); profiles.emplace_back(azureCloudShellProfile); } diff --git a/src/cascadia/TerminalApp/CascadiaSettings.cpp b/src/cascadia/TerminalApp/CascadiaSettings.cpp index d7046d9045f..ff861ab3e2b 100644 --- a/src/cascadia/TerminalApp/CascadiaSettings.cpp +++ b/src/cascadia/TerminalApp/CascadiaSettings.cpp @@ -9,6 +9,7 @@ #include "CascadiaSettings.h" #include "../../types/inc/utils.hpp" #include "../../inc/DefaultSettings.h" +#include "AppLogic.h" #include "Utils.h" #include "PowershellCoreProfileGenerator.h" @@ -26,6 +27,21 @@ static constexpr std::wstring_view PACKAGED_PROFILE_ICON_PATH{ L"ms-appx:///Prof static constexpr std::wstring_view PACKAGED_PROFILE_ICON_EXTENSION{ L".png" }; static constexpr std::wstring_view DEFAULT_LINUX_ICON_GUID{ L"{9acb9455-ca41-5af7-950f-6bca1bc9722f}" }; +// Method Description: +// - Returns the settings currently in use by the entire Terminal application. +// Throws: +// - HR E_INVALIDARG if the app isn't up and running. +const CascadiaSettings& CascadiaSettings::GetCurrentAppSettings() +{ + auto currentXamlApp{ winrt::Windows::UI::Xaml::Application::Current().as() }; + THROW_HR_IF_NULL(E_INVALIDARG, currentXamlApp); + + auto appLogic = winrt::get_self(currentXamlApp.Logic()); + THROW_HR_IF_NULL(E_INVALIDARG, appLogic); + + return *(appLogic->GetSettings()); +} + CascadiaSettings::CascadiaSettings() : CascadiaSettings(true) { diff --git a/src/cascadia/TerminalApp/CascadiaSettings.h b/src/cascadia/TerminalApp/CascadiaSettings.h index fde8177507c..a22a6df64e7 100644 --- a/src/cascadia/TerminalApp/CascadiaSettings.h +++ b/src/cascadia/TerminalApp/CascadiaSettings.h @@ -49,6 +49,8 @@ class TerminalApp::CascadiaSettings final static std::unique_ptr LoadDefaults(); static std::unique_ptr LoadAll(); + static const CascadiaSettings& GetCurrentAppSettings(); + winrt::Microsoft::Terminal::Settings::TerminalSettings MakeSettings(std::optional profileGuid) const; GlobalAppSettings& GlobalSettings(); diff --git a/src/cascadia/TerminalApp/Pane.cpp b/src/cascadia/TerminalApp/Pane.cpp index a0d3307898b..81a9c397707 100644 --- a/src/cascadia/TerminalApp/Pane.cpp +++ b/src/cascadia/TerminalApp/Pane.cpp @@ -3,6 +3,10 @@ #include "pch.h" #include "Pane.h" +#include "Profile.h" +#include "CascadiaSettings.h" + +#include "winrt/Microsoft.Terminal.TerminalConnection.h" using namespace winrt::Windows::Foundation; using namespace winrt::Windows::UI; @@ -11,7 +15,9 @@ using namespace winrt::Windows::UI::Core; using namespace winrt::Windows::UI::Xaml::Media; using namespace winrt::Microsoft::Terminal::Settings; using namespace winrt::Microsoft::Terminal::TerminalControl; +using namespace winrt::Microsoft::Terminal::TerminalConnection; using namespace winrt::TerminalApp; +using namespace TerminalApp; static const int PaneBorderSize = 2; static const int CombinedPaneBorderSize = 2 * PaneBorderSize; @@ -28,7 +34,7 @@ Pane::Pane(const GUID& profile, const TermControl& control, const bool lastFocus _root.Children().Append(_border); _border.Child(_control); - _connectionClosedToken = _control.ConnectionClosed({ this, &Pane::_ControlClosedHandler }); + _connectionStateChangedToken = _control.ConnectionStateChanged({ this, &Pane::_ControlConnectionStateChangedHandler }); // On the first Pane's creation, lookup resources we'll use to theme the // Pane, including the brushed to use for the focused/unfocused border @@ -302,7 +308,7 @@ bool Pane::NavigateFocus(const Direction& direction) // - // Return Value: // - -void Pane::_ControlClosedHandler() +void Pane::_ControlConnectionStateChangedHandler(const TermControl& /*sender*/, const winrt::Windows::Foundation::IInspectable& /*args*/) { std::unique_lock lock{ _createCloseLock }; // It's possible that this event handler started being executed, then before @@ -317,10 +323,24 @@ void Pane::_ControlClosedHandler() return; } - if (_control.ShouldCloseOnExit()) + const auto newConnectionState = _control.ConnectionState(); + + if (newConnectionState < ConnectionState::Closed) + { + // Pane doesn't care if the connection isn't entering a terminal state. + return; + } + + const auto& settings = CascadiaSettings::GetCurrentAppSettings(); + auto paneProfile = settings.FindProfile(_profile.value()); + if (paneProfile) { - // Fire our Closed event to tell our parent that we should be removed. - _closedHandlers(); + auto mode = paneProfile->GetCloseOnExitMode(); + if ((mode == CloseOnExitMode::Always) || + (mode == CloseOnExitMode::Graceful && newConnectionState == ConnectionState::Closed)) + { + _ClosedHandlers(nullptr, nullptr); + } } } @@ -333,7 +353,7 @@ void Pane::_ControlClosedHandler() void Pane::Close() { // Fire our Closed event to tell our parent that we should be removed. - _closedHandlers(); + _ClosedHandlers(nullptr, nullptr); } // Method Description: @@ -570,7 +590,7 @@ void Pane::_CloseChild(const bool closeFirst) _profile = remainingChild->_profile; // Add our new event handler before revoking the old one. - _connectionClosedToken = _control.ConnectionClosed({ this, &Pane::_ControlClosedHandler }); + _connectionStateChangedToken = _control.ConnectionStateChanged({ this, &Pane::_ControlConnectionStateChangedHandler }); // Revoke the old event handlers. Remove both the handlers for the panes // themselves closing, and remove their handlers for their controls @@ -578,8 +598,8 @@ void Pane::_CloseChild(const bool closeFirst) // they'll trigger only our event handler for the control's close. _firstChild->Closed(_firstClosedToken); _secondChild->Closed(_secondClosedToken); - closedChild->_control.ConnectionClosed(closedChild->_connectionClosedToken); - remainingChild->_control.ConnectionClosed(remainingChild->_connectionClosedToken); + closedChild->_control.ConnectionStateChanged(closedChild->_connectionStateChangedToken); + remainingChild->_control.ConnectionStateChanged(remainingChild->_connectionStateChangedToken); // If either of our children was focused, we want to take that focus from // them. @@ -659,7 +679,7 @@ void Pane::_CloseChild(const bool closeFirst) // Revoke event handlers on old panes and controls oldFirst->Closed(oldFirstToken); oldSecond->Closed(oldSecondToken); - closedChild->_control.ConnectionClosed(closedChild->_connectionClosedToken); + closedChild->_control.ConnectionStateChanged(closedChild->_connectionStateChangedToken); // Reset our UI: _root.Children().Clear(); @@ -720,13 +740,13 @@ void Pane::_CloseChild(const bool closeFirst) // - void Pane::_SetupChildCloseHandlers() { - _firstClosedToken = _firstChild->Closed([this]() { + _firstClosedToken = _firstChild->Closed([this](auto&& /*s*/, auto&& /*e*/) { _root.Dispatcher().RunAsync(CoreDispatcherPriority::Normal, [=]() { _CloseChild(true); }); }); - _secondClosedToken = _secondChild->Closed([this]() { + _secondClosedToken = _secondChild->Closed([this](auto&& /*s*/, auto&& /*e*/) { _root.Dispatcher().RunAsync(CoreDispatcherPriority::Normal, [=]() { _CloseChild(false); }); @@ -965,8 +985,8 @@ std::pair, std::shared_ptr> Pane::_Split(SplitState std::unique_lock lock{ _createCloseLock }; // revoke our handler - the child will take care of the control now. - _control.ConnectionClosed(_connectionClosedToken); - _connectionClosedToken.value = 0; + _control.ConnectionStateChanged(_connectionStateChangedToken); + _connectionStateChangedToken.value = 0; // Remove our old GotFocus handler from the control. We don't what the // control telling us that it's now focused, we want it telling its new @@ -1124,5 +1144,4 @@ void Pane::_SetupResources() } } -DEFINE_EVENT(Pane, Closed, _closedHandlers, ConnectionClosedEventArgs); DEFINE_EVENT(Pane, GotFocus, _GotFocusHandlers, winrt::delegate>); diff --git a/src/cascadia/TerminalApp/Pane.h b/src/cascadia/TerminalApp/Pane.h index 233216e27f1..3eec4df00ac 100644 --- a/src/cascadia/TerminalApp/Pane.h +++ b/src/cascadia/TerminalApp/Pane.h @@ -71,7 +71,7 @@ class Pane : public std::enable_shared_from_this void Close(); - DECLARE_EVENT(Closed, _closedHandlers, winrt::Microsoft::Terminal::TerminalControl::ConnectionClosedEventArgs); + WINRT_CALLBACK(Closed, winrt::Windows::Foundation::EventHandler); DECLARE_EVENT(GotFocus, _GotFocusHandlers, winrt::delegate>); private: @@ -89,7 +89,7 @@ class Pane : public std::enable_shared_from_this bool _lastActive{ false }; std::optional _profile{ std::nullopt }; - winrt::event_token _connectionClosedToken{ 0 }; + winrt::event_token _connectionStateChangedToken{ 0 }; winrt::event_token _firstClosedToken{ 0 }; winrt::event_token _secondClosedToken{ 0 }; @@ -119,7 +119,7 @@ class Pane : public std::enable_shared_from_this void _CloseChild(const bool closeFirst); void _FocusFirstChild(); - void _ControlClosedHandler(); + void _ControlConnectionStateChangedHandler(const winrt::Microsoft::Terminal::TerminalControl::TermControl& sender, const winrt::Windows::Foundation::IInspectable& /*args*/); std::pair _GetPaneSizes(const float& fullSize); diff --git a/src/cascadia/TerminalApp/Profile.cpp b/src/cascadia/TerminalApp/Profile.cpp index 70165a87c50..0d4f95c7a1e 100644 --- a/src/cascadia/TerminalApp/Profile.cpp +++ b/src/cascadia/TerminalApp/Profile.cpp @@ -49,6 +49,11 @@ static constexpr std::string_view BackgroundImageOpacityKey{ "backgroundImageOpa static constexpr std::string_view BackgroundImageStretchModeKey{ "backgroundImageStretchMode" }; static constexpr std::string_view BackgroundImageAlignmentKey{ "backgroundImageAlignment" }; +// Possible values for closeOnExit +static constexpr std::string_view CloseOnExitAlways{ "always" }; +static constexpr std::string_view CloseOnExitGraceful{ "graceful" }; +static constexpr std::string_view CloseOnExitNever{ "never" }; + // Possible values for Scrollbar state static constexpr std::wstring_view AlwaysVisible{ L"visible" }; static constexpr std::wstring_view AlwaysHide{ L"hidden" }; @@ -107,7 +112,7 @@ Profile::Profile(const std::optional& guid) : _acrylicTransparency{ 0.5 }, _useAcrylic{ false }, _scrollbarState{}, - _closeOnExit{ true }, + _closeOnExitMode{ CloseOnExitMode::Graceful }, _padding{ DEFAULT_PADDING }, _icon{}, _backgroundImage{}, @@ -168,7 +173,6 @@ TerminalSettings Profile::CreateTerminalSettings(const std::unordered_map ParseImageAlignment(const std::string_view imageAlignment); static std::tuple _ConvertJsonToAlignment(const Json::Value& json); + static CloseOnExitMode ParseCloseOnExitMode(const Json::Value& json); static std::string_view SerializeImageAlignment(const std::tuple imageAlignment); static winrt::Microsoft::Terminal::Settings::CursorStyle _ParseCursorShape(const std::wstring& cursorShapeString); @@ -141,7 +149,7 @@ class TerminalApp::Profile final std::optional> _backgroundImageAlignment; std::optional _scrollbarState; - bool _closeOnExit; + CloseOnExitMode _closeOnExitMode; std::wstring _padding; std::optional _icon; diff --git a/src/cascadia/TerminalApp/Tab.cpp b/src/cascadia/TerminalApp/Tab.cpp index 93ce7a6a50f..6969c1c4fed 100644 --- a/src/cascadia/TerminalApp/Tab.cpp +++ b/src/cascadia/TerminalApp/Tab.cpp @@ -19,8 +19,8 @@ Tab::Tab(const GUID& profile, const TermControl& control) { _rootPane = std::make_shared(profile, control, true); - _rootPane->Closed([=]() { - _closedHandlers(); + _rootPane->Closed([=](auto&& /*s*/, auto&& /*e*/) { + _ClosedHandlers(nullptr, nullptr); }); _activePane = _rootPane; @@ -335,5 +335,4 @@ void Tab::_AttachEventHandlersToPane(std::shared_ptr pane) }); } -DEFINE_EVENT(Tab, Closed, _closedHandlers, ConnectionClosedEventArgs); DEFINE_EVENT(Tab, ActivePaneChanged, _ActivePaneChangedHandlers, winrt::delegate<>); diff --git a/src/cascadia/TerminalApp/Tab.h b/src/cascadia/TerminalApp/Tab.h index fffb13926b8..336372a1d5f 100644 --- a/src/cascadia/TerminalApp/Tab.h +++ b/src/cascadia/TerminalApp/Tab.h @@ -35,7 +35,7 @@ class Tab void ClosePane(); - DECLARE_EVENT(Closed, _closedHandlers, winrt::Microsoft::Terminal::TerminalControl::ConnectionClosedEventArgs); + WINRT_CALLBACK(Closed, winrt::Windows::Foundation::EventHandler); DECLARE_EVENT(ActivePaneChanged, _ActivePaneChangedHandlers, winrt::delegate<>); private: diff --git a/src/cascadia/TerminalApp/TerminalPage.cpp b/src/cascadia/TerminalApp/TerminalPage.cpp index 5032ca7c241..b4a19bfdcfd 100644 --- a/src/cascadia/TerminalApp/TerminalPage.cpp +++ b/src/cascadia/TerminalApp/TerminalPage.cpp @@ -449,7 +449,7 @@ namespace winrt::TerminalApp::implementation tabViewItem.PointerPressed({ this, &TerminalPage::_OnTabClick }); // When the tab is closed, remove it from our list of tabs. - newTab->Closed([tabViewItem, this]() { + newTab->Closed([tabViewItem, this](auto&& /*s*/, auto&& /*e*/) { _tabView.Dispatcher().RunAsync(CoreDispatcherPriority::Normal, [tabViewItem, this]() { _RemoveTabViewItem(tabViewItem); }); diff --git a/src/cascadia/TerminalConnection/ConptyConnection.cpp b/src/cascadia/TerminalConnection/ConptyConnection.cpp index af95c04d61f..10733d52339 100644 --- a/src/cascadia/TerminalConnection/ConptyConnection.cpp +++ b/src/cascadia/TerminalConnection/ConptyConnection.cpp @@ -15,6 +15,21 @@ using namespace ::Microsoft::Console; +// Notes: +// There is a number of ways that the Conpty connection can be terminated (voluntarily or not): +// 1. The connection is Close()d +// 2. The pseudoconsole or process cannot be spawned during Start() +// 3. The client process exits with a code. +// (Successful (0) or any other code) +// 4. The read handle is terminated. +// (This usually happens when the pseudoconsole host crashes.) +// In each of these termination scenarios, we need to be mindful of tripping the others. +// Closing the pseudoconsole in response to the client exiting (3) can trigger (4). +// Close() (1) will cause the automatic triggering of (3) and (4). +// In a lot of cases, we use the connection state to stop "flapping." +// +// To figure out where we handle these, search for comments containing "EXIT POINT" + namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { // Function Description: @@ -190,11 +205,26 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation } catch (...) { + // EXIT POINT const auto hr = wil::ResultFromCaughtException(); winrt::hstring failureText{ wil::str_printf(RS_(L"ProcessFailedToLaunch").c_str(), static_cast(hr), _commandline.c_str()) }; _TerminalOutputHandlers(failureText); _transitionToState(ConnectionState::Failed); + + // Tear down any state we may have accumulated. + _hPC.reset(); + } + + void ConptyConnection::_indicateExitWithStatus(unsigned int status) noexcept + { + try + { + winrt::hstring exitText{ wil::str_printf(RS_(L"ProcessExited").c_str(), (unsigned int)status) }; + _TerminalOutputHandlers(L"\r\n"); + _TerminalOutputHandlers(exitText); + } + CATCH_LOG(); } void ConptyConnection::_ClientTerminated() noexcept @@ -205,24 +235,25 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation return; } + // EXIT POINT DWORD exitCode{ 0 }; GetExitCodeProcess(_piClient.hProcess, &exitCode); - if (exitCode == 0) - { - _transitionToState(ConnectionState::Closed); - } - else + // Signal the closing or failure of the process. + // Load bearing. Terminating the pseudoconsole will make the output thread exit unexpectedly, + // so we need to signal entry into the correct closing state before we do that. + _transitionToState(exitCode == 0 ? ConnectionState::Closed : ConnectionState::Failed); + + // Close the pseudoconsole and wait for all output to drain. + _hPC.reset(); + if (auto localOutputThreadHandle = std::move(_hOutputThread)) { - winrt::hstring failureText{ wil::str_printf(RS_(L"ProcessExitedUnexpectedly").c_str(), (unsigned int)exitCode) }; - _TerminalOutputHandlers(L"\r\n"); - _TerminalOutputHandlers(failureText); - _transitionToState(ConnectionState::Failed); + LOG_LAST_ERROR_IF(WAIT_FAILED == WaitForSingleObject(localOutputThreadHandle.get(), INFINITE)); } - // We don't need these any longer. + _indicateExitWithStatus(exitCode); + _piClient.reset(); - _hPC.reset(); } void ConptyConnection::WriteInput(hstring const& data) @@ -255,6 +286,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation { if (_transitionToState(ConnectionState::Closing)) { + // EXIT POINT _clientExitWait.reset(); // immediately stop waiting for the client to exit. _hPC.reset(); // tear down the pseudoconsole (this is like clicking X on a console window) @@ -298,6 +330,8 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation return 0; } + // EXIT POINT + _indicateExitWithStatus(result); // print a message _transitionToState(ConnectionState::Failed); return (DWORD)-1; } diff --git a/src/cascadia/TerminalConnection/ConptyConnection.h b/src/cascadia/TerminalConnection/ConptyConnection.h index 014c9618804..165f2b4819e 100644 --- a/src/cascadia/TerminalConnection/ConptyConnection.h +++ b/src/cascadia/TerminalConnection/ConptyConnection.h @@ -32,6 +32,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation private: HRESULT _LaunchAttachedClient() noexcept; + void _indicateExitWithStatus(unsigned int status) noexcept; void _ClientTerminated() noexcept; uint32_t _initialRows{}; diff --git a/src/cascadia/TerminalConnection/Resources/en-US/Resources.resw b/src/cascadia/TerminalConnection/Resources/en-US/Resources.resw index 0fac3cbb453..43a0242a418 100644 --- a/src/cascadia/TerminalConnection/Resources/en-US/Resources.resw +++ b/src/cascadia/TerminalConnection/Resources/en-US/Resources.resw @@ -205,13 +205,13 @@ n The user must enter this character to signify NO - - [process terminated with exit code %u] - The first argument (%u) is the (positive) error code of the process. + + [process exited with code %u] + The first argument (%u) is the (positive) error code of the process. 0 is success. - [error 0x%x when launching `%ls'] + [error 0x%8.08x when launching `%ls'] The first argument is the hexadecimal error code. The second is the process name. If this resource spans multiple lines, it will not be displayed properly. Yeah. - + \ No newline at end of file diff --git a/src/cascadia/TerminalControl/TermControl.cpp b/src/cascadia/TerminalControl/TermControl.cpp index b02ac8d102a..0fc1cdb60cb 100644 --- a/src/cascadia/TerminalControl/TermControl.cpp +++ b/src/cascadia/TerminalControl/TermControl.cpp @@ -153,13 +153,11 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation // DON'T CALL _InitializeTerminal here - wait until the swap chain is loaded to do that. // Subscribe to the connection's disconnected event and call our connection closed handlers. - _connectionStateChangedRevoker = _connection.StateChanged(winrt::auto_revoke, [=](auto&& s, auto&& /*v*/) { - auto v = s.State(); - auto ff = wil::str_printf(L"CONNECTION TRANSITIONED TO STATE %d\n", v); - OutputDebugStringW(ff.c_str()); - if (v == TerminalConnection::ConnectionState::Closed) + _connectionStateChangedRevoker = _connection.StateChanged(winrt::auto_revoke, [weakThis = get_weak()](auto&& /*s*/, auto&& /*v*/) + { + if (auto strongThis{ weakThis.get() }) { - _connectionClosedHandlers(); + strongThis->_ConnectionStateChangedHandlers(*strongThis, nullptr); } }); } @@ -410,6 +408,11 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation return _swapChainPanel.Margin(); } + TerminalConnection::ConnectionState TermControl::ConnectionState() const + { + return _connection.State(); + } + void TermControl::SwapChainChanged() { if (!_initializedTerminal) @@ -1834,17 +1837,6 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation return flags; } - // Method Description: - // - Returns true if this control should close when its connection is closed. - // Arguments: - // - - // Return Value: - // - true iff the control should close when the connection is closed. - bool TermControl::ShouldCloseOnExit() const noexcept - { - return _settings.CloseOnExit(); - } - // Method Description: // - Gets the corresponding viewport terminal position for the cursor // by excluding the padding and normalizing with the font size. @@ -1917,7 +1909,6 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation // Winrt events need a method for adding a callback to the event and removing the callback. // These macros will define them both for you. DEFINE_EVENT(TermControl, TitleChanged, _titleChangedHandlers, TerminalControl::TitleChangedEventArgs); - DEFINE_EVENT(TermControl, ConnectionClosed, _connectionClosedHandlers, TerminalControl::ConnectionClosedEventArgs); DEFINE_EVENT(TermControl, ScrollPositionChanged, _scrollPositionChangedHandlers, TerminalControl::ScrollPositionChangedEventArgs); DEFINE_EVENT_WITH_TYPED_EVENT_HANDLER(TermControl, PasteFromClipboard, _clipboardPasteHandlers, TerminalControl::TermControl, TerminalControl::PasteFromClipboardEventArgs); diff --git a/src/cascadia/TerminalControl/TermControl.h b/src/cascadia/TerminalControl/TermControl.h index 69cbae1e270..b0e32cd8f95 100644 --- a/src/cascadia/TerminalControl/TermControl.h +++ b/src/cascadia/TerminalControl/TermControl.h @@ -62,7 +62,6 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation bool CopySelectionToClipboard(bool trimTrailingWhitespace); void PasteTextFromClipboard(); void Close(); - bool ShouldCloseOnExit() const noexcept; Windows::Foundation::Size CharacterDimensions() const; Windows::Foundation::Size MinimumSize() const; @@ -81,16 +80,19 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation const FontInfo GetActualFont() const; const Windows::UI::Xaml::Thickness GetPadding() const; + TerminalConnection::ConnectionState ConnectionState() const; + static Windows::Foundation::Point GetProposedDimensions(Microsoft::Terminal::Settings::IControlSettings const& settings, const uint32_t dpi); // clang-format off // -------------------------------- WinRT Events --------------------------------- DECLARE_EVENT(TitleChanged, _titleChangedHandlers, TerminalControl::TitleChangedEventArgs); - DECLARE_EVENT(ConnectionClosed, _connectionClosedHandlers, TerminalControl::ConnectionClosedEventArgs); DECLARE_EVENT(ScrollPositionChanged, _scrollPositionChangedHandlers, TerminalControl::ScrollPositionChangedEventArgs); DECLARE_EVENT_WITH_TYPED_EVENT_HANDLER(PasteFromClipboard, _clipboardPasteHandlers, TerminalControl::TermControl, TerminalControl::PasteFromClipboardEventArgs); DECLARE_EVENT_WITH_TYPED_EVENT_HANDLER(CopyToClipboard, _clipboardCopyHandlers, TerminalControl::TermControl, TerminalControl::CopyToClipboardEventArgs); + + TYPED_EVENT(ConnectionStateChanged, TerminalControl::TermControl, IInspectable); // clang-format on private: diff --git a/src/cascadia/TerminalControl/TermControl.idl b/src/cascadia/TerminalControl/TermControl.idl index 2fc56ed7e78..5eedc9a5458 100644 --- a/src/cascadia/TerminalControl/TermControl.idl +++ b/src/cascadia/TerminalControl/TermControl.idl @@ -4,7 +4,6 @@ namespace Microsoft.Terminal.TerminalControl { delegate void TitleChangedEventArgs(String newTitle); - delegate void ConnectionClosedEventArgs(); delegate void ScrollPositionChangedEventArgs(Int32 viewTop, Int32 viewHeight, Int32 bufferLength); runtimeclass CopyToClipboardEventArgs @@ -29,16 +28,19 @@ namespace Microsoft.Terminal.TerminalControl void UpdateSettings(Microsoft.Terminal.Settings.IControlSettings newSettings); event TitleChangedEventArgs TitleChanged; - event ConnectionClosedEventArgs ConnectionClosed; event Windows.Foundation.TypedEventHandler CopyToClipboard; event Windows.Foundation.TypedEventHandler PasteFromClipboard; + // This is an event handler forwarder for the underlying connection. + // We expose this and ConnectionState here so that it might eventually be data bound. + event Windows.Foundation.TypedEventHandler ConnectionStateChanged; + Microsoft.Terminal.TerminalConnection.ConnectionState ConnectionState { get; }; + String Title { get; }; Boolean CopySelectionToClipboard(Boolean trimTrailingWhitespace); void PasteTextFromClipboard(); void Close(); - Boolean ShouldCloseOnExit { get; }; Windows.Foundation.Size CharacterDimensions { get; }; Windows.Foundation.Size MinimumSize { get; }; diff --git a/src/cascadia/TerminalSettings/IControlSettings.idl b/src/cascadia/TerminalSettings/IControlSettings.idl index 03de4407143..2b9f2db274a 100644 --- a/src/cascadia/TerminalSettings/IControlSettings.idl +++ b/src/cascadia/TerminalSettings/IControlSettings.idl @@ -20,7 +20,6 @@ namespace Microsoft.Terminal.Settings interface IControlSettings requires Microsoft.Terminal.Settings.ICoreSettings { Boolean UseAcrylic; - Boolean CloseOnExit; Double TintOpacity; ScrollbarState ScrollState; diff --git a/src/cascadia/TerminalSettings/TerminalSettings.cpp b/src/cascadia/TerminalSettings/TerminalSettings.cpp index ff13952ad6b..42d7fdcba77 100644 --- a/src/cascadia/TerminalSettings/TerminalSettings.cpp +++ b/src/cascadia/TerminalSettings/TerminalSettings.cpp @@ -24,7 +24,6 @@ namespace winrt::Microsoft::Terminal::Settings::implementation _wordDelimiters{ DEFAULT_WORD_DELIMITERS }, _copyOnSelect{ false }, _useAcrylic{ false }, - _closeOnExit{ true }, _tintOpacity{ 0.5 }, _padding{ DEFAULT_PADDING }, _fontFace{ DEFAULT_FONT_FACE }, @@ -181,16 +180,6 @@ namespace winrt::Microsoft::Terminal::Settings::implementation _useAcrylic = value; } - bool TerminalSettings::CloseOnExit() - { - return _closeOnExit; - } - - void TerminalSettings::CloseOnExit(bool value) - { - _closeOnExit = value; - } - double TerminalSettings::TintOpacity() { return _tintOpacity; diff --git a/src/cascadia/TerminalSettings/terminalsettings.h b/src/cascadia/TerminalSettings/terminalsettings.h index ecc2781a46e..7b5000f05d1 100644 --- a/src/cascadia/TerminalSettings/terminalsettings.h +++ b/src/cascadia/TerminalSettings/terminalsettings.h @@ -55,8 +55,6 @@ namespace winrt::Microsoft::Terminal::Settings::implementation bool UseAcrylic(); void UseAcrylic(bool value); - bool CloseOnExit(); - void CloseOnExit(bool value); double TintOpacity(); void TintOpacity(double value); hstring Padding(); @@ -111,7 +109,6 @@ namespace winrt::Microsoft::Terminal::Settings::implementation hstring _wordDelimiters; bool _useAcrylic; - bool _closeOnExit; double _tintOpacity; hstring _fontFace; int32_t _fontSize; From 0577fe0a7ee546b1e1d07b29d4bab4499eff75d7 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 18 Nov 2019 17:13:13 -0800 Subject: [PATCH 05/17] doc comments --- .../ConnectionStateHolder.h | 23 +++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/src/cascadia/TerminalConnection/ConnectionStateHolder.h b/src/cascadia/TerminalConnection/ConnectionStateHolder.h index a53e26c6511..5e64994a845 100644 --- a/src/cascadia/TerminalConnection/ConnectionStateHolder.h +++ b/src/cascadia/TerminalConnection/ConnectionStateHolder.h @@ -13,6 +13,13 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation TYPED_EVENT(StateChanged, ITerminalConnection, winrt::Windows::Foundation::IInspectable); protected: + // Method Description: + // - Attempt to transition to and signal the specified connection state. + // The transition will only be effected if the state is "beyond" the current state. + // Arguments: + // - state: the new state + // Return Value: + // Whether we've successfully transitioned to the new state. bool _transitionToState(const ConnectionState state) noexcept { { @@ -29,6 +36,12 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation return true; } + // Method Description: + // - Returns whether the state is one of the N specified states. + // Arguments: + // - "...": the states + // Return Value: + // Whether we're in one of the states. template bool _isStateOneOf(Args&&... args) const noexcept { @@ -39,12 +52,22 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation return (... || (_connectionState == args)); } + // Method Description: + // - Returns whether the state has reached or surpassed the specified state. + // Arguments: + // - state; the state to check against + // Return Value: + // Whether we're at or beyond the specified state bool _isStateAtOrBeyond(const ConnectionState state) const noexcept { std::lock_guard stateLock{ _stateMutex }; return _connectionState >= state; } + // Method Description: + // - (Convenience:) Returns whether we're "connected". + // Return Value: + // Whether we're "connected" bool _isConnected() const noexcept { return _isStateOneOf(ConnectionState::Connected); From 410bf59079a5f3782ecaa01c7cef575019baa4b6 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 18 Nov 2019 17:13:52 -0800 Subject: [PATCH 06/17] invoke-codeformat --- src/cascadia/TerminalControl/TermControl.cpp | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/cascadia/TerminalControl/TermControl.cpp b/src/cascadia/TerminalControl/TermControl.cpp index 0fc1cdb60cb..9db5a715aa7 100644 --- a/src/cascadia/TerminalControl/TermControl.cpp +++ b/src/cascadia/TerminalControl/TermControl.cpp @@ -153,8 +153,7 @@ namespace winrt::Microsoft::Terminal::TerminalControl::implementation // DON'T CALL _InitializeTerminal here - wait until the swap chain is loaded to do that. // Subscribe to the connection's disconnected event and call our connection closed handlers. - _connectionStateChangedRevoker = _connection.StateChanged(winrt::auto_revoke, [weakThis = get_weak()](auto&& /*s*/, auto&& /*v*/) - { + _connectionStateChangedRevoker = _connection.StateChanged(winrt::auto_revoke, [weakThis = get_weak()](auto&& /*s*/, auto&& /*v*/) { if (auto strongThis{ weakThis.get() }) { strongThis->_ConnectionStateChangedHandlers(*strongThis, nullptr); From 4661db8ec57b2aebf1d166dc271c72352e880668 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 18 Nov 2019 17:22:23 -0800 Subject: [PATCH 07/17] do the schema dance --- doc/cascadia/profiles.schema.json | 18 +++++++++++++++--- 1 file changed, 15 insertions(+), 3 deletions(-) diff --git a/doc/cascadia/profiles.schema.json b/doc/cascadia/profiles.schema.json index 2fe2177dc50..447f2982cfe 100644 --- a/doc/cascadia/profiles.schema.json +++ b/doc/cascadia/profiles.schema.json @@ -325,9 +325,21 @@ "type": "string" }, "closeOnExit": { - "default": true, - "description": "When set to true (default), the selected tab closes when the connected application exits. When set to false, the tab will remain open when the connected application exits.", - "type": "boolean" + "default": "graceful", + "description": "When set to graceful/true (default), the selected tab closes when the connected application exits successfully. When set to always, the tab will never be closed automatically. When set to never/false, the tab will remain open when the connected application exits.", + "oneOf": [ + { + "enum": [ + "never", + "graceful", + "always" + ], + "type": "string" + }, + { + "type": "boolean" + } + ] }, "colorScheme": { "default": "Campbell", From 1d537aef82bc0cc4d3d347c59f72e2da4f8536ca Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 18 Nov 2019 17:25:05 -0800 Subject: [PATCH 08/17] and the doc? --- doc/cascadia/SettingsSchema.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/doc/cascadia/SettingsSchema.md b/doc/cascadia/SettingsSchema.md index a82977b6e8b..fd9e227e957 100644 --- a/doc/cascadia/SettingsSchema.md +++ b/doc/cascadia/SettingsSchema.md @@ -28,7 +28,7 @@ Properties listed below are specific to each unique profile. | `backgroundImageAlignment` | Optional | String | `center` | Sets how the background image aligns to the boundaries of the window. Possible values: `"center"`, `"left"`, `"top"`, `"right"`, `"bottom"`, `"topLeft"`, `"topRight"`, `"bottomLeft"`, `"bottomRight"` | | `backgroundImageOpacity` | Optional | Number | `1.0` | Sets the transparency of the background image. Accepts floating point values from 0-1. | | `backgroundImageStretchMode` | Optional | String | `uniformToFill` | Sets how the background image is resized to fill the window. Possible values: `"none"`, `"fill"`, `"uniform"`, `"uniformToFill"` | -| `closeOnExit` | Optional | Boolean | `true` | When set to `true`, the selected tab closes when `exit` is typed. When set to `false`, the tab will remain open when `exit` is typed. | +| `closeOnExit` | Optional | String | `graceful` | Sets how the profile reacts to termination or failure to launch. Possible values: `"graceful"` (close when `exit` is typed or the process exits normally), `"always"` (always close) and `"never"` (never close). `true` and `false` are accepted as synonyms for `"graceful"` and `"never"` respectively. | | `colorScheme` | Optional | String | `Campbell` | Name of the terminal color scheme to use. Color schemes are defined under `schemes`. | | `colorTable` | Optional | Array[String] | | Array of colors used in the profile if `colorscheme` is not set. Array follows the format defined in `schemes`. | | `commandline` | Optional | String | | Executable used in the profile. | From f38b736aae0717f419dc78f3ada95102f80743f5 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 18 Nov 2019 17:36:59 -0800 Subject: [PATCH 09/17] couple more azure fail points --- src/cascadia/TerminalConnection/AzureConnection.cpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/cascadia/TerminalConnection/AzureConnection.cpp b/src/cascadia/TerminalConnection/AzureConnection.cpp index 449f998290b..99b4c686c9b 100644 --- a/src/cascadia/TerminalConnection/AzureConnection.cpp +++ b/src/cascadia/TerminalConnection/AzureConnection.cpp @@ -522,6 +522,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation if (tenantListAsArray.size() == 0) { _WriteStringWithNewline(RS_(L"AzureNoTenants")); + _transitionToState(ConnectionState::Failed); return E_FAIL; } else if (_tenantList.size() == 1) @@ -631,6 +632,7 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation if (settingsResponse.has_field(L"error")) { _WriteStringWithNewline(RS_(L"AzureNoCloudAccount")); + _transitionToState(ConnectionState::Failed); return E_FAIL; } From 270fb73cb5b9bc7eea518b38382e2bd8acfea78b Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 18 Nov 2019 17:47:44 -0800 Subject: [PATCH 10/17] I guess serialize the closeyboi? --- src/cascadia/TerminalApp/Profile.cpp | 25 +++++++++++++++++++++++-- src/cascadia/TerminalApp/Profile.h | 2 ++ 2 files changed, 25 insertions(+), 2 deletions(-) diff --git a/src/cascadia/TerminalApp/Profile.cpp b/src/cascadia/TerminalApp/Profile.cpp index 0d4f95c7a1e..c3cf16c49b1 100644 --- a/src/cascadia/TerminalApp/Profile.cpp +++ b/src/cascadia/TerminalApp/Profile.cpp @@ -296,13 +296,11 @@ Json::Value Profile::ToJson() const } root[JsonKey(CursorShapeKey)] = winrt::to_string(_SerializeCursorStyle(_cursorShape)); - ///// Control Settings ///// root[JsonKey(CommandlineKey)] = winrt::to_string(_commandline); root[JsonKey(FontFaceKey)] = winrt::to_string(_fontFace); root[JsonKey(FontSizeKey)] = _fontSize; root[JsonKey(AcrylicTransparencyKey)] = _acrylicTransparency; root[JsonKey(UseAcrylicKey)] = _useAcrylic; - /* TODO(DH) root[JsonKey(CloseOnExitKey)] = _closeOnExit; */ root[JsonKey(PaddingKey)] = winrt::to_string(_padding); if (_connectionType) @@ -351,6 +349,8 @@ Json::Value Profile::ToJson() const root[JsonKey(BackgroundImageAlignmentKey)] = SerializeImageAlignment(_backgroundImageAlignment.value()).data(); } + root[JsonKey(CloseOnExitKey)] = _SerializeCloseOnExitMode(_closeOnExitMode); + return root; } @@ -927,6 +927,27 @@ CloseOnExitMode Profile::ParseCloseOnExitMode(const Json::Value& json) return CloseOnExitMode::Never; } +// Method Description: +// - Helper function for converting a CloseOnExitMode to its corresponding string +// value. +// Arguments: +// - closeOnExitMode: The enum value to convert to a string. +// Return Value: +// - The string value for the given CloseOnExitMode +std::string_view Profile::_SerializeCloseOnExitMode(const CloseOnExitMode closeOnExitMode) +{ + switch (closeOnExitMode) + { + case CloseOnExitMode::Always: + return CloseOnExitAlways; + case CloseOnExitMode::Graceful: + return CloseOnExitGraceful; + default: + case CloseOnExitMode::Never: + return CloseOnExitNever; + } +} + // Method Description: // - Helper function for converting a user-specified scrollbar state to its corresponding enum // Arguments: diff --git a/src/cascadia/TerminalApp/Profile.h b/src/cascadia/TerminalApp/Profile.h index 356c99d4d7d..1e636ba98fe 100644 --- a/src/cascadia/TerminalApp/Profile.h +++ b/src/cascadia/TerminalApp/Profile.h @@ -108,7 +108,9 @@ class TerminalApp::Profile final static std::string_view SerializeImageStretchMode(const winrt::Windows::UI::Xaml::Media::Stretch imageStretchMode); static std::tuple ParseImageAlignment(const std::string_view imageAlignment); static std::tuple _ConvertJsonToAlignment(const Json::Value& json); + static CloseOnExitMode ParseCloseOnExitMode(const Json::Value& json); + static std::string_view _SerializeCloseOnExitMode(const CloseOnExitMode closeOnExitMode); static std::string_view SerializeImageAlignment(const std::tuple imageAlignment); static winrt::Microsoft::Terminal::Settings::CursorStyle _ParseCursorShape(const std::wstring& cursorShapeString); From 17f1025a3174da8ad43a3c68d6aae2a60fdf0d31 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 18 Nov 2019 17:48:48 -0800 Subject: [PATCH 11/17] but do it RIGHT --- src/cascadia/TerminalApp/Profile.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/cascadia/TerminalApp/Profile.cpp b/src/cascadia/TerminalApp/Profile.cpp index c3cf16c49b1..36aa0a4230c 100644 --- a/src/cascadia/TerminalApp/Profile.cpp +++ b/src/cascadia/TerminalApp/Profile.cpp @@ -349,7 +349,7 @@ Json::Value Profile::ToJson() const root[JsonKey(BackgroundImageAlignmentKey)] = SerializeImageAlignment(_backgroundImageAlignment.value()).data(); } - root[JsonKey(CloseOnExitKey)] = _SerializeCloseOnExitMode(_closeOnExitMode); + root[JsonKey(CloseOnExitKey)] = _SerializeCloseOnExitMode(_closeOnExitMode).data(); return root; } From 0f827860cd2f397af2e7c2ddcc0bedde11557f21 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 18 Nov 2019 17:51:56 -0800 Subject: [PATCH 12/17] one also must update the default settings -- wow, closeOnExit is EVERYWHERE --- src/cascadia/TerminalApp/defaults.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/cascadia/TerminalApp/defaults.json b/src/cascadia/TerminalApp/defaults.json index 06bccfdc69e..e77cc62cb72 100644 --- a/src/cascadia/TerminalApp/defaults.json +++ b/src/cascadia/TerminalApp/defaults.json @@ -17,7 +17,7 @@ "commandline": "powershell.exe", "hidden": false, "startingDirectory": "%USERPROFILE%", - "closeOnExit": true, + "closeOnExit": "graceful", "colorScheme": "Campbell Powershell", "cursorColor": "#FFFFFF", "cursorShape": "bar", @@ -35,7 +35,7 @@ "commandline": "cmd.exe", "hidden": false, "startingDirectory": "%USERPROFILE%", - "closeOnExit": true, + "closeOnExit": "graceful", "colorScheme": "Campbell", "cursorColor": "#FFFFFF", "cursorShape": "bar", From 2af2827718f785906fee1e03f32faab17cfe4385 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 18 Nov 2019 17:54:52 -0800 Subject: [PATCH 13/17] guess we could also test CoE parsing --- .../LocalTests_TerminalApp/SettingsTests.cpp | 25 +++++++++++++++++++ 1 file changed, 25 insertions(+) diff --git a/src/cascadia/LocalTests_TerminalApp/SettingsTests.cpp b/src/cascadia/LocalTests_TerminalApp/SettingsTests.cpp index a4446f984e3..6d48315d76f 100644 --- a/src/cascadia/LocalTests_TerminalApp/SettingsTests.cpp +++ b/src/cascadia/LocalTests_TerminalApp/SettingsTests.cpp @@ -55,6 +55,8 @@ namespace TerminalAppLocalTests TEST_METHOD(TestProfileIconWithEnvVar); TEST_METHOD(TestProfileBackgroundImageWithEnvVar); + TEST_METHOD(TestCloseOnExitCompatibilityShim); + TEST_CLASS_SETUP(ClassSetup) { InitializeJsonReader(); @@ -1414,4 +1416,27 @@ namespace TerminalAppLocalTests auto terminalSettings = settings._profiles[0].CreateTerminalSettings(globalSettings.GetColorSchemes()); VERIFY_ARE_EQUAL(expectedPath, terminalSettings.BackgroundImage()); } + void SettingsTests::TestCloseOnExitCompatibilityShim() + { + const std::string settingsJson{ R"( + { + "profiles": [ + { + "name": "profile0", + "closeOnExit": true + }, + { + "name": "profile1", + "closeOnExit": false + } + ] + })" }; + + VerifyParseSucceeded(settingsJson); + CascadiaSettings settings{}; + settings._ParseJsonString(settingsJson, false); + settings.LayerJson(settings._userSettings); + VERIFY_ARE_EQUAL(CloseOnExitMode::Graceful, settings._profiles[0].GetCloseOnExitMode()); + VERIFY_ARE_EQUAL(CloseOnExitMode::Never, settings._profiles[1].GetCloseOnExitMode()); + } } From 9cc256fafc72a582e1dcf7a9d5c5fb4ee505d839 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Tue, 19 Nov 2019 14:55:27 -0800 Subject: [PATCH 14/17] do the thing greg pointed out --- doc/cascadia/profiles.schema.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/doc/cascadia/profiles.schema.json b/doc/cascadia/profiles.schema.json index 447f2982cfe..a8d36f68fde 100644 --- a/doc/cascadia/profiles.schema.json +++ b/doc/cascadia/profiles.schema.json @@ -326,7 +326,7 @@ }, "closeOnExit": { "default": "graceful", - "description": "When set to graceful/true (default), the selected tab closes when the connected application exits successfully. When set to always, the tab will never be closed automatically. When set to never/false, the tab will remain open when the connected application exits.", + "description": "Sets how the profile reacts to termination or failure to launch. Possible values: \"graceful\" (close when exit is typed or the process exits normally), \"always\" (always close) and \"never\" (never close). true and false are accepted as synonyms for \"graceful\" and \"never\" respectively.", "oneOf": [ { "enum": [ From 7f3b83cf40868eebeed9082180c08fe3c8646e97 Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Fri, 22 Nov 2019 13:46:18 -0800 Subject: [PATCH 15/17] PR feedback from carlos --- src/cascadia/TerminalApp/Profile.cpp | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/src/cascadia/TerminalApp/Profile.cpp b/src/cascadia/TerminalApp/Profile.cpp index 36aa0a4230c..ebd8d54dd73 100644 --- a/src/cascadia/TerminalApp/Profile.cpp +++ b/src/cascadia/TerminalApp/Profile.cpp @@ -922,9 +922,13 @@ CloseOnExitMode Profile::ParseCloseOnExitMode(const Json::Value& json) { return CloseOnExitMode::Graceful; } + else if (closeOnExit == CloseOnExitNever) + { + return CloseOnExitMode::Never; + } } - return CloseOnExitMode::Never; + return CloseOnExitMode::Graceful; } // Method Description: @@ -940,11 +944,11 @@ std::string_view Profile::_SerializeCloseOnExitMode(const CloseOnExitMode closeO { case CloseOnExitMode::Always: return CloseOnExitAlways; - case CloseOnExitMode::Graceful: - return CloseOnExitGraceful; - default: case CloseOnExitMode::Never: return CloseOnExitNever; + case CloseOnExitMode::Graceful: + default: + return CloseOnExitGraceful; } } From eb0ff76753d6c858977c532e1955aa2c1bae4c7f Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Fri, 22 Nov 2019 13:50:03 -0800 Subject: [PATCH 16/17] testos. --- .../LocalTests_TerminalApp/SettingsTests.cpp | 41 +++++++++++++++++++ 1 file changed, 41 insertions(+) diff --git a/src/cascadia/LocalTests_TerminalApp/SettingsTests.cpp b/src/cascadia/LocalTests_TerminalApp/SettingsTests.cpp index 6d48315d76f..21deac0a216 100644 --- a/src/cascadia/LocalTests_TerminalApp/SettingsTests.cpp +++ b/src/cascadia/LocalTests_TerminalApp/SettingsTests.cpp @@ -55,6 +55,7 @@ namespace TerminalAppLocalTests TEST_METHOD(TestProfileIconWithEnvVar); TEST_METHOD(TestProfileBackgroundImageWithEnvVar); + TEST_METHOD(TestCloseOnExitParsing); TEST_METHOD(TestCloseOnExitCompatibilityShim); TEST_CLASS_SETUP(ClassSetup) @@ -1416,6 +1417,46 @@ namespace TerminalAppLocalTests auto terminalSettings = settings._profiles[0].CreateTerminalSettings(globalSettings.GetColorSchemes()); VERIFY_ARE_EQUAL(expectedPath, terminalSettings.BackgroundImage()); } + void SettingsTests::TestCloseOnExitParsing() + { + const std::string settingsJson{ R"( + { + "profiles": [ + { + "name": "profile0", + "closeOnExit": "graceful" + }, + { + "name": "profile1", + "closeOnExit": "always" + }, + { + "name": "profile2", + "closeOnExit": "never" + }, + { + "name": "profile3", + "closeOnExit": null + }, + { + "name": "profile4", + "closeOnExit": { "clearly": "not a string" } + } + ] + })" }; + + VerifyParseSucceeded(settingsJson); + CascadiaSettings settings{}; + settings._ParseJsonString(settingsJson, false); + settings.LayerJson(settings._userSettings); + VERIFY_ARE_EQUAL(CloseOnExitMode::Graceful, settings._profiles[0].GetCloseOnExitMode()); + VERIFY_ARE_EQUAL(CloseOnExitMode::Always, settings._profiles[1].GetCloseOnExitMode()); + VERIFY_ARE_EQUAL(CloseOnExitMode::Never, settings._profiles[2].GetCloseOnExitMode()); + + // Unknown modes parse as "Graceful" + VERIFY_ARE_EQUAL(CloseOnExitMode::Graceful, settings._profiles[3].GetCloseOnExitMode()); + VERIFY_ARE_EQUAL(CloseOnExitMode::Graceful, settings._profiles[4].GetCloseOnExitMode()); + } void SettingsTests::TestCloseOnExitCompatibilityShim() { const std::string settingsJson{ R"( From 91938674283d8ebb2824330ea29c9249bc3cbcfd Mon Sep 17 00:00:00 2001 From: Dustin Howett Date: Mon, 25 Nov 2019 13:59:39 -0800 Subject: [PATCH 17/17] comments for days --- src/cascadia/TerminalConnection/ConnectionStateHolder.h | 9 +++++++-- src/cascadia/TerminalConnection/ConptyConnection.cpp | 6 ++++++ 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/src/cascadia/TerminalConnection/ConnectionStateHolder.h b/src/cascadia/TerminalConnection/ConnectionStateHolder.h index 5e64994a845..b38e0f13187 100644 --- a/src/cascadia/TerminalConnection/ConnectionStateHolder.h +++ b/src/cascadia/TerminalConnection/ConnectionStateHolder.h @@ -45,10 +45,15 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation template bool _isStateOneOf(Args&&... args) const noexcept { + // Dark magic! This function uses C++17 fold expressions. These fold expressions roughly expand as follows: + // (... OP (expression_using_args)) + // into -> + // expression_using_args[0] OP expression_using_args[1] OP expression_using_args[2] OP (etc.) + // We use that to first check that All Args types are ConnectionState (type[0] == ConnectionState && type[1] == ConnectionState && etc.) + // and then later to check that the current state is one of the passed-in ones: + // (_state == args[0] || _state == args[1] || etc.) static_assert((... && std::is_same::value), "all queried connection states must be from the ConnectionState enum"); std::lock_guard stateLock{ _stateMutex }; - // dark magic: this is a C++17 fold expression that expands into - // state == 1 || state == 2 || state == 3 ... return (... || (_connectionState == args)); } diff --git a/src/cascadia/TerminalConnection/ConptyConnection.cpp b/src/cascadia/TerminalConnection/ConptyConnection.cpp index 10733d52339..06319381b67 100644 --- a/src/cascadia/TerminalConnection/ConptyConnection.cpp +++ b/src/cascadia/TerminalConnection/ConptyConnection.cpp @@ -216,6 +216,10 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation _hPC.reset(); } + // Method Description: + // - prints out the "process exited" message formatted with the exit code + // Arguments: + // - status: the exit code. void ConptyConnection::_indicateExitWithStatus(unsigned int status) noexcept { try @@ -227,6 +231,8 @@ namespace winrt::Microsoft::Terminal::TerminalConnection::implementation CATCH_LOG(); } + // Method Description: + // - called when the client application (not necessarily its pty) exits for any reason void ConptyConnection::_ClientTerminated() noexcept { if (_isStateAtOrBeyond(ConnectionState::Closing))