-
Notifications
You must be signed in to change notification settings - Fork 3.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
HBASE-22623 - Add RegionObserver coprocessor hook for preWALAppend #390
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1104,4 +1104,16 @@ default DeleteTracker postInstantiateDeleteTracker( | |
throws IOException { | ||
return delTracker; | ||
} | ||
|
||
/** | ||
* Called just before the WAL Entry is appended to the WAL. Implementing this hook allows | ||
* coprocessors to add extended attributes to the WALKey that then get persisted to the | ||
* WAL, and are available to replication endpoints to use in processing WAL Entries. | ||
* @param ctx the environment provided by the region server | ||
* @param key the WALKey associated with a particular append to a WAL | ||
*/ | ||
default void preWALAppend(ObserverContext<RegionCoprocessorEnvironment> ctx, WALKey key, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Please mark this method as deprecated so we keep consistent on that WALEdit should not be exposed directly. And no postWALAppend seems a bit strange to users but I think it is fine here. We can add it when we actually want to use it. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @Apache9 Just to make sure I understand you right...you want me to write a method which is Deprecated, as of the moment of its creation? Adding a method to a public interface tells people they can use it. Deprecating a method tells people they should stop using it or refrain from starting to. Adding a deprecated method...just adds dead code. Someday soon, there's going to be a Phoenix code review for changes I'll make that use this method, and someone's going to say, "You can't use this method; it's deprecated. -1" And they'll be completely right to say so. Huh? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Phoenix already uses lots of deprecated methods in HBase, and I believe you can not find alternate solutions, for example, this one
You can add javadoc to say that now it is only supposed to be used in Phoenix, but I still think we should mark it as deprecated, otherwise it will be confusing that why in WALObserver, we say that WALEdit is private and should not be used, but then in RegionObserver, we allow users to use it. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The goal isn't to create a Phoenix-specific hack, but a general-purpose HBase API which Phoenix can use. You're at the same time complaining about Phoenix's previous and existing use of unapproved hooks while resisting efforts to create approved APIs the right way. preStoreFileReaderOpen wasn't created deprecated -- it's still alive and well in branch-1 -- but was deprecated in HBASE-18793, which you know because you were the one who deprecated it. Given that I'm backporting this new hook to branch-1, if I create it deprecated that presumably means that as soon as 1.5 releases, the new hook is eligible for cleanup in master via semver rules before it's even been released! The comments are confusing, not because of this patch, but because the position that created those comments is already self-contradictory in existing code. WALEdit must never be exposed to coprocessors...except the 6 times it already is in RegionObserver and the 4 times it is in WALObserver. When a policy leads to increasingly strange and nonsensical results -- and creating an already deprecated method is nonsensical -- it's time to rethink the policy. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The new method is not going to be deprecated out of the bat. We are not monolithic in our approach (and hostility) to coprocessor interfaces as a community. Imposing that disagreement on contributors is not appropriate. I am going to merge this as is and we can follow up on what should or should not be deprecated as a larger conversation on the future of coprocessors. The community has some big disagreements in approach. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I also agree the "policy", such as it is, is contradictory and confusing. We need to attack the bigger picture on dev@ in a discussion about the future of coprocessors and our tolerance (or not) to the requests of the Phoenix project. The opinions are not monolithic. There are some supporters, there are some hostile positions, both are valid in my view, we need to sort out the disagreement. This issue isn't the right scope for that. The contradictory positions are evident in the tug and pull of the suggestions to the contributor. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I copied these comments over onto the JIRA for visibility. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I sent an email with the subject "Coprocessors, clean ups, compatibility, deprecations, Phoenix... it's a bit of a mess" to dev@ There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Waiting 24 hours for time zone turn around. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This API will be used by adding to the passed in WALEdit? Using WALEdit #setters? |
||
WALEdit edit) | ||
throws IOException { | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -7951,6 +7951,11 @@ private WriteEntry doWALAppend(WALEdit walEdit, Durability durability, List<UUID | |
if (walEdit.isReplay()) { | ||
walKey.setOrigLogSeqNum(origLogSeqNum); | ||
} | ||
//don't call the coproc hook for writes to the WAL caused by | ||
//system lifecycle events like flushes or compactions | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why not? I'd think we would to be consistent? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. No this API is so we can get the WALKey and add an extended attribute just before it commits to the WAL from the RPC context. See above discussion and JIRA for context. |
||
if (this.coprocessorHost != null && !walEdit.isMetaEdit()) { | ||
this.coprocessorHost.preWALAppend(walKey, walEdit); | ||
} | ||
WriteEntry writeEntry = null; | ||
try { | ||
long txid = this.wal.append(this.getRegionInfo(), walKey, walEdit, true); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -34,7 +34,6 @@ | |
|
||
/** | ||
* Key for WAL Entry. | ||
* Read-only. No Setters. For limited audience such as Coprocessors. | ||
*/ | ||
@InterfaceAudience.LimitedPrivate({HBaseInterfaceAudience.REPLICATION, | ||
HBaseInterfaceAudience.COPROC}) | ||
|
@@ -86,6 +85,13 @@ default long getNonce() { | |
*/ | ||
long getOrigLogSeqNum(); | ||
|
||
/** | ||
* Add a named String value to this WALKey to be persisted into the WAL | ||
* @param attributeKey Name of the attribute | ||
* @param attributeValue Value of the attribute | ||
*/ | ||
void addExtendedAttribute(String attributeKey, byte[] attributeValue); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Oh, so WALKey goes from being read-only to now carrying burden? WALEdit is for freight? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think I've made this remark before... Looking for the response then, I see loads of commentary in issue... Let me review. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yeah, don't find where its ok to add freight to WALKey marked read-only (but alternative of letting WALEdit be mutable is a 'rat-hole'. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. WALKey is not read only. WALedit is not where we want to add the state The WALKey change is already committed. Having a WALedit parameter on this hook is useful whether it is read only or not. Still going to merge this now. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This change is the second half of work that began with HBASE-22622 WALKey Extended Attributes (#352) |
||
|
||
/** | ||
* Return a named String value injected into the WALKey during processing, such as by a | ||
* coprocessor | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -195,6 +195,37 @@ public WALKeyImpl(final byte[] encodedRegionName, | |
mvcc, null, null); | ||
} | ||
|
||
/** | ||
* Copy constructor that takes in an existing WALKeyImpl plus some extended attributes. | ||
* Intended for coprocessors to add annotations to a system-generated WALKey | ||
* for persistence to the WAL. | ||
* @param key Key to be copied into this new key | ||
* @param extendedAttributes Extra attributes to copy into the new key | ||
*/ | ||
public WALKeyImpl(WALKeyImpl key, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Instead of CP creating a new WALKey object, we can give API to add the ext attributes? It allows to add any custom key value pair so that later CPs or Replication EPs an make use. So we might not even allow the CP to return a brand new WALKey object. (?) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Also WALKeyImpl is marked LimitedPrivate({HBaseInterfaceAudience.REPLICATION}) I can see. So what this patch trying to do is to expose the class for CPs also. I can not see where WALKeyImpl to be used by Replication area. WALKey was enough. Anyways for this CP hook, IMO its enough to pass WALKey interface. We can add the API to add ext attributes into the interface. That is just adding ext attributes. It should be fine as it is not exposing any setters which takes other key attributes of the object. |
||
Map<String, byte[]> extendedAttributes){ | ||
init(key.getEncodedRegionName(), key.getTableName(), key.getSequenceId(), | ||
key.getWriteTime(), key.getClusterIds(), key.getNonceGroup(), key.getNonce(), | ||
key.getMvcc(), key.getReplicationScopes(), extendedAttributes); | ||
|
||
} | ||
|
||
/** | ||
* Copy constructor that takes in an existing WALKey, the extra WALKeyImpl fields that the | ||
* parent interface is missing, plus some extended attributes. Intended | ||
* for coprocessors to add annotations to a system-generated WALKey for | ||
* persistence to the WAL. | ||
*/ | ||
public WALKeyImpl(WALKey key, | ||
List<UUID> clusterIds, | ||
MultiVersionConcurrencyControl mvcc, | ||
final NavigableMap<byte[], Integer> replicationScopes, | ||
Map<String, byte[]> extendedAttributes){ | ||
init(key.getEncodedRegionName(), key.getTableName(), key.getSequenceId(), | ||
key.getWriteTime(), clusterIds, key.getNonceGroup(), key.getNonce(), | ||
mvcc, replicationScopes, extendedAttributes); | ||
|
||
} | ||
/** | ||
* Create the log key for writing to somewhere. | ||
* We maintain the tablename mainly for debugging purposes. | ||
|
@@ -464,6 +495,14 @@ public UUID getOriginatingClusterId(){ | |
return clusterIds.isEmpty()? HConstants.DEFAULT_CLUSTER_ID: clusterIds.get(0); | ||
} | ||
|
||
@Override | ||
public void addExtendedAttribute(String attributeKey, byte[] attributeValue){ | ||
if (extendedAttributes == null){ | ||
extendedAttributes = new HashMap<String, byte[]>(); | ||
} | ||
extendedAttributes.put(attributeKey, attributeValue); | ||
} | ||
|
||
@Override | ||
public byte[] getExtendedAttribute(String attributeKey){ | ||
return extendedAttributes != null ? extendedAttributes.get(attributeKey) : null; | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Having trouble understanding 'add extended attributes to the WALKey'. WALKey is read-only. You mean WALEdit here?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
WALKey is no longer read only. You can get and set extended attributes after HBASE-22622 WALKey Extended Attributes (#352)