Fix default values in dynamic forms

This commit is contained in:
Gregory Schier
2025-01-17 15:51:00 -08:00
parent 7a6ab60d30
commit dcd1be3fec
4 changed files with 44 additions and 29 deletions

View File

@@ -37,16 +37,17 @@ use yaak_plugins::manager::PluginManager;
pub async fn send_http_request<R: Runtime>( pub async fn send_http_request<R: Runtime>(
window: &WebviewWindow<R>, window: &WebviewWindow<R>,
request: &HttpRequest, unrendered_request: &HttpRequest,
og_response: &HttpResponse, og_response: &HttpResponse,
environment: Option<Environment>, environment: Option<Environment>,
cookie_jar: Option<CookieJar>, cookie_jar: Option<CookieJar>,
cancelled_rx: &mut Receiver<bool>, cancelled_rx: &mut Receiver<bool>,
) -> Result<HttpResponse, String> { ) -> Result<HttpResponse, String> {
let plugin_manager = window.state::<PluginManager>(); let plugin_manager = window.state::<PluginManager>();
let workspace = let workspace = get_workspace(window, &unrendered_request.workspace_id)
get_workspace(window, &request.workspace_id).await.expect("Failed to get Workspace"); .await
let base_environment = get_base_environment(window, &request.workspace_id) .expect("Failed to get Workspace");
let base_environment = get_base_environment(window, &unrendered_request.workspace_id)
.await .await
.expect("Failed to get base environment"); .expect("Failed to get base environment");
let settings = get_or_create_settings(window).await; let settings = get_or_create_settings(window).await;
@@ -59,10 +60,11 @@ pub async fn send_http_request<R: Runtime>(
let response_id = og_response.id.clone(); let response_id = og_response.id.clone();
let response = Arc::new(Mutex::new(og_response.clone())); let response = Arc::new(Mutex::new(og_response.clone()));
let rendered_request = let request =
render_http_request(&request, &base_environment, environment.as_ref(), &cb).await; render_http_request(&unrendered_request, &base_environment, environment.as_ref(), &cb)
.await;
let mut url_string = rendered_request.url; let mut url_string = request.url;
url_string = ensure_proto(&url_string); url_string = ensure_proto(&url_string);
if !url_string.starts_with("http://") && !url_string.starts_with("https://") { if !url_string.starts_with("http://") && !url_string.starts_with("https://") {
@@ -155,7 +157,7 @@ pub async fn send_http_request<R: Runtime>(
// Render query parameters // Render query parameters
let mut query_params = Vec::new(); let mut query_params = Vec::new();
for p in rendered_request.url_parameters { for p in request.url_parameters.clone() {
if !p.enabled || p.name.is_empty() { if !p.enabled || p.name.is_empty() {
continue; continue;
} }
@@ -186,7 +188,7 @@ pub async fn send_http_request<R: Runtime>(
} }
}; };
let m = Method::from_bytes(rendered_request.method.to_uppercase().as_bytes()) let m = Method::from_bytes(request.method.to_uppercase().as_bytes())
.expect("Failed to create method"); .expect("Failed to create method");
let mut request_builder = client.request(m, url).query(&query_params); let mut request_builder = client.request(m, url).query(&query_params);
@@ -209,7 +211,7 @@ pub async fn send_http_request<R: Runtime>(
// ); // );
// } // }
for h in rendered_request.headers { for h in request.headers.clone() {
if h.name.is_empty() && h.value.is_empty() { if h.name.is_empty() && h.value.is_empty() {
continue; continue;
} }
@@ -236,8 +238,8 @@ pub async fn send_http_request<R: Runtime>(
headers.insert(header_name, header_value); headers.insert(header_name, header_value);
} }
let request_body = rendered_request.body; let request_body = request.body.clone();
if let Some(body_type) = &rendered_request.body_type { if let Some(body_type) = &request.body_type {
if body_type == "graphql" { if body_type == "graphql" {
let query = get_str_h(&request_body, "query"); let query = get_str_h(&request_body, "query");
let variables = get_str_h(&request_body, "variables"); let variables = get_str_h(&request_body, "variables");

View File

@@ -203,14 +203,15 @@ async fn cmd_grpc_go<R: Runtime>(
Some(id) => Some(get_environment(&window, id).await.map_err(|e| e.to_string())?), Some(id) => Some(get_environment(&window, id).await.map_err(|e| e.to_string())?),
None => None, None => None,
}; };
let req = get_grpc_request(&window, request_id) let unrendered_request = get_grpc_request(&window, request_id)
.await .await
.map_err(|e| e.to_string())? .map_err(|e| e.to_string())?
.ok_or("Failed to find GRPC request")?; .ok_or("Failed to find GRPC request")?;
let base_environment = let base_environment = get_base_environment(&window, &unrendered_request.workspace_id)
get_base_environment(&window, &req.workspace_id).await.map_err(|e| e.to_string())?; .await
let req = render_grpc_request( .map_err(|e| e.to_string())?;
&req, let request = render_grpc_request(
&unrendered_request,
&base_environment, &base_environment,
environment.as_ref(), environment.as_ref(),
&PluginTemplateCallback::new( &PluginTemplateCallback::new(
@@ -223,7 +224,7 @@ async fn cmd_grpc_go<R: Runtime>(
let mut metadata = BTreeMap::new(); let mut metadata = BTreeMap::new();
// Add the rest of metadata // Add the rest of metadata
for h in req.clone().metadata { for h in request.clone().metadata {
if h.name.is_empty() && h.value.is_empty() { if h.name.is_empty() && h.value.is_empty() {
continue; continue;
} }
@@ -235,12 +236,12 @@ async fn cmd_grpc_go<R: Runtime>(
metadata.insert(h.name, h.value); metadata.insert(h.name, h.value);
} }
if let Some(auth_name) = req.authentication_type.clone() { if let Some(auth_name) = request.authentication_type.clone() {
let auth = req.authentication.clone(); let auth = request.authentication.clone();
let plugin_req = CallHttpAuthenticationRequest { let plugin_req = CallHttpAuthenticationRequest {
config: serde_json::to_value(&auth).unwrap().as_object().unwrap().to_owned(), config: serde_json::to_value(&auth).unwrap().as_object().unwrap().to_owned(),
method: "POST".to_string(), method: "POST".to_string(),
url: req.url.clone(), url: request.url.clone(),
headers: metadata headers: metadata
.iter() .iter()
.map(|(name, value)| HttpHeader { .map(|(name, value)| HttpHeader {
@@ -259,7 +260,7 @@ async fn cmd_grpc_go<R: Runtime>(
} }
let conn = { let conn = {
let req = req.clone(); let req = request.clone();
upsert_grpc_connection( upsert_grpc_connection(
&window, &window,
&GrpcConnection { &GrpcConnection {
@@ -280,8 +281,8 @@ async fn cmd_grpc_go<R: Runtime>(
let conn_id = conn.id.clone(); let conn_id = conn.id.clone();
let base_msg = GrpcEvent { let base_msg = GrpcEvent {
workspace_id: req.clone().workspace_id, workspace_id: request.clone().workspace_id,
request_id: req.clone().id, request_id: request.clone().id,
connection_id: conn.clone().id, connection_id: conn.clone().id,
..Default::default() ..Default::default()
}; };
@@ -290,12 +291,12 @@ async fn cmd_grpc_go<R: Runtime>(
let maybe_in_msg_tx = std::sync::Mutex::new(Some(in_msg_tx.clone())); let maybe_in_msg_tx = std::sync::Mutex::new(Some(in_msg_tx.clone()));
let (cancelled_tx, mut cancelled_rx) = tokio::sync::watch::channel(false); let (cancelled_tx, mut cancelled_rx) = tokio::sync::watch::channel(false);
let uri = safe_uri(&req.url); let uri = safe_uri(&request.url);
let in_msg_stream = tokio_stream::wrappers::ReceiverStream::new(in_msg_rx); let in_msg_stream = tokio_stream::wrappers::ReceiverStream::new(in_msg_rx);
let (service, method) = { let (service, method) = {
let req = req.clone(); let req = request.clone();
match (req.service, req.method) { match (req.service, req.method) {
(Some(service), Some(method)) => (service, method), (Some(service), Some(method)) => (service, method),
_ => return Err("Service and method are required".to_string()), _ => return Err("Service and method are required".to_string()),
@@ -307,7 +308,7 @@ async fn cmd_grpc_go<R: Runtime>(
.lock() .lock()
.await .await
.connect( .connect(
&req.clone().id, &request.clone().id,
uri.as_str(), uri.as_str(),
&proto_files.iter().map(|p| PathBuf::from_str(p).unwrap()).collect(), &proto_files.iter().map(|p| PathBuf::from_str(p).unwrap()).collect(),
) )
@@ -438,7 +439,7 @@ async fn cmd_grpc_go<R: Runtime>(
let grpc_listen = { let grpc_listen = {
let window = window.clone(); let window = window.clone();
let base_event = base_msg.clone(); let base_event = base_msg.clone();
let req = req.clone(); let req = request.clone();
let msg = if req.message.is_empty() { "{}".to_string() } else { req.message }; let msg = if req.message.is_empty() { "{}".to_string() } else { req.message };
let msg = render_template( let msg = render_template(
msg.as_str(), msg.as_str(),

View File

@@ -31,17 +31,19 @@ export function DynamicForm<T extends Record<string, string | boolean>>({
data, data,
onChange, onChange,
useTemplating, useTemplating,
autocompleteVariables,
stateKey, stateKey,
}: { }: {
config: FormInput[]; config: FormInput[];
onChange: (value: T) => void; onChange: (value: T) => void;
data: T; data: T;
useTemplating?: boolean; useTemplating?: boolean;
autocompleteVariables?: boolean;
stateKey: string; stateKey: string;
}) { }) {
const setDataAttr = useCallback( const setDataAttr = useCallback(
(name: string, value: string | boolean | null) => { (name: string, value: string | boolean | null) => {
onChange({ ...data, [name]: value == null ? '__NULL__' : value }); onChange({ ...data, [name]: value == DYNAMIC_FORM_NULL_ARG ? undefined : value });
}, },
[data, onChange], [data, onChange],
); );
@@ -66,6 +68,7 @@ export function DynamicForm<T extends Record<string, string | boolean>>({
stateKey={stateKey} stateKey={stateKey}
arg={a} arg={a}
useTemplating={useTemplating || false} useTemplating={useTemplating || false}
autocompleteVariables={autocompleteVariables || false}
onChange={(v) => setDataAttr(a.name, v)} onChange={(v) => setDataAttr(a.name, v)}
value={data[a.name] ? String(data[a.name]) : DYNAMIC_FORM_NULL_ARG} value={data[a.name] ? String(data[a.name]) : DYNAMIC_FORM_NULL_ARG}
/> />
@@ -77,6 +80,7 @@ export function DynamicForm<T extends Record<string, string | boolean>>({
stateKey={stateKey} stateKey={stateKey}
arg={a} arg={a}
useTemplating={useTemplating || false} useTemplating={useTemplating || false}
autocompleteVariables={autocompleteVariables || false}
onChange={(v) => setDataAttr(a.name, v)} onChange={(v) => setDataAttr(a.name, v)}
value={data[a.name] ? String(data[a.name]) : DYNAMIC_FORM_NULL_ARG} value={data[a.name] ? String(data[a.name]) : DYNAMIC_FORM_NULL_ARG}
/> />
@@ -119,12 +123,14 @@ function TextArg({
onChange, onChange,
value, value,
useTemplating, useTemplating,
autocompleteVariables,
stateKey, stateKey,
}: { }: {
arg: FormInputText; arg: FormInputText;
value: string; value: string;
onChange: (v: string) => void; onChange: (v: string) => void;
useTemplating: boolean; useTemplating: boolean;
autocompleteVariables: boolean;
stateKey: string; stateKey: string;
}) { }) {
const handleChange = useCallback( const handleChange = useCallback(
@@ -145,6 +151,7 @@ function TextArg({
hideLabel={arg.label == null} hideLabel={arg.label == null}
placeholder={arg.placeholder ?? arg.defaultValue ?? ''} placeholder={arg.placeholder ?? arg.defaultValue ?? ''}
useTemplating={useTemplating} useTemplating={useTemplating}
autocompleteVariables={autocompleteVariables}
stateKey={stateKey} stateKey={stateKey}
forceUpdateKey={stateKey} forceUpdateKey={stateKey}
/> />
@@ -156,12 +163,14 @@ function EditorArg({
onChange, onChange,
value, value,
useTemplating, useTemplating,
autocompleteVariables,
stateKey, stateKey,
}: { }: {
arg: FormInputEditor; arg: FormInputEditor;
value: string; value: string;
onChange: (v: string) => void; onChange: (v: string) => void;
useTemplating: boolean; useTemplating: boolean;
autocompleteVariables: boolean;
stateKey: string; stateKey: string;
}) { }) {
const handleChange = useCallback( const handleChange = useCallback(
@@ -196,6 +205,7 @@ function EditorArg({
defaultValue={value === DYNAMIC_FORM_NULL_ARG ? arg.defaultValue : value} defaultValue={value === DYNAMIC_FORM_NULL_ARG ? arg.defaultValue : value}
placeholder={arg.placeholder ?? arg.defaultValue ?? ''} placeholder={arg.placeholder ?? arg.defaultValue ?? ''}
useTemplating={useTemplating} useTemplating={useTemplating}
autocompleteVariables={autocompleteVariables}
stateKey={stateKey} stateKey={stateKey}
forceUpdateKey={stateKey} forceUpdateKey={stateKey}
hideGutter hideGutter

View File

@@ -39,6 +39,8 @@ export function HttpAuthenticationEditor({ request }: Props) {
return ( return (
<DynamicForm <DynamicForm
autocompleteVariables
useTemplating
stateKey={`auth.${request.id}.${request.authenticationType}`} stateKey={`auth.${request.id}.${request.authenticationType}`}
config={auth.config} config={auth.config}
data={request.authentication} data={request.authentication}