Skip to content

Commit d39a4e4

Browse files
feat: Add ability to create a PCU Prefix at the object level (#2345)
* feat: Add ability to create a PCU Prefix at the object level * Allow no prefix and just create the objects at the ultimate object destination * Add in tests for the prefix and without a prefix * linter * Make using an actual prefix private * linter * add in to the directory where the part files will be created * Remove directory level in ObjectNamePrefix Naming Strategy * 🦉 Updates from OwlBot post-processor See https://github.com/googleapis/repo-automation-bots/blob/main/packages/owl-bot/README.md --------- Co-authored-by: Owl Bot <gcf-owl-bot[bot]@users.noreply.github.com>
1 parent c936551 commit d39a4e4

File tree

2 files changed

+80
-5
lines changed

2 files changed

+80
-5
lines changed

google-cloud-storage/src/main/java/com/google/cloud/storage/ParallelCompositeUploadBlobWriteSessionConfig.java

+63-5
Original file line numberDiff line numberDiff line change
@@ -462,10 +462,7 @@ String fmtName(String ultimateObjectName, PartRange partRange) {
462462

463463
// encode it to base 64, yielding 22 characters
464464
String randomKey = B64.encodeToString(bytes);
465-
HashCode hashCode =
466-
OBJECT_NAME_HASH_FUNCTION.hashString(ultimateObjectName, StandardCharsets.UTF_8);
467-
String nameDigest = B64.encodeToString(hashCode.asBytes());
468-
return fmtFields(randomKey, nameDigest, partRange.encode());
465+
return fmtFields(randomKey, ultimateObjectName, partRange.encode());
469466
}
470467

471468
abstract String fmtFields(String randomKey, String nameDigest, String partRange);
@@ -523,6 +520,35 @@ public static PartNamingStrategy prefix(String prefixPattern) {
523520
return new WithPrefix(rand, prefixPattern);
524521
}
525522

523+
/**
524+
* Strategy in which the end object name is the prefix included and is present on each part and
525+
* intermediary compose object.
526+
*
527+
* <p>General format is
528+
*
529+
* <pre><code>
530+
* {objectName}-{randomKeyDigest};{objectInfoDigest};{partIndex}.part
531+
* </code></pre>
532+
*
533+
* <p>{@code {objectInfoDigest}} will be fixed for an individual {@link BlobWriteSession}.
534+
*
535+
* <p><b><i>NOTE:</i></b>The way in which both {@code randomKeyDigest} and {@code
536+
* objectInfoDigest} are generated is undefined and subject to change at any time.
537+
*
538+
* @see #withPartNamingStrategy(PartNamingStrategy)
539+
* @since 2.30.2 This new api is in preview and is subject to breaking changes.
540+
*/
541+
@BetaApi
542+
public static PartNamingStrategy useObjectNameAsPrefix() {
543+
return useObjectNameAsPrefix("");
544+
}
545+
546+
private static PartNamingStrategy useObjectNameAsPrefix(String prefixPattern) {
547+
checkNotNull(prefixPattern, "prefixPattern must be non null");
548+
SecureRandom rand = new SecureRandom();
549+
return new WithObjectLevelPrefix(rand, prefixPattern);
550+
}
551+
526552
static final class WithPrefix extends PartNamingStrategy {
527553
private static final long serialVersionUID = 5709330763161570411L;
528554

@@ -534,7 +560,10 @@ private WithPrefix(SecureRandom rand, String prefix) {
534560
}
535561

536562
@Override
537-
protected String fmtFields(String randomKey, String nameDigest, String partRange) {
563+
protected String fmtFields(String randomKey, String ultimateObjectName, String partRange) {
564+
HashCode hashCode =
565+
OBJECT_NAME_HASH_FUNCTION.hashString(ultimateObjectName, StandardCharsets.UTF_8);
566+
String nameDigest = B64.encodeToString(hashCode.asBytes());
538567
return prefix
539568
+ "/"
540569
+ randomKey
@@ -546,6 +575,35 @@ protected String fmtFields(String randomKey, String nameDigest, String partRange
546575
}
547576
}
548577

578+
static final class WithObjectLevelPrefix extends PartNamingStrategy {
579+
580+
private static final long serialVersionUID = 5157942020618764450L;
581+
private final String prefix;
582+
583+
private WithObjectLevelPrefix(SecureRandom rand, String prefix) {
584+
super(rand);
585+
// If no prefix is specified we will create the part files under the same directory as the
586+
// ultimate object.
587+
this.prefix = prefix.isEmpty() ? prefix : prefix + "/";
588+
}
589+
590+
@Override
591+
protected String fmtFields(String randomKey, String ultimateObjectName, String partRange) {
592+
HashCode hashCode =
593+
OBJECT_NAME_HASH_FUNCTION.hashString(ultimateObjectName, StandardCharsets.UTF_8);
594+
String nameDigest = B64.encodeToString(hashCode.asBytes());
595+
return prefix
596+
+ ultimateObjectName
597+
+ "-"
598+
+ randomKey
599+
+ FIELD_SEPARATOR
600+
+ nameDigest
601+
+ FIELD_SEPARATOR
602+
+ partRange
603+
+ ".part";
604+
}
605+
}
606+
549607
static final class NoPrefix extends PartNamingStrategy {
550608
private static final long serialVersionUID = 5202415556658566017L;
551609

google-cloud-storage/src/test/java/com/google/cloud/storage/ParallelCompositeUploadBlobWriteSessionConfigTest.java

+17
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,23 @@ public void partNameStrategy_prefix_stillWorksWithFmtPattern() throws Exception
7070
() -> assertThat(fmt).startsWith("[%s]/"));
7171
}
7272

73+
@Test
74+
public void partNameStrategy_objectNamePrefix() throws Exception {
75+
// Creating an object level prefix without specifying an additional prefix will append the
76+
// object name to the beginning of the part name.
77+
PartNamingStrategy strategy = PartNamingStrategy.useObjectNameAsPrefix();
78+
79+
String fmt = strategy.fmtName("a/b/obj", PartRange.of(1, 96));
80+
assertAll(
81+
// random digest with prefix to spread over keyspace
82+
// digest is 22, objectName is 7, slash is 1
83+
() -> assertField(fmt, 0).hasLength(22 + 8),
84+
// name digest
85+
() -> assertField(fmt, 1).hasLength(22),
86+
() -> assertField(fmt, 2).isEqualTo("0001-0096.part"),
87+
() -> assertThat(fmt).startsWith("a/b/obj/"));
88+
}
89+
7390
private static StringSubject assertField(String fmt, int idx) {
7491
String[] split = fmt.split(";");
7592
String s = split[idx];

0 commit comments

Comments
 (0)