blob: 7b16e3ca5515a9956df560a806a8b3c37c539065 [file] [log] [blame]
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001// Copyright 2014 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package com.google.devtools.build.lib.packages;
16
17import com.google.common.base.Preconditions;
18import com.google.common.collect.ComparisonChain;
19import com.google.devtools.build.lib.syntax.Label.SyntaxException;
20import com.google.devtools.build.lib.util.StringCanonicalizer;
21import com.google.devtools.build.lib.util.StringUtilities;
22import com.google.devtools.build.lib.vfs.Canonicalizer;
23import com.google.devtools.build.lib.vfs.PathFragment;
24
25import java.io.IOException;
26import java.io.ObjectInputStream;
27import java.io.ObjectOutputStream;
28import java.io.ObjectStreamException;
29import java.io.Serializable;
30import java.util.Objects;
31
32import javax.annotation.concurrent.Immutable;
33
34/**
35 * Uniquely identifies a package, given a repository name and a package's path fragment.
36 *
37 * <p>The repository the build is happening in is the <i>default workspace</i>, and is identified
38 * by the workspace name "". Other repositories can be named in the WORKSPACE file. These
39 * workspaces are prefixed by {@literal @}.</p>
40 */
41@Immutable
42public final class PackageIdentifier implements Comparable<PackageIdentifier>, Serializable {
43
44 /**
45 * A human-readable name for the repository.
46 */
47 public static final class RepositoryName {
48 private final String name;
49
50 /**
51 * Makes sure that name is a valid repository name and creates a new RepositoryName using it.
52 * @throws SyntaxException if the name is invalid.
53 */
54 public static RepositoryName create(String name) throws SyntaxException {
55 String errorMessage = validate(name);
56 if (errorMessage != null) {
57 errorMessage = "invalid repository name '"
58 + StringUtilities.sanitizeControlChars(name) + "': " + errorMessage;
59 throw new SyntaxException(errorMessage);
60 }
61 return new RepositoryName(StringCanonicalizer.intern(name));
62 }
63
64 private RepositoryName(String name) {
65 this.name = name;
66 }
67
68 /**
69 * Performs validity checking. Returns null on success, an error message otherwise.
70 */
71 private static String validate(String name) {
72 if (name.isEmpty()) {
73 return null;
74 }
75
76 if (!name.startsWith("@")) {
77 return "workspace name must start with '@'";
78 }
79
80 // "@" isn't a valid workspace name.
81 if (name.length() == 1) {
82 return "empty workspace name";
83 }
84
85 // Check for any character outside of [/0-9A-Z_a-z-]. Try to evaluate the
86 // conditional quickly (by looking in decreasing order of character class
87 // likelihood).
88 for (int i = name.length() - 1; i >= 1; --i) {
89 char c = name.charAt(i);
90 if ((c < 'a' || c > 'z') && c != '_' && c != '-'
91 && (c < '0' || c > '9') && (c < 'A' || c > 'Z')) {
92 return "workspace names may contain only A-Z, a-z, 0-9, '-' and '_'";
93 }
94 }
95 return null;
96 }
97
98 /**
99 * Returns the repository name without the leading "{@literal @}". For the default repository,
100 * returns "".
101 */
102 public String strippedName() {
103 if (name.isEmpty()) {
104 return name;
105 }
106 return name.substring(1);
107 }
108
109 /**
110 * Returns if this is the default repository, that is, {@link #name} is "".
111 */
112 public boolean isDefault() {
113 return name.isEmpty();
114 }
115
116 /**
117 * Returns the repository name, with leading "{@literal @}" (or "" for the default repository).
118 */
119 @Override
120 public String toString() {
121 return name;
122 }
123
124 @Override
125 public boolean equals(Object object) {
126 if (this == object) {
127 return true;
128 }
129 if (object instanceof RepositoryName) {
130 return name.equals(((RepositoryName) object).name);
131 }
132 return false;
133 }
134
135 @Override
136 public int hashCode() {
137 return name.hashCode();
138 }
139 }
140
141 public static final String DEFAULT_REPOSITORY = "";
142
143 /**
144 * Helper for serializing PackageIdentifiers.
145 *
146 * <p>PackageIdentifier's field should be final, but then it couldn't be deserialized. This
147 * allows the fields to be deserialized and copied into a new PackageIdentifier.</p>
148 */
149 private static final class SerializationProxy implements Serializable {
150 PackageIdentifier packageId;
151
152 public SerializationProxy(PackageIdentifier packageId) {
153 this.packageId = packageId;
154 }
155
156 private void writeObject(ObjectOutputStream out) throws IOException {
157 out.writeObject(packageId.repository.toString());
158 out.writeObject(packageId.pkgName);
159 }
160
161 private void readObject(ObjectInputStream in)
162 throws IOException, ClassNotFoundException {
163 try {
164 packageId = new PackageIdentifier((String) in.readObject(), (PathFragment) in.readObject());
165 } catch (SyntaxException e) {
166 throw new IOException("Error serializing package identifier: " + e.getMessage());
167 }
168 }
169
170 @SuppressWarnings("unused")
171 private void readObjectNoData() throws ObjectStreamException {
172 }
173
174 private Object readResolve() {
175 return packageId;
176 }
177 }
178
179 // Temporary factory for identifiers without explicit repositories.
180 // TODO(bazel-team): remove all usages of this.
181 public static PackageIdentifier createInDefaultRepo(String name) {
182 return createInDefaultRepo(new PathFragment(name));
183 }
184
185 public static PackageIdentifier createInDefaultRepo(PathFragment name) {
186 try {
187 return new PackageIdentifier(DEFAULT_REPOSITORY, name);
188 } catch (SyntaxException e) {
189 throw new IllegalArgumentException("could not create package identifier for " + name
190 + ": " + e.getMessage());
191 }
192 }
193
194 /**
195 * The identifier for this repository. This is either "" or prefixed with an "@",
196 * e.g., "@myrepo".
197 */
198 private final RepositoryName repository;
199
200 /** The name of the package. Canonical (i.e. x.equals(y) <=> x==y). */
201 private final PathFragment pkgName;
202
203 public PackageIdentifier(String repository, PathFragment pkgName) throws SyntaxException {
204 this(RepositoryName.create(repository), pkgName);
205 }
206
207 public PackageIdentifier(RepositoryName repository, PathFragment pkgName) {
208 Preconditions.checkNotNull(repository);
209 Preconditions.checkNotNull(pkgName);
210 this.repository = repository;
211 this.pkgName = Canonicalizer.fragments().intern(pkgName);
212 }
213
214 private Object writeReplace() throws ObjectStreamException {
215 return new SerializationProxy(this);
216 }
217
218 private void readObject(ObjectInputStream in)
219 throws IOException, ClassNotFoundException {
220 throw new IOException("Serialization is allowed only by proxy");
221 }
222
223 @SuppressWarnings("unused")
224 private void readObjectNoData() throws ObjectStreamException {
225 }
226
227 public RepositoryName getRepository() {
228 return repository;
229 }
230
231 public PathFragment getPackageFragment() {
232 return pkgName;
233 }
234
235 /**
236 * Returns the name of this package.
237 *
238 * <p>There are certain places that expect the path fragment as the package name ('foo/bar') as a
239 * package identifier. This isn't specific enough for packages in other repositories, so their
240 * stringified version is '@baz//foo/bar'.</p>
241 */
242 @Override
243 public String toString() {
244 return (repository.isDefault() ? "" : repository + "//") + pkgName;
245 }
246
247 @Override
248 public boolean equals(Object object) {
249 if (this == object) {
250 return true;
251 }
252 if (object instanceof PackageIdentifier) {
253 PackageIdentifier that = (PackageIdentifier) object;
254 return repository.equals(that.repository) && pkgName.equals(that.pkgName);
255 }
256 return false;
257 }
258
259 @Override
260 public int hashCode() {
261 return Objects.hash(repository, pkgName);
262 }
263
264 @Override
265 public int compareTo(PackageIdentifier that) {
266 return ComparisonChain.start()
267 .compare(repository.toString(), that.repository.toString())
268 .compare(pkgName, that.pkgName)
269 .result();
270 }
271}