1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
|
// Copyright 2014 Google Inc. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.analysis;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableList;
import com.google.devtools.build.lib.actions.AbstractAction;
import com.google.devtools.build.lib.actions.ActionExecutionContext;
import com.google.devtools.build.lib.actions.ActionExecutionException;
import com.google.devtools.build.lib.actions.ActionOwner;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.actions.Executor;
import com.google.devtools.build.lib.actions.ResourceSet;
import com.google.devtools.build.lib.util.Fingerprint;
/**
* Action responsible for the symlink tree creation.
* Used to generate runfiles and fileset symlink farms.
*/
public class SymlinkTreeAction extends AbstractAction {
private static final String GUID = "63412bda-4026-4c8e-a3ad-7deb397728d4";
private final Artifact inputManifest;
private final Artifact outputManifest;
private final boolean filesetTree;
/**
* Creates SymlinkTreeAction instance.
*
* @param owner action owner
* @param inputManifest exec path to the input runfiles manifest
* @param outputManifest exec path to the generated symlink tree manifest
* (must have "MANIFEST" base name). Symlink tree root
* will be set to the artifact's parent directory.
* @param filesetTree true if this is fileset symlink tree,
* false if this is a runfiles symlink tree.
*/
public SymlinkTreeAction(ActionOwner owner, Artifact inputManifest, Artifact outputManifest,
boolean filesetTree) {
super(owner, ImmutableList.of(inputManifest), ImmutableList.of(outputManifest));
Preconditions.checkArgument(outputManifest.getPath().getBaseName().equals("MANIFEST"));
this.inputManifest = inputManifest;
this.outputManifest = outputManifest;
this.filesetTree = filesetTree;
}
public Artifact getInputManifest() {
return inputManifest;
}
public Artifact getOutputManifest() {
return outputManifest;
}
public boolean isFilesetTree() {
return filesetTree;
}
@Override
public String getMnemonic() {
return "SymlinkTree";
}
@Override
protected String getRawProgressMessage() {
return (filesetTree ? "Creating Fileset tree " : "Creating runfiles tree ")
+ outputManifest.getExecPath().getParentDirectory().getPathString();
}
@Override
protected String computeKey() {
Fingerprint f = new Fingerprint();
f.addString(GUID);
f.addInt(filesetTree ? 1 : 0);
return f.hexDigestAndReset();
}
@Override
public ResourceSet estimateResourceConsumption(Executor executor) {
// Return null here to indicate that resources would be managed manually
// during action execution.
return null;
}
@Override
public String describeStrategy(Executor executor) {
return "local"; // Symlink tree is always generated locally.
}
@Override
public void execute(
ActionExecutionContext actionExecutionContext)
throws ActionExecutionException, InterruptedException {
actionExecutionContext.getExecutor().getContext(SymlinkTreeActionContext.class)
.createSymlinks(this, actionExecutionContext);
}
}
|