2023-01-29 18:06:05 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2023 Jon Evans <jon@craftyjon.com>
|
|
|
|
* Copyright (C) 2023 KiCad Developers, see AUTHORS.txt for contributors.
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License as published by the
|
|
|
|
* Free Software Foundation, either version 3 of the License, or (at your
|
|
|
|
* option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <wx/process.h>
|
|
|
|
|
|
|
|
#include <utility>
|
|
|
|
|
2024-01-14 04:14:02 +00:00
|
|
|
#include <paths.h>
|
|
|
|
#include <python_manager.h>
|
|
|
|
|
2023-01-29 18:06:05 +00:00
|
|
|
|
|
|
|
class PYTHON_PROCESS : public wxProcess
|
|
|
|
{
|
|
|
|
public:
|
2024-01-14 04:14:02 +00:00
|
|
|
PYTHON_PROCESS( std::function<void(int, const wxString&, const wxString&)> aCallback ) :
|
2023-01-29 18:06:05 +00:00
|
|
|
wxProcess(),
|
|
|
|
m_callback( std::move( aCallback ) )
|
|
|
|
{}
|
|
|
|
|
|
|
|
void OnTerminate( int aPid, int aStatus ) override
|
|
|
|
{
|
|
|
|
if( m_callback )
|
|
|
|
{
|
2024-01-14 04:14:02 +00:00
|
|
|
wxString output, error;
|
2023-01-29 18:06:05 +00:00
|
|
|
wxInputStream* processOut = GetInputStream();
|
|
|
|
size_t bytesRead = 0;
|
|
|
|
|
|
|
|
while( processOut->CanRead() && bytesRead < MAX_OUTPUT_LEN )
|
|
|
|
{
|
|
|
|
char buffer[4096];
|
|
|
|
buffer[ processOut->Read( buffer, sizeof( buffer ) - 1 ).LastRead() ] = '\0';
|
|
|
|
output.append( buffer, sizeof( buffer ) );
|
|
|
|
bytesRead += processOut->LastRead();
|
|
|
|
}
|
|
|
|
|
2024-01-14 04:14:02 +00:00
|
|
|
processOut = GetErrorStream();
|
|
|
|
bytesRead = 0;
|
|
|
|
|
|
|
|
while( processOut->CanRead() && bytesRead < MAX_OUTPUT_LEN )
|
|
|
|
{
|
|
|
|
char buffer[4096];
|
|
|
|
buffer[ processOut->Read( buffer, sizeof( buffer ) - 1 ).LastRead() ] = '\0';
|
|
|
|
error.append( buffer, sizeof( buffer ) );
|
|
|
|
bytesRead += processOut->LastRead();
|
|
|
|
}
|
|
|
|
|
|
|
|
m_callback( aStatus, output, error );
|
2023-01-29 18:06:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr size_t MAX_OUTPUT_LEN = 1024L * 1024L;
|
|
|
|
|
|
|
|
private:
|
2024-01-14 04:14:02 +00:00
|
|
|
std::function<void(int, const wxString&, const wxString&)> m_callback;
|
2023-01-29 18:06:05 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
void PYTHON_MANAGER::Execute( const wxString& aArgs,
|
2024-01-14 04:14:02 +00:00
|
|
|
const std::function<void( int, const wxString&,
|
|
|
|
const wxString& )>& aCallback,
|
|
|
|
const wxExecuteEnv* aEnv )
|
2023-01-29 18:06:05 +00:00
|
|
|
{
|
|
|
|
PYTHON_PROCESS* process = new PYTHON_PROCESS( aCallback );
|
|
|
|
process->Redirect();
|
|
|
|
|
|
|
|
wxString cmd = wxString::Format( wxS( "%s %s" ), m_interpreterPath, aArgs );
|
2024-01-14 04:14:02 +00:00
|
|
|
long pid = wxExecute( cmd, wxEXEC_ASYNC, process, aEnv );
|
2023-01-29 18:06:05 +00:00
|
|
|
|
|
|
|
if( pid == 0 )
|
2024-01-14 04:14:02 +00:00
|
|
|
aCallback( -1, wxEmptyString, _( "Process could not be created" ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
wxString PYTHON_MANAGER::FindPythonInterpreter()
|
|
|
|
{
|
|
|
|
#ifdef __WXMSW__
|
|
|
|
// TODO(JE) where
|
|
|
|
#else
|
|
|
|
wxArrayString output;
|
|
|
|
|
|
|
|
if( 0 == wxExecute( wxS( "which -a python" ), output, wxEXEC_SYNC ) )
|
|
|
|
{
|
|
|
|
if( !output.IsEmpty() )
|
|
|
|
return output[0];
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return wxEmptyString;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::optional<wxString> PYTHON_MANAGER::GetPythonEnvironment( const wxString& aNamespace )
|
|
|
|
{
|
|
|
|
wxFileName path( PATHS::GetUserCachePath(), wxEmptyString );
|
|
|
|
path.AppendDir( wxS( "python-environments" ) );
|
|
|
|
path.AppendDir( aNamespace );
|
|
|
|
|
|
|
|
if( !PATHS::EnsurePathExists( path.GetPath() ) )
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
return path.GetPath();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::optional<wxString> PYTHON_MANAGER::GetVirtualPython( const wxString& aNamespace )
|
|
|
|
{
|
|
|
|
std::optional<wxString> envPath = GetPythonEnvironment( aNamespace );
|
|
|
|
|
|
|
|
if( !envPath )
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
wxFileName python( *envPath, wxEmptyString );
|
|
|
|
python.AppendDir( "bin" );
|
|
|
|
python.SetFullName( "python" );
|
|
|
|
|
|
|
|
if( !python.IsFileExecutable() )
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
return python.GetFullPath();
|
2023-01-29 18:06:05 +00:00
|
|
|
}
|