mirror of
https://gitlab.com/kicad/code/kicad.git
synced 2025-09-13 17:53:11 +02:00
(cherry picked from commit 44a3bb9274845593db733464aa3f68288e3eb150) Co-authored-by: Alex Shvartzkop <dudesuchamazing@gmail.com>
328 lines
10 KiB
C++
328 lines
10 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2024 Mark Roszko <mark.roszko@gmail.com>
|
|
* Copyright The 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 <common.h>
|
|
#include <cli/exit_codes.h>
|
|
#include <jobs_runner.h>
|
|
#include <jobs/job_registry.h>
|
|
#include <jobs/jobset.h>
|
|
#include <jobs/job_special_copyfiles.h>
|
|
#include <jobs/job_special_execute.h>
|
|
#include <kiway.h>
|
|
#include <kiway_express.h>
|
|
#include <reporter.h>
|
|
#include <wx/process.h>
|
|
#include <wx/txtstrm.h>
|
|
#include <wx/sstream.h>
|
|
#include <wx/wfstream.h>
|
|
#include <gestfich.h>
|
|
|
|
JOBS_RUNNER::JOBS_RUNNER( KIWAY* aKiway, JOBSET* aJobsFile, PROJECT* aProject,
|
|
REPORTER& aReporter, JOBS_PROGRESS_REPORTER* aProgressReporter ) :
|
|
m_kiway( aKiway ),
|
|
m_jobsFile( aJobsFile ),
|
|
m_reporter( aReporter ),
|
|
m_progressReporter( aProgressReporter ),
|
|
m_project( aProject )
|
|
{
|
|
}
|
|
|
|
|
|
bool JOBS_RUNNER::RunJobsAllDestinations( bool aBail )
|
|
{
|
|
bool success = true;
|
|
|
|
for( JOBSET_DESTINATION& destination : m_jobsFile->GetDestinations() )
|
|
success &= RunJobsForDestination( &destination, aBail );
|
|
|
|
return success;
|
|
}
|
|
|
|
|
|
int JOBS_RUNNER::runSpecialExecute( const JOBSET_JOB* aJob, REPORTER* aReporter, PROJECT* aProject )
|
|
{
|
|
JOB_SPECIAL_EXECUTE* specialJob = static_cast<JOB_SPECIAL_EXECUTE*>( aJob->m_job.get() );
|
|
wxString cmd = ExpandEnvVarSubstitutions( specialJob->m_command, m_project );
|
|
|
|
aReporter->Report( cmd, RPT_SEVERITY_INFO );
|
|
aReporter->Report( wxEmptyString, RPT_SEVERITY_INFO );
|
|
|
|
wxProcess process;
|
|
process.Redirect();
|
|
|
|
// static cast required because wx uses `long` which is 64-bit on Linux but 32-bit on Windows
|
|
int result = static_cast<int>( wxExecute( cmd, wxEXEC_SYNC, &process ) );
|
|
|
|
wxInputStream* inputStream = process.GetInputStream();
|
|
wxInputStream* errorStream = process.GetErrorStream();
|
|
|
|
if( inputStream && errorStream )
|
|
{
|
|
wxTextInputStream inputTextStream( *inputStream );
|
|
wxTextInputStream errorTextStream( *errorStream );
|
|
|
|
while( !inputStream->Eof() )
|
|
aReporter->Report( inputTextStream.ReadLine(), RPT_SEVERITY_INFO );
|
|
|
|
while( !errorStream->Eof() )
|
|
aReporter->Report( errorTextStream.ReadLine(), RPT_SEVERITY_ERROR );
|
|
|
|
if( specialJob->m_recordOutput )
|
|
{
|
|
if( specialJob->GetConfiguredOutputPath().IsEmpty() )
|
|
{
|
|
wxFileName fn( aJob->m_id );
|
|
fn.SetExt( wxT( "log" ) );
|
|
specialJob->SetConfiguredOutputPath( fn.GetFullPath() );
|
|
}
|
|
|
|
wxFFileOutputStream procOutput( specialJob->GetFullOutputPath( aProject ) );
|
|
|
|
if( !procOutput.IsOk() )
|
|
return CLI::EXIT_CODES::ERR_INVALID_OUTPUT_CONFLICT;
|
|
|
|
inputStream->Reset();
|
|
*inputStream >> procOutput;
|
|
}
|
|
}
|
|
|
|
if( specialJob->m_ignoreExitcode )
|
|
return CLI::EXIT_CODES::OK;
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
int JOBS_RUNNER::runSpecialCopyFiles( const JOBSET_JOB* aJob, PROJECT* aProject )
|
|
{
|
|
JOB_SPECIAL_COPYFILES* job = static_cast<JOB_SPECIAL_COPYFILES*>( aJob->m_job.get() );
|
|
|
|
wxString source = ExpandEnvVarSubstitutions( job->m_source, aProject );
|
|
|
|
if( source.IsEmpty() )
|
|
return CLI::EXIT_CODES::ERR_ARGS;
|
|
|
|
wxString projectPath = aProject->GetProjectPath();
|
|
wxFileName sourceFn( source );
|
|
sourceFn.MakeAbsolute( projectPath );
|
|
|
|
wxFileName destFn( job->GetFullOutputPath( aProject ) );
|
|
|
|
if( !job->m_dest.IsEmpty() )
|
|
destFn.AppendDir( job->m_dest );
|
|
|
|
std::vector<wxString> exclusions;
|
|
|
|
for( const JOBSET_DESTINATION& destination : m_jobsFile->GetDestinations() )
|
|
exclusions.push_back( projectPath + destination.m_outputHandler->GetOutputPath() );
|
|
|
|
wxString errors;
|
|
int copyCount = 0;
|
|
bool success = CopyFilesOrDirectory( sourceFn.GetFullPath(), destFn.GetFullPath(),
|
|
errors, copyCount, exclusions );
|
|
|
|
if( !success )
|
|
return CLI::EXIT_CODES::ERR_UNKNOWN;
|
|
|
|
if( job->m_generateErrorOnNoCopy && copyCount == 0 )
|
|
return CLI::EXIT_CODES::ERR_UNKNOWN;
|
|
|
|
return CLI::EXIT_CODES::OK;
|
|
}
|
|
|
|
|
|
bool JOBS_RUNNER::RunJobsForDestination( JOBSET_DESTINATION* aDestination, bool aBail )
|
|
{
|
|
bool genOutputs = true;
|
|
bool success = true;
|
|
std::vector<JOBSET_JOB> jobsForDestination = m_jobsFile->GetJobsForDestination( aDestination );
|
|
wxString msg;
|
|
|
|
wxFileName tmp;
|
|
tmp.AssignDir( wxFileName::GetTempDir() );
|
|
tmp.AppendDir( KIID().AsString() );
|
|
|
|
aDestination->m_lastRunSuccessMap.clear();
|
|
aDestination->m_lastRunReporters.clear();
|
|
|
|
wxString tempDirPath = tmp.GetFullPath();
|
|
|
|
if( !wxFileName::Mkdir( tempDirPath, wxS_DIR_DEFAULT ) )
|
|
{
|
|
msg = wxString::Format( wxT( "Failed to create temporary directory %s" ), tempDirPath );
|
|
m_reporter.Report( msg, RPT_SEVERITY_ERROR );
|
|
|
|
aDestination->m_lastRunSuccess = false;
|
|
|
|
return false;
|
|
}
|
|
|
|
bool continueOuput = aDestination->m_outputHandler->OutputPrecheck();
|
|
|
|
if( !continueOuput )
|
|
{
|
|
msg = wxString::Format( wxT( "Destination precheck failed for destination %s" ),
|
|
aDestination->m_id );
|
|
m_reporter.Report( msg, RPT_SEVERITY_ERROR );
|
|
|
|
aDestination->m_lastRunSuccess = false;
|
|
return false;
|
|
}
|
|
|
|
msg += wxT( "|--------------------------------\n" );
|
|
msg += wxT( "| " );
|
|
msg += wxString::Format( wxT( "Running jobs for destination %s" ), aDestination->m_id );
|
|
msg += wxT( "\n" );
|
|
msg += wxT( "|--------------------------------\n" );
|
|
|
|
msg += wxString::Format( wxT( "|%-5s | %-50s\n" ), wxT( "No." ), wxT( "Description" ) );
|
|
|
|
int jobNum = 1;
|
|
|
|
for( const JOBSET_JOB& job : jobsForDestination )
|
|
{
|
|
msg += wxString::Format( wxT( "|%-5d | %-50s\n" ), jobNum, job.GetDescription() );
|
|
jobNum++;
|
|
}
|
|
|
|
msg += wxT( "|--------------------------------\n" );
|
|
msg += wxT( "\n" );
|
|
msg += wxT( "\n" );
|
|
|
|
m_reporter.Report( msg, RPT_SEVERITY_INFO );
|
|
|
|
std::vector<JOB_OUTPUT> outputs;
|
|
|
|
jobNum = 1;
|
|
int failCount = 0;
|
|
int successCount = 0;
|
|
|
|
wxSetEnv( OUTPUT_TMP_PATH_VAR_NAME, tempDirPath );
|
|
|
|
for( const JOBSET_JOB& job : jobsForDestination )
|
|
{
|
|
msg = wxT( "|--------------------------------\n" );
|
|
|
|
msg += wxString::Format( wxT( "| Running job %d: %s" ), jobNum, job.GetDescription() );
|
|
|
|
msg += wxT( "\n" );
|
|
msg += wxT( "|--------------------------------\n" );
|
|
|
|
m_reporter.Report( msg, RPT_SEVERITY_INFO );
|
|
|
|
if( m_progressReporter )
|
|
{
|
|
msg.Printf( _( "Running job %d: %s" ), jobNum, job.GetDescription() );
|
|
m_progressReporter->AdvanceJob( msg );
|
|
m_progressReporter->KeepRefreshing();
|
|
}
|
|
|
|
jobNum++;
|
|
|
|
KIWAY::FACE_T iface = JOB_REGISTRY::GetKifaceType( job.m_type );
|
|
|
|
job.m_job->SetTempOutputDirectory( tempDirPath );
|
|
|
|
REPORTER* targetReporter = &m_reporter;
|
|
|
|
if( targetReporter == &NULL_REPORTER::GetInstance() )
|
|
{
|
|
aDestination->m_lastRunReporters[job.m_id] =
|
|
std::make_shared<JOBSET_OUTPUT_REPORTER>( tempDirPath, m_progressReporter );
|
|
|
|
targetReporter = aDestination->m_lastRunReporters[job.m_id].get();
|
|
}
|
|
|
|
// Use a redirect reporter so we don't have error flags set after running previous jobs
|
|
REDIRECT_REPORTER isolatedReporter( targetReporter );
|
|
int result = CLI::EXIT_CODES::SUCCESS;
|
|
|
|
if( iface < KIWAY::KIWAY_FACE_COUNT )
|
|
{
|
|
result = m_kiway->ProcessJob( iface, job.m_job.get(), &isolatedReporter,
|
|
m_progressReporter );
|
|
}
|
|
else
|
|
{
|
|
// special jobs
|
|
if( job.m_job->GetType() == "special_execute" )
|
|
{
|
|
result = runSpecialExecute( &job, &isolatedReporter, m_project );
|
|
}
|
|
else if( job.m_job->GetType() == "special_copyfiles" )
|
|
{
|
|
result = runSpecialCopyFiles( &job, m_project );
|
|
}
|
|
}
|
|
|
|
aDestination->m_lastRunSuccessMap[job.m_id] = ( result == CLI::EXIT_CODES::SUCCESS );
|
|
|
|
if( result == CLI::EXIT_CODES::SUCCESS )
|
|
{
|
|
wxString msg_fmt = wxT( "\033[32;1m%s\033[0m\n" );
|
|
msg = wxString::Format( msg_fmt, _( "Job successful" ) );
|
|
|
|
successCount++;
|
|
}
|
|
else
|
|
{
|
|
wxString msg_fmt = wxT( "\033[31;1m%s\033[0m\n" );
|
|
msg = wxString::Format( msg_fmt, _( "Job failed" ) );
|
|
|
|
failCount++;
|
|
}
|
|
|
|
msg += wxT( "\n\n" );
|
|
m_reporter.Report( msg, RPT_SEVERITY_INFO );
|
|
|
|
if( result == CLI::EXIT_CODES::ERR_RC_VIOLATIONS )
|
|
{
|
|
success = false;
|
|
|
|
if( aBail )
|
|
break;
|
|
}
|
|
else if( result != CLI::EXIT_CODES::SUCCESS )
|
|
{
|
|
genOutputs = false;
|
|
success = false;
|
|
|
|
if( aBail )
|
|
break;
|
|
}
|
|
}
|
|
|
|
wxUnsetEnv( OUTPUT_TMP_PATH_VAR_NAME );
|
|
|
|
if( genOutputs )
|
|
success &= aDestination->m_outputHandler->HandleOutputs( tempDirPath, m_project, outputs );
|
|
|
|
aDestination->m_lastRunSuccess = success;
|
|
|
|
msg = wxString::Format( wxT( "\n\n\033[33;1m%d %s, %d %s\033[0m\n" ),
|
|
successCount,
|
|
wxT( "jobs succeeded" ),
|
|
failCount,
|
|
wxT( "job failed" ) );
|
|
|
|
m_reporter.Report( msg, RPT_SEVERITY_INFO );
|
|
|
|
return success;
|
|
} |