Skip to content

Commit 53d3ef9

Browse files
authored
Merge pull request #88 from matthewlowry/bugfix/number-value-extra-var
Handle Number-typed extra var values.
2 parents a8b005f + 3007473 commit 53d3ef9

File tree

5 files changed

+81
-17
lines changed

5 files changed

+81
-17
lines changed

README.md

+7-3
Original file line numberDiff line numberDiff line change
@@ -487,8 +487,10 @@ node {
487487

488488
### Extra Variables
489489

490-
Extra variables can be passed to ansible by using a map in the pipeline script. Use the `hidden` parameter
491-
to keep the variable secret in the build log.
490+
Extra variables can be passed to ansible by using a map in the pipeline script.
491+
Supported value types are: `String`, `Boolean`, `Number`.
492+
By default the value will be considered potentially sensitive and masked in the logs.
493+
To override this give a map with keys `value` and `hidden`.
492494

493495
```groovy
494496
node {
@@ -497,7 +499,9 @@ node {
497499
playbook: 'cloud_playbooks/create-aws.yml',
498500
extraVars: [
499501
login: 'mylogin',
500-
secret_key: [value: 'g4dfKWENpeF6pY05', hidden: true]
502+
toggle: true,
503+
forks: 8,
504+
not_secret: [value: 'I want to see this in the logs', hidden: false]
501505
])
502506
}
503507
```

src/main/java/org/jenkinsci/plugins/ansible/AbstractAnsibleInvocation.java

+4
Original file line numberDiff line numberDiff line change
@@ -120,6 +120,10 @@ public T setExtraVars(List<ExtraVar> extraVars) {
120120
public ArgumentListBuilder appendExtraVars(ArgumentListBuilder args) {
121121
if (extraVars != null && ! extraVars.isEmpty()) {
122122
for (ExtraVar var : extraVars) {
123+
if (var.getSecretValue() == null) {
124+
listener.getLogger().println("[WARN] Omitting extra var " + var.getKey() + ": check value is a supported type.");
125+
continue;
126+
}
123127
args.add("-e");
124128
String value = envVars.expand(var.getSecretValue().getPlainText());
125129
if (Pattern.compile("\\s").matcher(value).find()) {

src/main/java/org/jenkinsci/plugins/ansible/workflow/AnsiblePlaybookStep.java

+31-14
Original file line numberDiff line numberDiff line change
@@ -374,7 +374,7 @@ public static final class AnsiblePlaybookExecution extends AbstractSynchronousNo
374374
@StepContextParameter
375375
private transient Computer computer;
376376

377-
private List<ExtraVar> convertExtraVars(Map<String, Object> extraVars) {
377+
private static List<ExtraVar> convertExtraVars(Map<String, Object> extraVars) {
378378
if (extraVars == null) {
379379
return null;
380380
}
@@ -384,25 +384,42 @@ private List<ExtraVar> convertExtraVars(Map<String, Object> extraVars) {
384384
var.setKey(entry.getKey());
385385
Object o = entry.getValue();
386386
if (o instanceof Map) {
387-
var.setSecretValue(Secret.fromString((String)((Map)o).get("value")));
388-
var.setHidden((Boolean)((Map)o).get("hidden"));
389-
}
390-
else if (o instanceof String) {
391-
var.setSecretValue(Secret.fromString((String)o));
392-
var.setHidden(true);
393-
}
394-
else if (o instanceof Boolean) {
395-
var.setSecretValue(Secret.fromString(o.toString()));
396-
var.setHidden(true);
397-
}
398-
else if (o instanceof Secret) {
399-
var.setSecretValue((Secret)o);
387+
var.setSecretValue(getSecretFromScalarValue(((Map<?,?>)o).get("value")));
388+
Object hidden = ((Map<?,?>)o).get("hidden");
389+
// If we are given a Boolean value for hidden, respect that.
390+
// Otherwise if omitted or explictly null or any other type adopt the safe default of hidden=true.
391+
if (hidden instanceof Boolean) {
392+
var.setHidden((Boolean)hidden);
393+
} else {
394+
var.setHidden(true);
395+
}
396+
} else {
397+
var.setSecretValue(getSecretFromScalarValue(o));
398+
// Consistent with above: for a scalar value effectively hidden is omitted so adopt the safe default of hidden=true.
400399
var.setHidden(true);
401400
}
402401
extraVarList.add(var);
403402
}
404403
return extraVarList;
405404
}
405+
406+
private static Secret getSecretFromScalarValue(Object o) {
407+
if (o instanceof String) {
408+
return Secret.fromString((String)o);
409+
}
410+
else if (o instanceof Boolean) {
411+
return Secret.fromString(o.toString());
412+
}
413+
else if (o instanceof Number) {
414+
return Secret.fromString(o.toString());
415+
}
416+
else if (o instanceof Secret) {
417+
return (Secret)o;
418+
}
419+
else {
420+
return null;
421+
}
422+
}
406423

407424
@Override
408425
protected Void run() throws Exception {

src/test/java/org/jenkinsci/plugins/ansible/PipelineTest.java

+12
Original file line numberDiff line numberDiff line change
@@ -87,6 +87,18 @@ public void testExtraVarsBoolean() throws Exception {
8787
));
8888
}
8989

90+
@Test
91+
public void testExtraVarsNumeric() throws Exception {
92+
String pipeline = IOUtils.toString(PipelineTest.class.getResourceAsStream("/pipelines/extraVarsNumeric.groovy"), StandardCharsets.UTF_8);
93+
WorkflowJob workflowJob = jenkins.createProject(WorkflowJob.class);
94+
workflowJob.setDefinition(new CpsFlowDefinition(pipeline, true));
95+
WorkflowRun run1 = workflowJob.scheduleBuild2(0).waitForStart();
96+
jenkins.waitForCompletion(run1);
97+
assertThat(run1.getLog(), allOf(
98+
containsString("ansible-playbook playbook.yml -e ********")
99+
));
100+
}
101+
90102
@Test
91103
public void testAnsiblePlaybookSshPass() throws Exception {
92104

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
pipeline {
2+
agent {
3+
label('test-agent')
4+
}
5+
stages {
6+
stage('Create playbook') {
7+
steps {
8+
writeFile(encoding: 'UTF-8', file: 'playbook.yml', text: '''- hosts: localhost
9+
connection: local
10+
gather_facts: no
11+
tasks:
12+
- debug: msg=test
13+
''')
14+
}
15+
}
16+
stage('Ansible playbook') {
17+
steps {
18+
warnError(message: 'ansible command not found?') {
19+
ansiblePlaybook(
20+
playbook: 'playbook.yml',
21+
extraVars: [foo1: 8],
22+
)
23+
}
24+
}
25+
}
26+
}
27+
}

0 commit comments

Comments
 (0)