26
26
import org .slf4j .Logger ;
27
27
import org .slf4j .LoggerFactory ;
28
28
29
+ import org .apache .hadoop .classification .InterfaceAudience ;
30
+ import org .apache .hadoop .classification .InterfaceStability ;
29
31
import org .apache .hadoop .fs .FileAlreadyExistsException ;
30
32
import org .apache .hadoop .fs .FileStatus ;
31
33
import org .apache .hadoop .fs .Path ;
54
56
* <li>If needed, one PUT</li>
55
57
* </ol>
56
58
*/
59
+ @ InterfaceAudience .Private
60
+ @ InterfaceStability .Evolving
57
61
public class MkdirOperation extends ExecutingStoreOperation <Boolean > {
58
62
59
63
private static final Logger LOG = LoggerFactory .getLogger (
60
64
MkdirOperation .class );
61
65
66
+ /**
67
+ * Path of the directory to be created.
68
+ */
62
69
private final Path dir ;
63
70
71
+ /**
72
+ * Mkdir Callbacks object to be used by the Mkdir operation.
73
+ */
64
74
private final MkdirCallbacks callbacks ;
65
75
66
76
/**
67
- * Should checks for ancestors existing be skipped?
68
- * This flag is set when working with magic directories.
77
+ * Whether to skip the validation of the parent directory.
78
+ */
79
+ private final boolean performanceMkdir ;
80
+
81
+ /**
82
+ * Whether the path is magic commit path.
69
83
*/
70
84
private final boolean isMagicPath ;
71
85
86
+ /**
87
+ * Initialize Mkdir Operation context for S3A.
88
+ *
89
+ * @param storeContext Store context.
90
+ * @param dir Dir path of the directory.
91
+ * @param callbacks MkdirCallbacks object used by the Mkdir operation.
92
+ * @param isMagicPath True if the path is magic commit path.
93
+ * @param performanceMkdir If true, skip validation of the parent directory
94
+ * structure.
95
+ */
72
96
public MkdirOperation (
73
97
final StoreContext storeContext ,
74
98
final Path dir ,
75
99
final MkdirCallbacks callbacks ,
76
- final boolean isMagicPath ) {
100
+ final boolean isMagicPath ,
101
+ final boolean performanceMkdir ) {
77
102
super (storeContext );
78
103
this .dir = dir ;
79
104
this .callbacks = callbacks ;
80
105
this .isMagicPath = isMagicPath ;
106
+ this .performanceMkdir = performanceMkdir ;
81
107
}
82
108
83
109
/**
@@ -124,7 +150,32 @@ public Boolean execute() throws IOException {
124
150
return true ;
125
151
}
126
152
127
- // Walk path to root, ensuring closest ancestor is a directory, not file
153
+ // if performance creation mode is set, no need to check
154
+ // whether the closest ancestor is dir.
155
+ if (!performanceMkdir ) {
156
+ verifyFileStatusOfClosestAncestor ();
157
+ }
158
+
159
+ // if we get here there is no directory at the destination.
160
+ // so create one.
161
+
162
+ // Create the marker file, delete the parent entries
163
+ // if the filesystem isn't configured to retain them
164
+ callbacks .createFakeDirectory (dir , false );
165
+ return true ;
166
+ }
167
+
168
+ /**
169
+ * Verify the file status of the closest ancestor, if it is
170
+ * dir, the mkdir operation should proceed. If it is file,
171
+ * the mkdir operation should throw error.
172
+ *
173
+ * @throws IOException If either file status could not be retrieved,
174
+ * or if the closest ancestor is a file.
175
+ */
176
+ private void verifyFileStatusOfClosestAncestor () throws IOException {
177
+ FileStatus fileStatus ;
178
+ // Walk path to root, ensuring the closest ancestor is a directory, not file
128
179
Path fPart = dir .getParent ();
129
180
try {
130
181
while (fPart != null && !fPart .isRoot ()) {
@@ -140,24 +191,18 @@ public Boolean execute() throws IOException {
140
191
}
141
192
142
193
// there's a file at the parent entry
143
- throw new FileAlreadyExistsException (String .format (
144
- "Can't make directory for path '%s' since it is a file." ,
145
- fPart ));
194
+ throw new FileAlreadyExistsException (
195
+ String .format (
196
+ "Can't make directory for path '%s' since it is a file." ,
197
+ fPart ));
146
198
}
147
199
} catch (AccessDeniedException e ) {
148
200
LOG .info ("mkdirs({}}: Access denied when looking"
149
201
+ " for parent directory {}; skipping checks" ,
150
- dir , fPart );
202
+ dir ,
203
+ fPart );
151
204
LOG .debug ("{}" , e , e );
152
205
}
153
-
154
- // if we get here there is no directory at the destination.
155
- // so create one.
156
-
157
- // Create the marker file, delete the parent entries
158
- // if the filesystem isn't configured to retain them
159
- callbacks .createFakeDirectory (dir , false );
160
- return true ;
161
206
}
162
207
163
208
/**
0 commit comments