@theia/cpp
Version:
Theia - Cpp Extension
197 lines (179 loc) • 7.39 kB
text/typescript
/********************************************************************************
* Copyright (C) 2018 Ericsson and others.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the Eclipse
* Public License v. 2.0 are satisfied: GNU General Public License, version 2
* with the GNU Classpath Exception which is available at
* https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
********************************************************************************/
import parseArgv = require('string-argv');
import { inject, injectable, postConstruct } from 'inversify';
import { ProcessTaskConfiguration } from '@theia/task/lib/common/process/task-protocol';
import { TaskContribution, TaskProvider, TaskProviderRegistry, TaskResolver, TaskResolverRegistry } from '@theia/task/lib/browser/task-contribution';
import { CppBuildConfiguration } from '../common/cpp-build-configuration-protocol';
import { CppBuildConfigurationManager } from './cpp-build-configurations';
import { ContributedTaskConfiguration, TaskConfiguration } from '@theia/task/lib/common/task-protocol';
import { TaskDefinitionRegistry } from '@theia/task/lib/browser/task-definition-registry';
import { ProblemMatcherRegistry } from '@theia/task/lib/browser/task-problem-matcher-registry';
import { ProblemPatternRegistry } from '@theia/task/lib/browser/task-problem-pattern-registry';
/**
* Representation of a C/C++ build task configuration.
* Describes the data required to define a C/C++ build task the user could run.
*/
interface CppBuildTaskConfiguration extends ContributedTaskConfiguration {
/**
* The C/C++ build configuration.
*/
config: CppBuildConfiguration;
}
/**
* The C/C++ build task type key.
*/
const CPP_BUILD_TASK_TYPE_KEY: string = 'cpp.build';
/**
* The C/C++ build task source.
*/
const CPP_BUILD_TASK_SOURCE: string = 'cpp';
()
export class CppTaskProvider implements TaskContribution, TaskProvider, TaskResolver {
(TaskResolverRegistry) protected readonly taskResolverRegistry: TaskResolverRegistry;
(TaskDefinitionRegistry) protected readonly taskDefinitionRegistry: TaskDefinitionRegistry;
(CppBuildConfigurationManager) protected readonly cppBuildConfigurationManager: CppBuildConfigurationManager;
(ProblemMatcherRegistry) protected readonly problemMatcherRegistry: ProblemMatcherRegistry;
(ProblemPatternRegistry) protected readonly problemPatternRegistry: ProblemPatternRegistry;
/**
* Initialize the task provider.
*/
()
protected init(): void {
this.registerTaskDefinition();
this.problemPatternRegistry.register({
'name': 'clangTidyPattern',
'regexp': '^(.+):(\\d+):(\\d+):\\s+(error|warning|info|note):\\s+(.+?)\\s+\\[(.+)\\]$',
'file': 1,
'line': 2,
'character': 3,
'severity': 4,
'message': 5,
'code': 6
});
this.problemMatcherRegistry.register({
'name': 'clangTidyMatcher',
'label': 'Clang-tidy problems',
'owner': 'clang-tidy',
'source': 'clang-tidy-task',
'applyTo': 'alldocuments',
'fileLocation': [
'absolute'
],
'pattern': 'clangTidyPattern'
});
}
/**
* Register the task provider.
* @param registry the task provider registry.
*/
registerProviders(registry: TaskProviderRegistry): void {
registry.register(CPP_BUILD_TASK_SOURCE, this);
}
/**
* Register the task resolver.
* @param registry the task resolver registry.
*/
registerResolvers(registry: TaskResolverRegistry): void {
registry.register(CPP_BUILD_TASK_TYPE_KEY, this);
}
/**
* Resolve the C/C++ build task configuration.
* @param task the C/C++ build task configuration.
*
* @returns a Promise resolving to the task configuration.
*/
async resolveTask(task: CppBuildTaskConfiguration): Promise<TaskConfiguration> {
const resolver = await this.taskResolverRegistry.getResolver('shell');
if (!resolver) {
throw new Error('No shell resolver found, cannot build.');
}
const buildCommand = task.config.commands && task.config.commands['build'];
if (buildCommand === undefined) {
throw new Error(`No build command defined in build configuration ${task.config.name}.`);
}
// FIXME: the task API should allow to run commands through the shell.
// In the mean time, we split the arguments ourselves.
const argv: string[] = parseArgv(buildCommand);
if (argv.length === 0) {
throw new Error(`Empty build command in the active build configuration (${task.config.name})`);
}
const command: string = argv[0];
const args: string[] = argv.slice(1);
const resolvedTask: ProcessTaskConfiguration = {
...task,
type: 'shell',
command,
args,
options: {
cwd: task.config.directory,
}
};
return resolver.resolveTask(resolvedTask);
}
/**
* Return a C/C++ build task configuration based on `config`, or undefined
* if `config` doesn't specify a build command.
*/
makeTaskConfiguration(config: CppBuildConfiguration): CppBuildTaskConfiguration | undefined {
if (config.commands && config.commands.build) {
return {
type: CPP_BUILD_TASK_TYPE_KEY,
_source: CPP_BUILD_TASK_SOURCE,
_scope: config.directory,
label: `C/C++ Build - ${config.name}`,
config
};
}
return undefined;
}
/**
* Return the C/C++ build tasks (one task per existing build config).
*/
async provideTasks(): Promise<CppBuildTaskConfiguration[]> {
const buildConfigs = this.cppBuildConfigurationManager.getConfigs();
const taskConfigs: CppBuildTaskConfiguration[] = [];
for (const buildConfig of buildConfigs) {
const taskConfig = this.makeTaskConfiguration(buildConfig);
if (taskConfig) {
taskConfigs.push(taskConfig);
}
}
return taskConfigs;
}
/**
* Register the task definition.
*/
private registerTaskDefinition(): void {
this.taskDefinitionRegistry.register({
taskType: CPP_BUILD_TASK_TYPE_KEY,
source: 'cpp',
properties: {
required: ['label'],
all: ['label'],
schema: {
type: CPP_BUILD_TASK_TYPE_KEY,
required: ['label'],
properties: {
label: {
type: 'string'
}
}
}
}
});
}
}