From 82aed6fef390ac56cd4659ce17b26f0867ceee6e Mon Sep 17 00:00:00 2001 From: Daniel Gaspar Date: Mon, 14 Nov 2022 11:25:20 +0000 Subject: [PATCH 1/2] refactor: add deprecated decorator for old SupersetView's --- superset/views/base.py | 24 +++++++++++++ superset/views/core.py | 79 +++++++----------------------------------- 2 files changed, 37 insertions(+), 66 deletions(-) diff --git a/superset/views/base.py b/superset/views/base.py index b0a0b5a749474..85b6ec52b46ec 100644 --- a/superset/views/base.py +++ b/superset/views/base.py @@ -184,6 +184,30 @@ def generate_download_headers( return headers +def deprecated( + eol_version: str = "3.0.0", +) -> Callable[[Callable[..., FlaskResponse]], Callable[..., FlaskResponse]]: + """ + A decorator to set an API endpoint from SupersetView has deprecated. + Issues a log warning + """ + + def _deprecated(f: Callable[..., FlaskResponse]) -> Callable[..., FlaskResponse]: + def wraps(self: "BaseSupersetView", *args: Any, **kwargs: Any) -> FlaskResponse: + logger.warning( + "%s.%s " + "This API endpoint is deprecated and will be removed in version %s", + self.__class__.__name__, + f.__name__, + eol_version, + ) + return f(self, *args, **kwargs) + + return functools.update_wrapper(wraps, f) + + return _deprecated + + def api(f: Callable[..., FlaskResponse]) -> Callable[..., FlaskResponse]: """ A decorator to label an endpoint as an API. Catches uncaught exceptions and diff --git a/superset/views/core.py b/superset/views/core.py index f88fc7573f93a..f44d66775d089 100755 --- a/superset/views/core.py +++ b/superset/views/core.py @@ -145,6 +145,7 @@ create_table_permissions, CsvResponse, data_payload_response, + deprecated, generate_download_headers, get_error_msg, handle_api_exception, @@ -278,13 +279,8 @@ def override_role_permissions(self) -> FlaskResponse: @has_access @event_logger.log_this @expose("/request_access/", methods=["POST"]) + @deprecated() def request_access(self) -> FlaskResponse: - logger.warning( - "%s.approve " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) - datasources = set() dashboard_id = request.args.get("dashboard_id") if dashboard_id: @@ -327,6 +323,7 @@ def request_access(self) -> FlaskResponse: @has_access @event_logger.log_this @expose("/approve", methods=["POST"]) + @deprecated() def approve(self) -> FlaskResponse: # pylint: disable=too-many-locals def clean_fulfilled_requests(session: Session) -> None: for dar in session.query(DAR).all(): @@ -338,12 +335,6 @@ def clean_fulfilled_requests(session: Session) -> None: session.delete(dar) session.commit() - logger.warning( - "%s.approve " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) - datasource_type = request.args["datasource_type"] datasource_id = request.args["datasource_id"] created_by_username = request.args.get("created_by") @@ -787,6 +778,7 @@ def get_redirect_url() -> str: @event_logger.log_this @expose("/explore///", methods=["GET", "POST"]) @expose("/explore/", methods=["GET", "POST"]) + @deprecated() # pylint: disable=too-many-locals,too-many-branches,too-many-statements def explore( self, @@ -794,11 +786,6 @@ def explore( datasource_id: Optional[int] = None, key: Optional[str] = None, ) -> FlaskResponse: - logger.warning( - "%s.explore " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) if request.method == "GET": return redirect(Superset.get_redirect_url()) @@ -1359,13 +1346,9 @@ def add_slices( # pylint: disable=no-self-use @has_access_api @event_logger.log_this @expose("/testconn", methods=["POST", "GET"]) + @deprecated() def testconn(self) -> FlaskResponse: """Tests a sqla connection""" - logger.warning( - "%s.testconn " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) db_name = request.json.get("name") uri = request.json.get("uri") try: @@ -1562,17 +1545,13 @@ def recent_activity( # pylint: disable=too-many-locals @has_access_api @event_logger.log_this @expose("/available_domains/", methods=["GET"]) + @deprecated() def available_domains(self) -> FlaskResponse: """ Returns the list of available Superset Webserver domains (if any) defined in config. This enables charts embedded in other apps to leverage domain sharding if appropriately configured. """ - logger.warning( - "%s.available_domains " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) return Response( json.dumps(conf.get("SUPERSET_WEBSERVER_DOMAINS")), mimetype="text/json" ) @@ -1581,13 +1560,9 @@ def available_domains(self) -> FlaskResponse: @has_access_api @event_logger.log_this @expose("/fave_dashboards_by_username//", methods=["GET"]) + @deprecated() def fave_dashboards_by_username(self, username: str) -> FlaskResponse: """This lets us use a user's username to pull favourite dashboards""" - logger.warning( - "%s.fave_dashboards_by_username " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) user = security_manager.find_user(username=username) return self.fave_dashboards(user.id) @@ -1595,12 +1570,8 @@ def fave_dashboards_by_username(self, username: str) -> FlaskResponse: @has_access_api @event_logger.log_this @expose("/fave_dashboards//", methods=["GET"]) + @deprecated() def fave_dashboards(self, user_id: int) -> FlaskResponse: - logger.warning( - "%s.fave_dashboards " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) error_obj = self.get_user_activity_access_error(user_id) if error_obj: return error_obj @@ -1636,13 +1607,8 @@ def fave_dashboards(self, user_id: int) -> FlaskResponse: @has_access_api @event_logger.log_this @expose("/created_dashboards//", methods=["GET"]) + @deprecated() def created_dashboards(self, user_id: int) -> FlaskResponse: - logger.warning( - "%s.created_dashboards " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) - error_obj = self.get_user_activity_access_error(user_id) if error_obj: return error_obj @@ -1726,13 +1692,9 @@ def user_slices(self, user_id: Optional[int] = None) -> FlaskResponse: @event_logger.log_this @expose("/created_slices", methods=["GET"]) @expose("/created_slices//", methods=["GET"]) + @deprecated() def created_slices(self, user_id: Optional[int] = None) -> FlaskResponse: """List of slices created by this user""" - logger.warning( - "%s.created_slices " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) if not user_id: user_id = cast(int, get_user_id()) error_obj = self.get_user_activity_access_error(user_id) @@ -2154,15 +2116,10 @@ def sqllab_viz(self) -> FlaskResponse: # pylint: disable=no-self-use @has_access @expose("/extra_table_metadata////") @event_logger.log_this + @deprecated() def extra_table_metadata( self, database_id: int, table_name: str, schema: str ) -> FlaskResponse: - logger.warning( - "%s.extra_table_metadata " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) - parsed_schema = utils.parse_js_uri_path_item(schema, eval_undefined=True) table_name = utils.parse_js_uri_path_item(table_name) # type: ignore mydb = db.session.query(Database).filter_by(id=database_id).one() @@ -2363,6 +2320,7 @@ def stop_query(self) -> FlaskResponse: @has_access_api @event_logger.log_this @expose("/validate_sql_json/", methods=["POST", "GET"]) + @deprecated() def validate_sql_json( # pylint: disable=too-many-locals self, @@ -2370,12 +2328,6 @@ def validate_sql_json( """Validates that arbitrary sql is acceptable for the given database. Returns a list of error/warning annotations as json. """ - logger.warning( - "%s.validate_sql_json " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) - sql = request.form["sql"] database_id = request.form["database_id"] schema = request.form.get("schema") or None @@ -2591,6 +2543,7 @@ def csv(self, client_id: str) -> FlaskResponse: # pylint: disable=no-self-use @has_access @event_logger.log_this @expose("/fetch_datasource_metadata") + @deprecated() def fetch_datasource_metadata(self) -> FlaskResponse: # pylint: disable=no-self-use """ Fetch the datasource metadata. @@ -2598,12 +2551,6 @@ def fetch_datasource_metadata(self) -> FlaskResponse: # pylint: disable=no-self :returns: The Flask response :raises SupersetSecurityException: If the user cannot access the resource """ - logger.warning( - "%s.fetch_datasource_metadata " - "This API endpoint is deprecated and will be removed in version 3.0.0", - self.__class__.__name__, - ) - datasource_id, datasource_type = request.args["datasourceKey"].split("__") datasource = DatasourceDAO.get_datasource( db.session, DatasourceType(datasource_type), int(datasource_id) From 0e1f03aeda576215d85995089dea0c7c0761d629 Mon Sep 17 00:00:00 2001 From: Daniel Gaspar Date: Mon, 14 Nov 2022 11:49:41 +0000 Subject: [PATCH 2/2] fix lint --- superset/views/core.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/superset/views/core.py b/superset/views/core.py index f44d66775d089..2cafccbf9c5f7 100755 --- a/superset/views/core.py +++ b/superset/views/core.py @@ -324,7 +324,7 @@ def request_access(self) -> FlaskResponse: @event_logger.log_this @expose("/approve", methods=["POST"]) @deprecated() - def approve(self) -> FlaskResponse: # pylint: disable=too-many-locals + def approve(self) -> FlaskResponse: # pylint: disable=too-many-locals,no-self-use def clean_fulfilled_requests(session: Session) -> None: for dar in session.query(DAR).all(): datasource = DatasourceDAO.get_datasource( @@ -1347,7 +1347,7 @@ def add_slices( # pylint: disable=no-self-use @event_logger.log_this @expose("/testconn", methods=["POST", "GET"]) @deprecated() - def testconn(self) -> FlaskResponse: + def testconn(self) -> FlaskResponse: # pylint: disable=no-self-use """Tests a sqla connection""" db_name = request.json.get("name") uri = request.json.get("uri") @@ -1546,7 +1546,7 @@ def recent_activity( # pylint: disable=too-many-locals @event_logger.log_this @expose("/available_domains/", methods=["GET"]) @deprecated() - def available_domains(self) -> FlaskResponse: + def available_domains(self) -> FlaskResponse: # pylint: disable=no-self-use """ Returns the list of available Superset Webserver domains (if any) defined in config. This enables charts embedded in other apps to @@ -2117,7 +2117,7 @@ def sqllab_viz(self) -> FlaskResponse: # pylint: disable=no-self-use @expose("/extra_table_metadata////") @event_logger.log_this @deprecated() - def extra_table_metadata( + def extra_table_metadata( # pylint: disable=no-self-use self, database_id: int, table_name: str, schema: str ) -> FlaskResponse: parsed_schema = utils.parse_js_uri_path_item(schema, eval_undefined=True) @@ -2322,7 +2322,7 @@ def stop_query(self) -> FlaskResponse: @expose("/validate_sql_json/", methods=["POST", "GET"]) @deprecated() def validate_sql_json( - # pylint: disable=too-many-locals + # pylint: disable=too-many-locals,no-self-use self, ) -> FlaskResponse: """Validates that arbitrary sql is acceptable for the given database.