169 lines
5.6 KiB
Python
169 lines
5.6 KiB
Python
# Copyright (c) Streamlit Inc. (2018-2022) Snowflake Inc. (2022-2025)
|
|
#
|
|
# Licensed 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.
|
|
|
|
from __future__ import annotations
|
|
|
|
from typing import Any
|
|
from urllib import parse
|
|
|
|
from streamlit.errors import StreamlitAPIException
|
|
from streamlit.proto.ForwardMsg_pb2 import ForwardMsg
|
|
from streamlit.runtime.metrics_util import gather_metrics
|
|
from streamlit.runtime.scriptrunner_utils.script_run_context import get_script_run_ctx
|
|
from streamlit.runtime.state.query_params import (
|
|
EMBED_OPTIONS_QUERY_PARAM,
|
|
EMBED_QUERY_PARAM,
|
|
EMBED_QUERY_PARAMS_KEYS,
|
|
)
|
|
|
|
|
|
@gather_metrics("experimental_get_query_params")
|
|
def get_query_params() -> dict[str, list[str]]:
|
|
"""Return the query parameters that is currently showing in the browser's URL bar.
|
|
|
|
Returns
|
|
-------
|
|
dict
|
|
The current query parameters as a dict. "Query parameters" are the part of the URL that comes
|
|
after the first "?".
|
|
|
|
Example
|
|
-------
|
|
Let's say the user's web browser is at
|
|
`http://localhost:8501/?show_map=True&selected=asia&selected=america`.
|
|
Then, you can get the query parameters using the following:
|
|
|
|
>>> import streamlit as st
|
|
>>>
|
|
>>> st.experimental_get_query_params()
|
|
{"show_map": ["True"], "selected": ["asia", "america"]}
|
|
|
|
Note that the values in the returned dict are *always* lists. This is
|
|
because we internally use Python's urllib.parse.parse_qs(), which behaves
|
|
this way. And this behavior makes sense when you consider that every item
|
|
in a query string is potentially a 1-element array.
|
|
|
|
"""
|
|
ctx = get_script_run_ctx()
|
|
if ctx is None:
|
|
return {}
|
|
ctx.mark_experimental_query_params_used()
|
|
# Return new query params dict, but without embed, embed_options query params
|
|
return _exclude_keys_in_dict(
|
|
parse.parse_qs(ctx.query_string, keep_blank_values=True),
|
|
keys_to_exclude=EMBED_QUERY_PARAMS_KEYS,
|
|
)
|
|
|
|
|
|
@gather_metrics("experimental_set_query_params")
|
|
def set_query_params(**query_params: Any) -> None:
|
|
"""Set the query parameters that are shown in the browser's URL bar.
|
|
|
|
.. warning::
|
|
Query param `embed` cannot be set using this method.
|
|
|
|
Parameters
|
|
----------
|
|
**query_params : dict
|
|
The query parameters to set, as key-value pairs.
|
|
|
|
Example
|
|
-------
|
|
|
|
To point the user's web browser to something like
|
|
"http://localhost:8501/?show_map=True&selected=asia&selected=america",
|
|
you would do the following:
|
|
|
|
>>> import streamlit as st
|
|
>>>
|
|
>>> st.experimental_set_query_params(
|
|
... show_map=True,
|
|
... selected=["asia", "america"],
|
|
... )
|
|
|
|
"""
|
|
ctx = get_script_run_ctx()
|
|
if ctx is None:
|
|
return
|
|
ctx.mark_experimental_query_params_used()
|
|
|
|
msg = ForwardMsg()
|
|
msg.page_info_changed.query_string = _ensure_no_embed_params(
|
|
query_params, ctx.query_string
|
|
)
|
|
ctx.query_string = msg.page_info_changed.query_string
|
|
ctx.enqueue(msg)
|
|
|
|
|
|
def _exclude_keys_in_dict(
|
|
d: dict[str, Any], keys_to_exclude: list[str]
|
|
) -> dict[str, Any]:
|
|
"""Returns new object but without keys defined in keys_to_exclude."""
|
|
return {
|
|
key: value for key, value in d.items() if key.lower() not in keys_to_exclude
|
|
}
|
|
|
|
|
|
def _extract_key_query_params(
|
|
query_params: dict[str, list[str]], param_key: str
|
|
) -> set[str]:
|
|
"""Extracts key (case-insensitive) query params from Dict, and returns them as Set of str."""
|
|
return {
|
|
item.lower()
|
|
for sublist in [
|
|
[value.lower() for value in query_params[key]]
|
|
for key in query_params
|
|
if key.lower() == param_key and query_params.get(key)
|
|
]
|
|
for item in sublist
|
|
}
|
|
|
|
|
|
def _ensure_no_embed_params(
|
|
query_params: dict[str, list[str] | str], query_string: str
|
|
) -> str:
|
|
"""Ensures there are no embed params set (raises StreamlitAPIException) if there is a try,
|
|
also makes sure old param values in query_string are preserved. Returns query_string : str.
|
|
"""
|
|
# Get query params dict without embed, embed_options params
|
|
query_params_without_embed = _exclude_keys_in_dict(
|
|
query_params, keys_to_exclude=EMBED_QUERY_PARAMS_KEYS
|
|
)
|
|
if query_params != query_params_without_embed:
|
|
raise StreamlitAPIException(
|
|
"Query param embed and embed_options (case-insensitive) cannot be set using set_query_params method."
|
|
)
|
|
|
|
all_current_params = parse.parse_qs(query_string, keep_blank_values=True)
|
|
current_embed_params = parse.urlencode(
|
|
{
|
|
EMBED_QUERY_PARAM: list(
|
|
_extract_key_query_params(
|
|
all_current_params, param_key=EMBED_QUERY_PARAM
|
|
)
|
|
),
|
|
EMBED_OPTIONS_QUERY_PARAM: list(
|
|
_extract_key_query_params(
|
|
all_current_params, param_key=EMBED_OPTIONS_QUERY_PARAM
|
|
)
|
|
),
|
|
},
|
|
doseq=True,
|
|
)
|
|
query_string = parse.urlencode(query_params, doseq=True)
|
|
|
|
if query_string:
|
|
separator = "&" if current_embed_params else ""
|
|
return separator.join([query_string, current_embed_params])
|
|
return current_embed_params
|