This is an automated email from the ASF dual-hosted git repository.

fanng pushed a commit to branch main
in repository https://gitbox.apache.org/repos/asf/gravitino.git


The following commit(s) were added to refs/heads/main by this push:
     new c9c9a99dff [#6076] improve(core): Support model pre event to Gravitino 
server (#6250)
c9c9a99dff is described below

commit c9c9a99dff6d5f90a08e1f1624554f79e8d531aa
Author: Lord of Abyss <103809695+abyss-l...@users.noreply.github.com>
AuthorDate: Tue Mar 11 13:04:55 2025 +0800

    [#6076] improve(core): Support model pre event to Gravitino server (#6250)
    
    ### What changes were proposed in this pull request?
    
    Support model pre event to Gravitino server, based on #6129 , Both
    synchronize Dispatcher changes with each other
    
    
    
https://docs.google.com/document/d/1_aVCd_tKiEebpzp9tg07Lzdk1j6EalNn8YOIdRn-3z4/edit?tab=t.0#heading=h.k85t4bueowc5
    
    #### PreEvent
    
    | PreEvent | OperationType | ModelCatalog |
    | --- | --- | --- |
    | `RegisterModelPreEvent` | `REGISTER_MODEL` | `registerModel` |
    | `GetModelPreEvent` | `LOAD_MODEL` | `getModel` |
    | `DeleteModelEvent` | `Delete_MODEL` | `deleteModel` |
    | `ListModelPreEvent` | `LIST_MODEL` | `listModels` |
    | `LinkModelVersionPreEvent` | `LINK_MODEL_VERSION` | `linkModelVersion`
    |
    | `GetModelVersionPreEvent` | `GET_MODEL_VERSION` | `getModelVersion` |
    | `DeleteModelVersionPreEvent` | `Delete_MODEL_VERSION` |
    `deleteModelVersion` |
    | `ListModelVersionsPreEvent` | `LIST_MODEL_VERSIONS` |
    `listModelVersions` |
    
    #### ModelEventDispatcher
    
    
    
![image](https://github.com/user-attachments/assets/ad80c319-f17f-4c76-9b65-b42c023075cb)
    
    ### Why are the changes needed?
    
    Fix: #6076
    
    ### Does this PR introduce _any_ user-facing change?
    
    No
    
    ### How was this patch tested?
    
    #### Model Event
    
    1. `testRegisterModelEvent`
    2. `testGetModelEvent`
    3. `testDeleteExistsModelEvent`
    4. `testDeleteNotExistsModelEvent`
    5. `testListModelEvent`
    
    #### Model Version Event
    
    1. `testLinkModelVersionEvent`
    2. `testGetModelVersionEventViaVersion`
    3. `testGetModelVersionEventViaAlias`
    4. `testDeleteModelVersionEventViaVersion`
    5. `testDeleteModelVersionEventViaAlias`
    6. `testDeleteModelVersionEventViaVersionNotExists`
    7. `testListModelVersionsEvent`
    
    ---------
    
    Signed-off-by: dependabot[bot] <supp...@github.com>
    Co-authored-by: luoxin <34674439+fourfrie...@users.noreply.github.com>
    Co-authored-by: luoxin5 <luox...@xiaomi.com>
    Co-authored-by: Yuhui <h...@datastrato.com>
    Co-authored-by: Qi Yu <y...@datastrato.com>
    Co-authored-by: YangJie <yangji...@baidu.com>
    Co-authored-by: this-user <this-u...@users.noreply.github.com>
    Co-authored-by: Qian Xia <lauraxiaq...@gmail.com>
    Co-authored-by: vitamin43 <104159582+vitami...@users.noreply.github.com>
    Co-authored-by: roryqi <ror...@apache.org>
    Co-authored-by: Pranay Kumar Karvi <pranayka...@gmail.com>
    Co-authored-by: AndreVale69 <57899285+andreval...@users.noreply.github.com>
    Co-authored-by: Abdullah Javed 
<65340910+javedabdul...@users.noreply.github.com>
    Co-authored-by: Brijesh Thummar <brijeshthumma...@gmail.com>
    Co-authored-by: Zhengke Zhou <madzh...@gmail.com>
    Co-authored-by: Justin Mclean <jus...@classsoftware.com>
    Co-authored-by: Jerry Shao <jerrys...@datastrato.com>
    Co-authored-by: Kang <zhoukan...@gmail.com>
    Co-authored-by: dependabot[bot] 
<49699333+dependabot[bot]@users.noreply.github.com>
    Co-authored-by: FANNG <xiaoj...@datastrato.com>
    Co-authored-by: Eric Chang <e850...@gmail.com>
    Co-authored-by: Tianhang <58762426+teoteo...@users.noreply.github.com>
    Co-authored-by: teo <litianh...@bilibili.com>
    Co-authored-by: teo <litianh...@bilibili.co>
---
 .../gravitino/listener/ModelEventDispatcher.java   | 270 +++++++++++++
 .../listener/api/event/DeleteModelPreEvent.java    |  48 +++
 .../api/event/DeleteModelVersionPreEvent.java      |  88 +++++
 .../listener/api/event/GetModelPreEvent.java       |  47 +++
 .../api/event/GetModelVersionPreEvent.java         |  78 ++++
 .../api/event/LinkModelVersionPreEvent.java        |  63 +++
 .../listener/api/event/ListModelPreEvent.java      |  60 +++
 .../api/event/ListModelVersionPreEvent.java        |  41 ++
 .../listener/api/event/ModelPreEvent.java          |  37 ++
 .../listener/api/event/OperationType.java          |  13 +
 .../api/event/RegisterAndLinkModelPreEvent.java    |  77 ++++
 .../listener/api/event/RegisterModelPreEvent.java  |  62 +++
 .../gravitino/listener/api/info/ModelInfo.java     | 118 ++++++
 .../listener/api/info/ModelVersionInfo.java        | 126 ++++++
 .../listener/api/event/TestModelEvent.java         | 427 +++++++++++++++++++++
 docs/gravitino-server-config.md                    |  19 +-
 16 files changed, 1565 insertions(+), 9 deletions(-)

diff --git 
a/core/src/main/java/org/apache/gravitino/listener/ModelEventDispatcher.java 
b/core/src/main/java/org/apache/gravitino/listener/ModelEventDispatcher.java
new file mode 100644
index 0000000000..9ffd7e8ae6
--- /dev/null
+++ b/core/src/main/java/org/apache/gravitino/listener/ModelEventDispatcher.java
@@ -0,0 +1,270 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener;
+
+import java.util.Map;
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.Namespace;
+import org.apache.gravitino.catalog.ModelDispatcher;
+import org.apache.gravitino.exceptions.ModelAlreadyExistsException;
+import 
org.apache.gravitino.exceptions.ModelVersionAliasesAlreadyExistException;
+import org.apache.gravitino.exceptions.NoSuchModelException;
+import org.apache.gravitino.exceptions.NoSuchModelVersionException;
+import org.apache.gravitino.exceptions.NoSuchSchemaException;
+import org.apache.gravitino.listener.api.event.DeleteModelPreEvent;
+import org.apache.gravitino.listener.api.event.DeleteModelVersionPreEvent;
+import org.apache.gravitino.listener.api.event.GetModelPreEvent;
+import org.apache.gravitino.listener.api.event.GetModelVersionPreEvent;
+import org.apache.gravitino.listener.api.event.LinkModelVersionPreEvent;
+import org.apache.gravitino.listener.api.event.ListModelPreEvent;
+import org.apache.gravitino.listener.api.event.ListModelVersionPreEvent;
+import org.apache.gravitino.listener.api.event.RegisterAndLinkModelPreEvent;
+import org.apache.gravitino.listener.api.event.RegisterModelPreEvent;
+import org.apache.gravitino.listener.api.info.ModelInfo;
+import org.apache.gravitino.listener.api.info.ModelVersionInfo;
+import org.apache.gravitino.model.Model;
+import org.apache.gravitino.model.ModelVersion;
+import org.apache.gravitino.utils.PrincipalUtils;
+
+/**
+ * {@code ModelEventDispatcher} is a decorator for {@link ModelDispatcher} 
that not only delegates
+ * model operations to the underlying catalog dispatcher but also dispatches 
corresponding events to
+ * an {@link org.apache.gravitino.listener.EventBus} after each operation is 
completed. This allows
+ * for event-driven workflows or monitoring of model operations.
+ */
+public class ModelEventDispatcher implements ModelDispatcher {
+  private final EventBus eventBus;
+  private final ModelDispatcher dispatcher;
+
+  /**
+   * Constructs a {@link ModelEventDispatcher} with a specified EventBus and 
{@link
+   * ModelDispatcher}.
+   *
+   * @param eventBus The EventBus to which events will be dispatched.
+   * @param dispatcher The underlying {@link ModelDispatcher} that will 
perform the actual model
+   *     operations.
+   */
+  public ModelEventDispatcher(EventBus eventBus, ModelDispatcher dispatcher) {
+    this.eventBus = eventBus;
+    this.dispatcher = dispatcher;
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public Model registerModel(NameIdentifier ident, String comment, Map<String, 
String> properties)
+      throws NoSuchSchemaException, ModelAlreadyExistsException {
+    ModelInfo registerRequest = new ModelInfo(ident.name(), properties, 
comment);
+    eventBus.dispatchEvent(
+        new RegisterModelPreEvent(PrincipalUtils.getCurrentUserName(), ident, 
registerRequest));
+    try {
+      Model model = dispatcher.registerModel(ident, comment, properties);
+      // TODO: ModelEvent
+      return model;
+    } catch (Exception e) {
+      // TODO: failureEvent
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public Model registerModel(
+      NameIdentifier ident,
+      String uri,
+      String[] aliases,
+      String comment,
+      Map<String, String> properties)
+      throws NoSuchSchemaException, ModelAlreadyExistsException,
+          ModelVersionAliasesAlreadyExistException {
+    ModelInfo registerModelRequest = new ModelInfo(ident.name(), properties, 
comment);
+    ModelVersionInfo linkModelVersionRequest =
+        new ModelVersionInfo(uri, comment, properties, aliases);
+
+    RegisterAndLinkModelPreEvent registerAndLinkModelPreEvent =
+        new RegisterAndLinkModelPreEvent(
+            PrincipalUtils.getCurrentUserName(),
+            ident,
+            registerModelRequest,
+            linkModelVersionRequest);
+    eventBus.dispatchEvent(registerAndLinkModelPreEvent);
+    try {
+      // TODO: ModelEvent
+      return dispatcher.registerModel(ident, uri, aliases, comment, 
properties);
+    } catch (Exception e) {
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public Model getModel(NameIdentifier ident) throws NoSuchModelException {
+    eventBus.dispatchEvent(new 
GetModelPreEvent(PrincipalUtils.getCurrentUserName(), ident));
+    try {
+      Model model = dispatcher.getModel(ident);
+      // TODO: ModelEvent
+      return model;
+    } catch (Exception e) {
+      // TODO: failureEvent
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public boolean deleteModel(NameIdentifier ident) {
+    eventBus.dispatchEvent(new 
DeleteModelPreEvent(PrincipalUtils.getCurrentUserName(), ident));
+    try {
+      // TODO: ModelEvent
+      return dispatcher.deleteModel(ident);
+    } catch (Exception e) {
+      // TODO: failureEvent
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public NameIdentifier[] listModels(Namespace namespace) throws 
NoSuchSchemaException {
+    eventBus.dispatchEvent(new 
ListModelPreEvent(PrincipalUtils.getCurrentUserName(), namespace));
+    try {
+      NameIdentifier[] models = dispatcher.listModels(namespace);
+      // TODO: ModelEvent
+      return models;
+    } catch (Exception e) {
+      // TODO: failureEvent
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public void linkModelVersion(
+      NameIdentifier ident,
+      String uri,
+      String[] aliases,
+      String comment,
+      Map<String, String> properties)
+      throws NoSuchModelException, ModelVersionAliasesAlreadyExistException {
+    ModelVersionInfo linkModelRequest = new ModelVersionInfo(uri, comment, 
properties, aliases);
+    eventBus.dispatchEvent(
+        new LinkModelVersionPreEvent(PrincipalUtils.getCurrentUserName(), 
ident, linkModelRequest));
+    try {
+      dispatcher.linkModelVersion(ident, uri, aliases, comment, properties);
+      // TODO: ModelEvent
+    } catch (Exception e) {
+      // TODO: failureEvent
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public ModelVersion getModelVersion(NameIdentifier ident, int version)
+      throws NoSuchModelVersionException {
+    eventBus.dispatchEvent(
+        new GetModelVersionPreEvent(PrincipalUtils.getCurrentUserName(), 
ident, null, version));
+    try {
+      // TODO: ModelEvent
+      return dispatcher.getModelVersion(ident, version);
+    } catch (Exception e) {
+      // TODO: failureEvent
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public ModelVersion getModelVersion(NameIdentifier ident, String alias)
+      throws NoSuchModelVersionException {
+    eventBus.dispatchEvent(
+        new GetModelVersionPreEvent(PrincipalUtils.getCurrentUserName(), 
ident, alias, null));
+    try {
+      ModelVersion modelVersion = dispatcher.getModelVersion(ident, alias);
+      // TODO: ModelEvent
+      return modelVersion;
+    } catch (Exception e) {
+      // TODO: failureEvent
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public boolean deleteModelVersion(NameIdentifier ident, int version) {
+    eventBus.dispatchEvent(
+        new DeleteModelVersionPreEvent(PrincipalUtils.getCurrentUserName(), 
ident, null, version));
+    try {
+      boolean isExists = dispatcher.deleteModelVersion(ident, version);
+      // TODO: ModelEvent
+      return isExists;
+    } catch (Exception e) {
+      // TODO: failureEvent
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public boolean deleteModelVersion(NameIdentifier ident, String alias) {
+    eventBus.dispatchEvent(
+        new DeleteModelVersionPreEvent(PrincipalUtils.getCurrentUserName(), 
ident, alias, null));
+    try {
+      boolean isExists = dispatcher.deleteModelVersion(ident, alias);
+      // TODO: ModelEvent
+      return isExists;
+    } catch (Exception e) {
+      // TODO: failureEvent
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public int[] listModelVersions(NameIdentifier ident) throws 
NoSuchModelException {
+    eventBus.dispatchEvent(
+        new ListModelVersionPreEvent(PrincipalUtils.getCurrentUserName(), 
ident));
+    try {
+      int[] versions = dispatcher.listModelVersions(ident);
+      // TODO: ModelEvent
+      return versions;
+    } catch (Exception e) {
+      // TODO: failureEvent
+      throw e;
+    }
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public boolean modelExists(NameIdentifier ident) {
+    return dispatcher.modelExists(ident);
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public boolean modelVersionExists(NameIdentifier ident, int version) {
+    return dispatcher.modelVersionExists(ident, version);
+  }
+
+  /** {@inheritDoc} */
+  @Override
+  public boolean modelVersionExists(NameIdentifier ident, String alias) {
+    return dispatcher.modelVersionExists(ident, alias);
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/DeleteModelPreEvent.java
 
b/core/src/main/java/org/apache/gravitino/listener/api/event/DeleteModelPreEvent.java
new file mode 100644
index 0000000000..2e8c452e22
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/DeleteModelPreEvent.java
@@ -0,0 +1,48 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.annotation.DeveloperApi;
+
+/** Represents an event triggered before deleting a model. */
+@DeveloperApi
+public class DeleteModelPreEvent extends ModelPreEvent {
+
+  /**
+   * Create a new {@link DeleteModelPreEvent} instance.
+   *
+   * @param user the user who triggered the event.
+   * @param identifier the identifier of the model being operated on.
+   */
+  public DeleteModelPreEvent(String user, NameIdentifier identifier) {
+    super(user, identifier);
+  }
+
+  /**
+   * Returns the type of operation.
+   *
+   * @return the operation type.
+   */
+  @Override
+  public OperationType operationType() {
+    return OperationType.DELETE_MODEL;
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/DeleteModelVersionPreEvent.java
 
b/core/src/main/java/org/apache/gravitino/listener/api/event/DeleteModelVersionPreEvent.java
new file mode 100644
index 0000000000..0b70db145d
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/DeleteModelVersionPreEvent.java
@@ -0,0 +1,88 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import java.util.Optional;
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.annotation.DeveloperApi;
+
+/** Represents an event triggered before deleting a model version. */
+@DeveloperApi
+public class DeleteModelVersionPreEvent extends ModelPreEvent {
+  private final Optional<String> alias;
+  private final Optional<Integer> version;
+
+  /**
+   * Create a new {@link DeleteModelVersionPreEvent} instance.
+   *
+   * @param user The username of the individual who initiated the model 
version linking.
+   * @param identifier The unique identifier of the model that was linked.
+   */
+  public DeleteModelVersionPreEvent(String user, NameIdentifier identifier) {
+    this(user, identifier, null, null);
+  }
+
+  /**
+   * Create a new {@link DeleteModelVersionPreEvent} instance. only one of 
alias or version are
+   * valid.
+   *
+   * @param user The username of the individual who initiated the model 
version deleted.
+   * @param identifier The unique identifier of the model that was deleted.
+   * @param alias The alias of the model version to be deleted.
+   * @param version The version of the model version to be deleted.
+   */
+  public DeleteModelVersionPreEvent(
+      String user, NameIdentifier identifier, String alias, Integer version) {
+    super(user, identifier);
+
+    this.alias = Optional.ofNullable(alias);
+    this.version = Optional.ofNullable(version);
+  }
+
+  /**
+   * Returns the alias of the model version to be deleted.
+   *
+   * @return A {@link Optional} instance containing the alias if it was 
provided, or an empty {@link
+   *     Optional} otherwise.
+   */
+  public Optional<String> alias() {
+    return alias;
+  }
+
+  /**
+   * Returns the version of the model version to be deleted.
+   *
+   * @return A {@link Optional} instance containing the version if it was 
provided, or an empty
+   *     {@link Optional} otherwise.
+   */
+  public Optional<Integer> version() {
+    return version;
+  }
+
+  /**
+   * Returns the type of operation.
+   *
+   * @return the operation type.
+   */
+  @Override
+  public OperationType operationType() {
+    return OperationType.DELETE_MODEL_VERSION;
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/GetModelPreEvent.java
 
b/core/src/main/java/org/apache/gravitino/listener/api/event/GetModelPreEvent.java
new file mode 100644
index 0000000000..1161a15688
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/GetModelPreEvent.java
@@ -0,0 +1,47 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.annotation.DeveloperApi;
+
+/** Represents an event triggered before getting a model. */
+@DeveloperApi
+public class GetModelPreEvent extends ModelPreEvent {
+  /**
+   * Create a new {@link GetModelPreEvent} instance.
+   *
+   * @param user the user who triggered the event.
+   * @param identifier the identifier of the model being operated on.
+   */
+  public GetModelPreEvent(String user, NameIdentifier identifier) {
+    super(user, identifier);
+  }
+
+  /**
+   * Returns the type of operation.
+   *
+   * @return the operation type.
+   */
+  @Override
+  public OperationType operationType() {
+    return OperationType.GET_MODEL;
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/GetModelVersionPreEvent.java
 
b/core/src/main/java/org/apache/gravitino/listener/api/event/GetModelVersionPreEvent.java
new file mode 100644
index 0000000000..3b671d8e31
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/GetModelVersionPreEvent.java
@@ -0,0 +1,78 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import java.util.Optional;
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.annotation.DeveloperApi;
+
+/** Represents an event triggered before getting the version of a model. */
+@DeveloperApi
+public class GetModelVersionPreEvent extends ModelPreEvent {
+  private final Optional<String> alias;
+  private final Optional<Integer> version;
+
+  /**
+   * Create a new {@link GetModelVersionPreEvent} instance with optional alias 
and version
+   * arguments. only one of alias or version are valid.
+   *
+   * @param user The username of the individual who initiated the model 
version to get.
+   * @param identifier The unique identifier of the model that was getting the 
version.
+   * @param alias The alias of the model version to get.
+   * @param version The version of the model version to get.
+   */
+  public GetModelVersionPreEvent(
+      String user, NameIdentifier identifier, String alias, Integer version) {
+    super(user, identifier);
+
+    this.alias = Optional.ofNullable(alias);
+    this.version = Optional.ofNullable(version);
+  }
+
+  /**
+   * Returns the alias of the model version to be deleted.
+   *
+   * @return A {@link Optional} instance containing the alias if it was 
provided, or an empty {@link
+   *     Optional} otherwise.
+   */
+  public Optional<String> alias() {
+    return alias;
+  }
+
+  /**
+   * Returns the version of the model version to be deleted.
+   *
+   * @return A {@link Optional} instance containing the version if it was 
provided, or an empty
+   *     {@link Optional} otherwise.
+   */
+  public Optional<Integer> version() {
+    return version;
+  }
+
+  /**
+   * Returns the type of operation.
+   *
+   * @return the operation type.
+   */
+  @Override
+  public OperationType operationType() {
+    return OperationType.GET_MODEL_VERSION;
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/LinkModelVersionPreEvent.java
 
b/core/src/main/java/org/apache/gravitino/listener/api/event/LinkModelVersionPreEvent.java
new file mode 100644
index 0000000000..7f05aa0a34
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/LinkModelVersionPreEvent.java
@@ -0,0 +1,63 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.annotation.DeveloperApi;
+import org.apache.gravitino.listener.api.info.ModelVersionInfo;
+
+/** Represents an event triggered before the linking of a model version. */
+@DeveloperApi
+public class LinkModelVersionPreEvent extends ModelPreEvent {
+  private final ModelVersionInfo linkModelVersionRequest;
+
+  /**
+   * Create a new {@link LinkModelVersionPreEvent} instance.
+   *
+   * @param user The username of the individual who initiated the model 
version linking.
+   * @param identifier The unique identifier of the model that was linked.
+   * @param linkModelVersionRequest The version information of the model that 
was requested to be
+   *     linked.
+   */
+  public LinkModelVersionPreEvent(
+      String user, NameIdentifier identifier, ModelVersionInfo 
linkModelVersionRequest) {
+    super(user, identifier);
+    this.linkModelVersionRequest = linkModelVersionRequest;
+  }
+
+  /**
+   * Retrieves the linked model version information.
+   *
+   * @return the model version information.
+   */
+  public ModelVersionInfo linkModelVersionRequest() {
+    return linkModelVersionRequest;
+  }
+
+  /**
+   * Returns the type of operation.
+   *
+   * @return the operation type.
+   */
+  @Override
+  public OperationType operationType() {
+    return OperationType.LINK_MODEL_VERSION;
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/ListModelPreEvent.java
 
b/core/src/main/java/org/apache/gravitino/listener/api/event/ListModelPreEvent.java
new file mode 100644
index 0000000000..75b444b3d1
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/ListModelPreEvent.java
@@ -0,0 +1,60 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.Namespace;
+import org.apache.gravitino.annotation.DeveloperApi;
+
+/** Represents an event triggered before listing of models within a namespace. 
*/
+@DeveloperApi
+public class ListModelPreEvent extends ModelPreEvent {
+  private final Namespace namespace;
+
+  /**
+   * Create a new {@link ListModelPreEvent} instance.
+   *
+   * @param user the user who triggered the event.
+   * @param namespace the namespace to list models from.
+   */
+  public ListModelPreEvent(String user, Namespace namespace) {
+    super(user, NameIdentifier.of(namespace.levels()));
+    this.namespace = namespace;
+  }
+
+  /**
+   * Provides the namespace associated with this event.
+   *
+   * @return A {@link Namespace} instance from which models were listed.
+   */
+  public Namespace namespace() {
+    return namespace;
+  }
+
+  /**
+   * Returns the type of operation.
+   *
+   * @return the operation type.
+   */
+  @Override
+  public OperationType operationType() {
+    return OperationType.LIST_MODEL;
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/ListModelVersionPreEvent.java
 
b/core/src/main/java/org/apache/gravitino/listener/api/event/ListModelVersionPreEvent.java
new file mode 100644
index 0000000000..fb18905b25
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/ListModelVersionPreEvent.java
@@ -0,0 +1,41 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.annotation.DeveloperApi;
+
+/** Represents an event triggered before listing model versions. */
+@DeveloperApi
+public class ListModelVersionPreEvent extends ModelPreEvent {
+  public ListModelVersionPreEvent(String user, NameIdentifier identifier) {
+    super(user, identifier);
+  }
+
+  /**
+   * Returns the type of operation.
+   *
+   * @return the operation type.
+   */
+  @Override
+  public OperationType operationType() {
+    return OperationType.LIST_MODEL_VERSIONS;
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/ModelPreEvent.java 
b/core/src/main/java/org/apache/gravitino/listener/api/event/ModelPreEvent.java
new file mode 100644
index 0000000000..11faf3deeb
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/ModelPreEvent.java
@@ -0,0 +1,37 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.annotation.DeveloperApi;
+
+/** Represents a pre-event for model operations. */
+@DeveloperApi
+public abstract class ModelPreEvent extends PreEvent {
+  /**
+   * Create a new {@link ModelPreEvent} instance.
+   *
+   * @param user the user who triggered the event.
+   * @param identifier the identifier of the model being operated on.
+   */
+  protected ModelPreEvent(String user, NameIdentifier identifier) {
+    super(user, identifier);
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/OperationType.java 
b/core/src/main/java/org/apache/gravitino/listener/api/event/OperationType.java
index 515e63a7c3..9ad171f10f 100644
--- 
a/core/src/main/java/org/apache/gravitino/listener/api/event/OperationType.java
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/OperationType.java
@@ -99,5 +99,18 @@ public enum OperationType {
   RENAME_VIEW,
   LIST_VIEW,
 
+  // Model event
+  REGISTER_MODEL,
+  DELETE_MODEL,
+  GET_MODEL,
+  LIST_MODEL,
+
+  // Model Version
+  LINK_MODEL_VERSION,
+  DELETE_MODEL_VERSION,
+  GET_MODEL_VERSION,
+  LIST_MODEL_VERSIONS,
+  REGISTER_AND_LINK_MODEL_VERSION,
+
   UNKNOWN,
 }
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/RegisterAndLinkModelPreEvent.java
 
b/core/src/main/java/org/apache/gravitino/listener/api/event/RegisterAndLinkModelPreEvent.java
new file mode 100644
index 0000000000..afce9bda8b
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/RegisterAndLinkModelPreEvent.java
@@ -0,0 +1,77 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.listener.api.info.ModelInfo;
+import org.apache.gravitino.listener.api.info.ModelVersionInfo;
+
+/** Represents an event triggered before registering a model and linking a 
model version. */
+public class RegisterAndLinkModelPreEvent extends ModelPreEvent {
+  private final ModelInfo registerModelRequest;
+  private final ModelVersionInfo linkModelVersionRequest;
+
+  /**
+   * Create a new {@code RegisterAndLinkModelEvent} instance.
+   *
+   * @param user the user who triggered the event.
+   * @param identifier the identifier of the model being operated on.
+   * @param registerModelRequest the model information that was requested to 
be registered.
+   * @param linkModelVersionRequest The version information of the model that 
was requested to be
+   *     linked.
+   */
+  public RegisterAndLinkModelPreEvent(
+      String user,
+      NameIdentifier identifier,
+      ModelInfo registerModelRequest,
+      ModelVersionInfo linkModelVersionRequest) {
+    super(user, identifier);
+    this.registerModelRequest = registerModelRequest;
+    this.linkModelVersionRequest = linkModelVersionRequest;
+  }
+
+  /**
+   * Retrieves the registered model information.
+   *
+   * @return the model information.
+   */
+  public ModelInfo registerModelRequest() {
+    return registerModelRequest;
+  }
+
+  /**
+   * Retrieves the linked model version information.
+   *
+   * @return the model version information.
+   */
+  public ModelVersionInfo linkModelVersionRequest() {
+    return linkModelVersionRequest;
+  }
+
+  /**
+   * Returns the type of operation.
+   *
+   * @return the operation type.
+   */
+  @Override
+  public OperationType operationType() {
+    return OperationType.REGISTER_AND_LINK_MODEL_VERSION;
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/event/RegisterModelPreEvent.java
 
b/core/src/main/java/org/apache/gravitino/listener/api/event/RegisterModelPreEvent.java
new file mode 100644
index 0000000000..905358c53d
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/event/RegisterModelPreEvent.java
@@ -0,0 +1,62 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.annotation.DeveloperApi;
+import org.apache.gravitino.listener.api.info.ModelInfo;
+
+/** Represents an event triggered before registering a model. */
+@DeveloperApi
+public class RegisterModelPreEvent extends ModelPreEvent {
+  private final ModelInfo registerModelRequest;
+
+  /**
+   * Create a new {@link RegisterModelPreEvent} instance.
+   *
+   * @param user the user who triggered the event.
+   * @param identifier the identifier of the model being operated on.
+   * @param registerModelRequest the model information that was requested to 
be registered.
+   */
+  public RegisterModelPreEvent(
+      String user, NameIdentifier identifier, ModelInfo registerModelRequest) {
+    super(user, identifier);
+    this.registerModelRequest = registerModelRequest;
+  }
+
+  /**
+   * Retrieves the registered model information.
+   *
+   * @return the model information.
+   */
+  public ModelInfo registerModelRequest() {
+    return registerModelRequest;
+  }
+
+  /**
+   * Returns the type of operation.
+   *
+   * @return the operation type.
+   */
+  @Override
+  public OperationType operationType() {
+    return OperationType.REGISTER_MODEL;
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/info/ModelInfo.java 
b/core/src/main/java/org/apache/gravitino/listener/api/info/ModelInfo.java
new file mode 100644
index 0000000000..870f725277
--- /dev/null
+++ b/core/src/main/java/org/apache/gravitino/listener/api/info/ModelInfo.java
@@ -0,0 +1,118 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.info;
+
+import com.google.common.collect.ImmutableMap;
+import java.util.Map;
+import java.util.Optional;
+import org.apache.gravitino.Audit;
+import org.apache.gravitino.annotation.DeveloperApi;
+import org.apache.gravitino.model.Model;
+
+/**
+ * ModelInfo exposes model information for event listener, it's supposed to be 
read only. Most of
+ * the fields are shallow copied internally not deep copies for performance.
+ */
+@DeveloperApi
+public class ModelInfo {
+  private final String name;
+  private final Map<String, String> properties;
+  private final Optional<String> comment;
+  private final Optional<Audit> audit;
+  private final Optional<Integer> lastVersion;
+
+  /**
+   * Constructs model information based on a given model.
+   *
+   * @param model the model to expose information for.
+   */
+  public ModelInfo(Model model) {
+    this.name = model.name();
+    this.properties =
+        model.properties() == null ? ImmutableMap.of() : 
ImmutableMap.copyOf(model.properties());
+
+    this.comment = Optional.ofNullable(model.comment());
+    this.audit = Optional.ofNullable(model.auditInfo());
+    this.lastVersion = Optional.ofNullable(model.latestVersion());
+  }
+
+  public ModelInfo(String name, Map<String, String> properties, String 
comment) {
+    this(name, properties, comment, null, null);
+  }
+
+  public ModelInfo(
+      String name,
+      Map<String, String> properties,
+      String comment,
+      Audit audit,
+      Integer lastVersion) {
+    this.name = name;
+
+    this.properties = properties == null ? ImmutableMap.of() : 
ImmutableMap.copyOf(properties);
+    this.comment = Optional.ofNullable(comment);
+    this.audit = Optional.ofNullable(audit);
+    this.lastVersion = Optional.ofNullable(lastVersion);
+  }
+
+  /**
+   * Returns the name of the model.
+   *
+   * @return the name of the model.
+   */
+  public String name() {
+    return name;
+  }
+
+  /**
+   * Returns the properties of the model.
+   *
+   * @return the properties of the model.
+   */
+  public Map<String, String> properties() {
+    return properties;
+  }
+
+  /**
+   * Returns the comment of the model.
+   *
+   * @return the comment of the model or null if not set.
+   */
+  public Optional<String> comment() {
+    return comment;
+  }
+
+  /**
+   * Returns the audit information of the model.
+   *
+   * @return the audit information of the model or null if not set.
+   */
+  public Optional<Audit> audit() {
+    return audit;
+  }
+
+  /**
+   * returns the last version of the model.
+   *
+   * @return the last version of the model, or empty if not set.
+   */
+  public Optional<Integer> lastVersion() {
+    return lastVersion;
+  }
+}
diff --git 
a/core/src/main/java/org/apache/gravitino/listener/api/info/ModelVersionInfo.java
 
b/core/src/main/java/org/apache/gravitino/listener/api/info/ModelVersionInfo.java
new file mode 100644
index 0000000000..a9a953a86e
--- /dev/null
+++ 
b/core/src/main/java/org/apache/gravitino/listener/api/info/ModelVersionInfo.java
@@ -0,0 +1,126 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.info;
+
+import com.google.common.collect.ImmutableMap;
+import java.util.Map;
+import java.util.Optional;
+import org.apache.gravitino.Audit;
+import org.apache.gravitino.model.ModelVersion;
+
+/**
+ * {@link ModelVersionInfo} exposes model version information for event 
listener, it's supposed to
+ * be read only. Most of the fields are shallow copied internally not deep 
copies for performance.
+ */
+public class ModelVersionInfo {
+  private final String uri;
+  private final Map<String, String> properties;
+
+  private final Optional<String[]> aliases;
+  private final Optional<String> comment;
+  private final Optional<Audit> auditInfo;
+
+  /**
+   * Constructs model version information based on a given {@link 
ModelVersion}.
+   *
+   * @param modelVersion the model version to expose information for.
+   */
+  public ModelVersionInfo(ModelVersion modelVersion) {
+    this(
+        modelVersion.uri(),
+        modelVersion.comment(),
+        modelVersion.properties(),
+        modelVersion.aliases(),
+        modelVersion.auditInfo());
+  }
+
+  public ModelVersionInfo(
+      String uri, String comment, Map<String, String> properties, String[] 
aliases) {
+    this(uri, comment, properties, aliases, null);
+  }
+
+  /**
+   * Constructs model version information based on a given arguments.
+   *
+   * @param uri
+   * @param aliases
+   * @param comment
+   * @param properties
+   */
+  public ModelVersionInfo(
+      String uri,
+      String comment,
+      Map<String, String> properties,
+      String[] aliases,
+      Audit auditInfo) {
+    this.uri = uri;
+
+    this.properties = properties == null ? ImmutableMap.of() : 
ImmutableMap.copyOf(properties);
+    this.comment = Optional.ofNullable(comment);
+    this.auditInfo = Optional.ofNullable(auditInfo);
+    this.aliases = Optional.ofNullable(aliases);
+  }
+
+  /**
+   * Returns the URI of the model version.
+   *
+   * @return the URI of the model version.
+   */
+  public String uri() {
+    return uri;
+  }
+
+  /**
+   * Returns the properties associated with the model version.
+   *
+   * @return Map of table properties.
+   */
+  public Map<String, String> properties() {
+    return properties;
+  }
+
+  /**
+   * Returns the aliases of the model version.
+   *
+   * @return the aliases of the model version (a {@code Optional<String[]>} 
instance) or null if not
+   *     set.
+   */
+  public Optional<String[]> aliases() {
+    return aliases;
+  }
+
+  /**
+   * Returns the comment of the model version.
+   *
+   * @return the comment of the model version or null if not set.
+   */
+  public Optional<String> comment() {
+    return comment;
+  }
+
+  /**
+   * Returns the audit information of the model version.
+   *
+   * @return the audit information of the model version or null if not set.
+   */
+  public Optional<Audit> audit() {
+    return auditInfo;
+  }
+}
diff --git 
a/core/src/test/java/org/apache/gravitino/listener/api/event/TestModelEvent.java
 
b/core/src/test/java/org/apache/gravitino/listener/api/event/TestModelEvent.java
new file mode 100644
index 0000000000..8489301435
--- /dev/null
+++ 
b/core/src/test/java/org/apache/gravitino/listener/api/event/TestModelEvent.java
@@ -0,0 +1,427 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you 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 org.apache.gravitino.listener.api.event;
+
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+import com.google.common.collect.ImmutableMap;
+import java.time.Instant;
+import java.util.Collections;
+import java.util.Map;
+import org.apache.gravitino.Audit;
+import org.apache.gravitino.NameIdentifier;
+import org.apache.gravitino.Namespace;
+import org.apache.gravitino.catalog.ModelDispatcher;
+import org.apache.gravitino.exceptions.GravitinoRuntimeException;
+import org.apache.gravitino.listener.DummyEventListener;
+import org.apache.gravitino.listener.EventBus;
+import org.apache.gravitino.listener.ModelEventDispatcher;
+import org.apache.gravitino.listener.api.info.ModelInfo;
+import org.apache.gravitino.listener.api.info.ModelVersionInfo;
+import org.apache.gravitino.model.Model;
+import org.apache.gravitino.model.ModelVersion;
+import org.apache.gravitino.utils.NameIdentifierUtil;
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.BeforeAll;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.TestInstance;
+
+@TestInstance(TestInstance.Lifecycle.PER_CLASS)
+public class TestModelEvent {
+  private ModelEventDispatcher dispatcher;
+  private ModelEventDispatcher failureDispatcher;
+  private DummyEventListener dummyEventListener;
+  private Model modelA;
+  private Model modelB;
+  private NameIdentifier existingIdentA;
+  private NameIdentifier existingIdentB;
+  private NameIdentifier notExistingIdent;
+  private Namespace namespace;
+  private ModelVersion firstModelVersion;
+  private ModelVersion secondModelVersion;
+
+  @BeforeAll
+  void init() {
+    this.namespace = Namespace.of("metalake", "catalog", "schema");
+    this.modelA = getMockModel("modelA", "commentA");
+    this.modelB = getMockModel("modelB", "commentB");
+    this.firstModelVersion =
+        mockModelVersion("uriA", new String[] {"aliasProduction"}, 
"versionInfoA");
+    this.secondModelVersion = mockModelVersion("uriB", new String[] 
{"aliasTest"}, "versionInfoB");
+    System.out.println(secondModelVersion.toString());
+    this.existingIdentA = NameIdentifierUtil.ofModel("metalake", "catalog", 
"schema", "modelA");
+    this.existingIdentB = NameIdentifierUtil.ofModel("metalake", "catalog", 
"schema", "modelB");
+    this.notExistingIdent =
+        NameIdentifierUtil.ofModel("metalake", "catalog", "schema", 
"not_exist");
+    this.dummyEventListener = new DummyEventListener();
+
+    EventBus eventBus = new 
EventBus(Collections.singletonList(dummyEventListener));
+    this.dispatcher = new ModelEventDispatcher(eventBus, mockTagDispatcher());
+    this.failureDispatcher = new ModelEventDispatcher(eventBus, 
mockExceptionModelDispatcher());
+    // TODO: add failure dispatcher tests.
+    System.out.println(this.failureDispatcher.toString());
+  }
+
+  @Test
+  void testModelInfo() {
+    Model mockModel = getMockModel("model", "comment");
+    ModelInfo modelInfo = new ModelInfo(mockModel);
+
+    Assertions.assertEquals("model", modelInfo.name());
+    Assertions.assertEquals(1, modelInfo.properties().size());
+    Assertions.assertEquals("#FFFFFF", modelInfo.properties().get("color"));
+
+    Assertions.assertTrue(modelInfo.comment().isPresent());
+    String comment = modelInfo.comment().get();
+    Assertions.assertEquals("comment", comment);
+
+    Assertions.assertFalse(modelInfo.audit().isPresent());
+
+    Assertions.assertTrue(modelInfo.lastVersion().isPresent());
+    int lastVersion = modelInfo.lastVersion().get();
+    Assertions.assertEquals(1, lastVersion);
+  }
+
+  @Test
+  void testModelInfoWithoutComment() {
+    Model mockModel = getMockModel("model", null);
+    ModelInfo modelInfo = new ModelInfo(mockModel);
+
+    Assertions.assertFalse(modelInfo.comment().isPresent());
+  }
+
+  @Test
+  void testModelInfoWithAudit() {
+    Model mockModel = getMockModelWithAudit("model", "comment");
+    ModelInfo modelInfo = new ModelInfo(mockModel);
+
+    Assertions.assertEquals("model", modelInfo.name());
+    Assertions.assertEquals(1, modelInfo.properties().size());
+    Assertions.assertEquals("#FFFFFF", modelInfo.properties().get("color"));
+
+    Assertions.assertTrue(modelInfo.comment().isPresent());
+    String comment = modelInfo.comment().get();
+    Assertions.assertEquals("comment", comment);
+
+    Assertions.assertTrue(modelInfo.audit().isPresent());
+    Audit audit = modelInfo.audit().get();
+    Assertions.assertEquals("demo_user", audit.creator());
+    Assertions.assertEquals(1611111111111L, audit.createTime().toEpochMilli());
+    Assertions.assertEquals("demo_user", audit.lastModifier());
+    Assertions.assertEquals(1611111111111L, 
audit.lastModifiedTime().toEpochMilli());
+  }
+
+  @Test
+  void testRegisterModelEvent() {
+    dispatcher.registerModel(existingIdentA, "commentA", 
ImmutableMap.of("color", "#FFFFFF"));
+
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(RegisterModelPreEvent.class, preEvent.getClass());
+    Assertions.assertEquals(OperationType.REGISTER_MODEL, 
preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    // validate pre-event model info
+    RegisterModelPreEvent registerModelPreEvent = (RegisterModelPreEvent) 
preEvent;
+    Assertions.assertEquals(existingIdentA, 
registerModelPreEvent.identifier());
+    ModelInfo modelInfoPreEvent = registerModelPreEvent.registerModelRequest();
+
+    Assertions.assertEquals("modelA", modelInfoPreEvent.name());
+    Assertions.assertEquals(ImmutableMap.of("color", "#FFFFFF"), 
modelInfoPreEvent.properties());
+    Assertions.assertTrue(modelInfoPreEvent.comment().isPresent());
+    String comment = modelInfoPreEvent.comment().get();
+    Assertions.assertEquals("commentA", comment);
+    Assertions.assertFalse(modelInfoPreEvent.audit().isPresent());
+    Assertions.assertFalse(modelInfoPreEvent.lastVersion().isPresent());
+  }
+
+  @Test
+  void testRegisterAndLinkModelEvent() {
+    dispatcher.registerModel(
+        existingIdentA,
+        "uriA",
+        new String[] {"aliasProduction", "aliasTest"},
+        "commentA",
+        ImmutableMap.of("color", "#FFFFFF"));
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(RegisterAndLinkModelPreEvent.class, 
preEvent.getClass());
+    Assertions.assertEquals(
+        OperationType.REGISTER_AND_LINK_MODEL_VERSION, 
preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    // validate pre-event model info
+    RegisterAndLinkModelPreEvent registerAndLinkModelPreEvent =
+        (RegisterAndLinkModelPreEvent) preEvent;
+    ModelInfo registerModelRequest = 
registerAndLinkModelPreEvent.registerModelRequest();
+
+    Assertions.assertEquals("modelA", registerModelRequest.name());
+    Assertions.assertEquals(ImmutableMap.of("color", "#FFFFFF"), 
registerModelRequest.properties());
+    Assertions.assertTrue(registerModelRequest.comment().isPresent());
+    String comment = registerModelRequest.comment().get();
+    Assertions.assertEquals("commentA", comment);
+    Assertions.assertFalse(registerModelRequest.audit().isPresent());
+    Assertions.assertFalse(registerModelRequest.lastVersion().isPresent());
+
+    // validate pre-event model version info
+    ModelVersionInfo modelVersionInfoPreEvent =
+        registerAndLinkModelPreEvent.linkModelVersionRequest();
+    Assertions.assertEquals("uriA", modelVersionInfoPreEvent.uri());
+    Assertions.assertTrue(modelVersionInfoPreEvent.aliases().isPresent());
+    String[] aliases = modelVersionInfoPreEvent.aliases().get();
+    Assertions.assertEquals(2, aliases.length);
+    Assertions.assertEquals("aliasProduction", aliases[0]);
+    Assertions.assertEquals("aliasTest", aliases[1]);
+  }
+
+  @Test
+  void testGetModelEvent() {
+    dispatcher.getModel(existingIdentA);
+
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(GetModelPreEvent.class, preEvent.getClass());
+    Assertions.assertEquals(OperationType.GET_MODEL, preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    GetModelPreEvent getModelPreEvent = (GetModelPreEvent) preEvent;
+    Assertions.assertEquals(existingIdentA, getModelPreEvent.identifier());
+  }
+
+  @Test
+  void testDeleteExistsModelEvent() {
+    dispatcher.deleteModel(existingIdentA);
+
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(DeleteModelPreEvent.class, preEvent.getClass());
+    Assertions.assertEquals(OperationType.DELETE_MODEL, 
preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    DeleteModelPreEvent deleteModelPreEvent = (DeleteModelPreEvent) preEvent;
+    Assertions.assertEquals(existingIdentA, deleteModelPreEvent.identifier());
+  }
+
+  @Test
+  void testListModelEvent() {
+    dispatcher.listModels(namespace);
+
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(ListModelPreEvent.class, preEvent.getClass());
+    Assertions.assertEquals(OperationType.LIST_MODEL, 
preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    ListModelPreEvent listModelPreEvent = (ListModelPreEvent) preEvent;
+    Assertions.assertEquals(namespace, listModelPreEvent.namespace());
+  }
+
+  @Test
+  void testLinkModelVersionEvent() {
+    dispatcher.linkModelVersion(
+        existingIdentA,
+        "uriA",
+        new String[] {"aliasProduction", "aliasTest"},
+        "versionInfoA",
+        ImmutableMap.of("color", "#FFFFFF"));
+
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(LinkModelVersionPreEvent.class, 
preEvent.getClass());
+    Assertions.assertEquals(OperationType.LINK_MODEL_VERSION, 
preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    LinkModelVersionPreEvent linkModelVersionPreEvent = 
(LinkModelVersionPreEvent) preEvent;
+    Assertions.assertEquals(existingIdentA, 
linkModelVersionPreEvent.identifier());
+    ModelVersionInfo modelVersionInfo = 
linkModelVersionPreEvent.linkModelVersionRequest();
+
+    Assertions.assertEquals(1, modelVersionInfo.properties().size());
+    Assertions.assertEquals("#FFFFFF", 
modelVersionInfo.properties().get("color"));
+
+    Assertions.assertEquals("uriA", modelVersionInfo.uri());
+    Assertions.assertTrue(modelVersionInfo.aliases().isPresent());
+    String[] aliases = modelVersionInfo.aliases().get();
+    Assertions.assertEquals(2, aliases.length);
+    Assertions.assertEquals("aliasProduction", aliases[0]);
+    Assertions.assertEquals("aliasTest", aliases[1]);
+
+    Assertions.assertTrue(modelVersionInfo.comment().isPresent());
+    String comment = modelVersionInfo.comment().get();
+    Assertions.assertEquals("versionInfoA", comment);
+
+    Assertions.assertFalse(modelVersionInfo.audit().isPresent());
+  }
+
+  @Test
+  void testGetModelVersionEventViaVersion() {
+    dispatcher.getModelVersion(existingIdentA, 1);
+
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(GetModelVersionPreEvent.class, 
preEvent.getClass());
+    Assertions.assertEquals(OperationType.GET_MODEL_VERSION, 
preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    GetModelVersionPreEvent getModelVersionPreEvent = 
(GetModelVersionPreEvent) preEvent;
+    Assertions.assertEquals(existingIdentA, 
getModelVersionPreEvent.identifier());
+  }
+
+  @Test
+  void testGetModelVersionEventViaAlias() {
+    dispatcher.getModelVersion(existingIdentB, "aliasTest");
+
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(GetModelVersionPreEvent.class, 
preEvent.getClass());
+    Assertions.assertEquals(OperationType.GET_MODEL_VERSION, 
preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    GetModelVersionPreEvent getModelVersionPreEvent = 
(GetModelVersionPreEvent) preEvent;
+    Assertions.assertEquals(existingIdentB, 
getModelVersionPreEvent.identifier());
+    Assertions.assertTrue(getModelVersionPreEvent.alias().isPresent());
+    Assertions.assertEquals("aliasTest", 
getModelVersionPreEvent.alias().get());
+    Assertions.assertFalse(getModelVersionPreEvent.version().isPresent());
+  }
+
+  @Test
+  void testDeleteModelVersionEventViaVersion() {
+    dispatcher.deleteModelVersion(existingIdentA, 1);
+
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(DeleteModelVersionPreEvent.class, 
preEvent.getClass());
+    Assertions.assertEquals(OperationType.DELETE_MODEL_VERSION, 
preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    DeleteModelVersionPreEvent deleteModelVersionPreEvent = 
(DeleteModelVersionPreEvent) preEvent;
+    Assertions.assertEquals(existingIdentA, 
deleteModelVersionPreEvent.identifier());
+    Assertions.assertTrue(deleteModelVersionPreEvent.version().isPresent());
+    Assertions.assertEquals(1, deleteModelVersionPreEvent.version().get());
+    Assertions.assertFalse(deleteModelVersionPreEvent.alias().isPresent());
+  }
+
+  @Test
+  void testDeleteModelVersionEventViaAlias() {
+    dispatcher.deleteModelVersion(existingIdentB, "aliasTest");
+
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(DeleteModelVersionPreEvent.class, 
preEvent.getClass());
+    Assertions.assertEquals(OperationType.DELETE_MODEL_VERSION, 
preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    DeleteModelVersionPreEvent deleteModelVersionPreEvent = 
(DeleteModelVersionPreEvent) preEvent;
+    Assertions.assertEquals(existingIdentB, 
deleteModelVersionPreEvent.identifier());
+    Assertions.assertTrue(deleteModelVersionPreEvent.alias().isPresent());
+    Assertions.assertEquals("aliasTest", 
deleteModelVersionPreEvent.alias().get());
+    Assertions.assertFalse(deleteModelVersionPreEvent.version().isPresent());
+  }
+
+  @Test
+  void testListModelVersionsEvent() {
+    dispatcher.listModelVersions(existingIdentA);
+
+    // validate pre-event
+    PreEvent preEvent = dummyEventListener.popPreEvent();
+    Assertions.assertEquals(ListModelVersionPreEvent.class, 
preEvent.getClass());
+    Assertions.assertEquals(OperationType.LIST_MODEL_VERSIONS, 
preEvent.operationType());
+    Assertions.assertEquals(OperationStatus.UNPROCESSED, 
preEvent.operationStatus());
+
+    ListModelVersionPreEvent listModelVersionsPreEvent = 
(ListModelVersionPreEvent) preEvent;
+    Assertions.assertEquals(existingIdentA, 
listModelVersionsPreEvent.identifier());
+  }
+
+  private ModelDispatcher mockExceptionModelDispatcher() {
+    return mock(
+        ModelDispatcher.class,
+        invocation -> {
+          throw new GravitinoRuntimeException("Exception for all methods");
+        });
+  }
+
+  private ModelDispatcher mockTagDispatcher() {
+    ModelDispatcher dispatcher = mock(ModelDispatcher.class);
+
+    when(dispatcher.registerModel(existingIdentA, "commentA", 
ImmutableMap.of("color", "#FFFFFF")))
+        .thenReturn(modelA);
+    when(dispatcher.registerModel(existingIdentB, "commentB", 
ImmutableMap.of("color", "#FFFFFF")))
+        .thenReturn(modelB);
+
+    when(dispatcher.getModel(existingIdentA)).thenReturn(modelA);
+    when(dispatcher.getModel(existingIdentB)).thenReturn(modelB);
+
+    when(dispatcher.deleteModel(existingIdentA)).thenReturn(true);
+    when(dispatcher.deleteModel(notExistingIdent)).thenReturn(false);
+
+    when(dispatcher.listModels(namespace))
+        .thenReturn(new NameIdentifier[] {existingIdentA, existingIdentB});
+
+    when(dispatcher.getModelVersion(existingIdentA, 
1)).thenReturn(firstModelVersion);
+    when(dispatcher.getModelVersion(existingIdentB, 
"aliasTest")).thenReturn(secondModelVersion);
+
+    when(dispatcher.deleteModelVersion(existingIdentA, 1)).thenReturn(true);
+    when(dispatcher.deleteModelVersion(existingIdentB, 
"aliasTest")).thenReturn(true);
+    when(dispatcher.deleteModelVersion(existingIdentA, 3)).thenReturn(false);
+
+    when(dispatcher.listModelVersions(existingIdentA)).thenReturn(new int[] 
{1, 2});
+
+    return dispatcher;
+  }
+
+  private Model getMockModel(String name, String comment) {
+    return getMockModel(name, comment, ImmutableMap.of("color", "#FFFFFF"));
+  }
+
+  private Model getMockModel(String name, String comment, Map<String, String> 
properties) {
+    Model mockModel = mock(Model.class);
+    when(mockModel.name()).thenReturn(name);
+    when(mockModel.comment()).thenReturn(comment);
+    when(mockModel.properties()).thenReturn(properties);
+    when(mockModel.latestVersion()).thenReturn(1);
+    when(mockModel.auditInfo()).thenReturn(null);
+
+    return mockModel;
+  }
+
+  private Model getMockModelWithAudit(String name, String comment) {
+    Model model = getMockModel(name, comment);
+    Audit mockAudit = mock(Audit.class);
+
+    when(mockAudit.creator()).thenReturn("demo_user");
+    
when(mockAudit.createTime()).thenReturn(Instant.ofEpochMilli(1611111111111L));
+    when(mockAudit.lastModifier()).thenReturn("demo_user");
+    
when(mockAudit.lastModifiedTime()).thenReturn(Instant.ofEpochMilli(1611111111111L));
+    when(model.auditInfo()).thenReturn(mockAudit);
+
+    return model;
+  }
+
+  private ModelVersion mockModelVersion(String uri, String[] aliases, String 
comment) {
+    ModelVersion modelVersion = mock(ModelVersion.class);
+    when(modelVersion.version()).thenReturn(1);
+    when(modelVersion.uri()).thenReturn(uri);
+    when(modelVersion.aliases()).thenReturn(aliases);
+    when(modelVersion.comment()).thenReturn("model version " + comment);
+    when(modelVersion.properties()).thenReturn(ImmutableMap.of("color", 
"#FFFFFF"));
+
+    return modelVersion;
+  }
+}
diff --git a/docs/gravitino-server-config.md b/docs/gravitino-server-config.md
index 88edf23470..3f30c2f2c2 100644
--- a/docs/gravitino-server-config.md
+++ b/docs/gravitino-server-config.md
@@ -129,15 +129,16 @@ Gravitino triggers a pre-event before the operation, a 
post-event after the comp
 
 ##### Pre-event
 
-| Operation type                       | Pre-event                             
                                                                                
                                                                                
    | Since Version     |
-|--------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------------|
-| Iceberg REST server table operation  | `IcebergCreateTablePreEvent`, 
`IcebergUpdateTablePreEvent`, `IcebergDropTablePreEvent`, 
`IcebergLoadTablePreEvent`, `IcebergListTablePreEvent`, 
`IcebergTableExistsPreEvent`, `IcebergRenameTablePreEvent` | 0.7.0-incubating  |
-| Gravitino server table operation     | `CreateTablePreEvent`, 
`AlterTablePreEvent`, `DropTablePreEvent`, `PurgeTablePreEvent`, 
`LoadTablePreEvent`, `ListTablePreEvent`                                        
                                  | 0.8.0-incubating  |
-| Gravitino server schema operation    | `CreateSchemaPreEvent`, 
`AlterSchemaPreEvent`, `DropSchemaPreEvent`, `LoadSchemaPreEvent`, 
`ListSchemaPreEvent`                                                            
                               | 0.8.0-incubating  |
-| Gravitino server catalog operation   | `CreateCatalogPreEvent`, 
`AlterCatalogPreEvent`, `DropCatalogPreEvent`, `LoadCatalogPreEvent`, 
`ListCatalogPreEvent`                                                           
                           | 0.8.0-incubating  |
-| Gravitino server metalake operation  | `CreateMetalakePreEvent`, 
`AlterMetalakePreEvent`,`DropMetalakePreEvent`,`LoadMetalakePreEvent`,`ListMetalakePreEvent`
                                                                                
    | 0.8.0-incubating |
-| Gravitino server partition operation | `AddPartitionPreEvent`, 
`DropPartitionPreEvent`, `GetPartitionPreEvent`, 
`PurgePartitionPreEvent`,`ListPartitionPreEvent`,`ListPartitionNamesPreEvent`   
                                                 | 0.8.0-incubating |
-| Gravitino server fileset operation   | `CreateFilesetPreEvent`, 
`AlterFilesetPreEvent`, `DropFilesetPreEvent`, 
`LoadFilesetPreEvent`,`ListFilesetPreEvent`,`GetFileLocationPreEvent`           
                                                  | 0.8.0-incubating |
+| Operation type                       | Pre-event                             
                                                                                
                                                                                
                                                                                
                                     | Since Version    |
+|--------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------------|
+| Iceberg REST server table operation  | `IcebergCreateTablePreEvent`, 
`IcebergUpdateTablePreEvent`, `IcebergDropTablePreEvent`, 
`IcebergLoadTablePreEvent`, `IcebergListTablePreEvent`, 
`IcebergTableExistsPreEvent`, `IcebergRenameTablePreEvent`                      
                                                                                
           | 0.7.0-incubating |
+| Gravitino server table operation     | `CreateTablePreEvent`, 
`AlterTablePreEvent`, `DropTablePreEvent`, `PurgeTablePreEvent`, 
`LoadTablePreEvent`, `ListTablePreEvent`                                        
                                                                                
                                                                   | 
0.8.0-incubating |
+| Gravitino server schema operation    | `CreateSchemaPreEvent`, 
`AlterSchemaPreEvent`, `DropSchemaPreEvent`, `LoadSchemaPreEvent`, 
`ListSchemaPreEvent`                                                            
                                                                                
                                                                | 
0.8.0-incubating |
+| Gravitino server catalog operation   | `CreateCatalogPreEvent`, 
`AlterCatalogPreEvent`, `DropCatalogPreEvent`, `LoadCatalogPreEvent`, 
`ListCatalogPreEvent`                                                           
                                                                                
                                                            | 0.8.0-incubating |
+| Gravitino server metalake operation  | `CreateMetalakePreEvent`, 
`AlterMetalakePreEvent`,`DropMetalakePreEvent`,`LoadMetalakePreEvent`,`ListMetalakePreEvent`
                                                                                
                                                                                
                                     | 0.8.0-incubating |
+| Gravitino server partition operation | `AddPartitionPreEvent`, 
`DropPartitionPreEvent`, `GetPartitionPreEvent`, 
`PurgePartitionPreEvent`,`ListPartitionPreEvent`,`ListPartitionNamesPreEvent`   
                                                                                
                                                                                
  | 0.8.0-incubating |
+| Gravitino server fileset operation   | `CreateFilesetPreEvent`, 
`AlterFilesetPreEvent`, `DropFilesetPreEvent`, 
`LoadFilesetPreEvent`,`ListFilesetPreEvent`,`GetFileLocationPreEvent`           
                                                                                
                                                                                
   | 0.8.0-incubating |
+| Gravitino server model operation     | `DeleteModelPreEvent`, 
`DeleteModelVersionPreEvent`, 
`RegisterAndLinkModelPreEvent`,`GetModelPreEvent`, 
`GetModelVersionPreEvent`,`LinkModelVersionPreEvent`,`ListModelPreEvent`,`RegisterModelPreEvent`
                                                                                
                                                                | 
0.9.0-incubating |
 | Gravitino server tag operation       | `ListTagsPreEvent`, 
`ListTagsInfoPreEvent`, `CreateTagPreEvent`, `GetTagPreEvent`, 
`AlterTagPreEvent`, `DeleteTagPreEvent`, `ListMetadataObjectsForTagPreEvent`, 
`ListTagsForMetadataObjectPreEvent`, `ListTagsInfoForMetadataObjectPreEvent`, 
`AssociateTagsForMetadataObjectPreEvent`, `GetTagForMetadataObjectPreEvent` | 
0.9.0-incubating |
 
 #### Event listener plugin

Reply via email to