-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.ts
112 lines (93 loc) · 2.6 KB
/
index.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
import { useForm } from "@inertiajs/vue3";
interface FormProps {
action?: string;
[key: string]: any;
}
interface SessionResponse {
csrfTokenName: string;
csrfTokenValue: string;
}
interface CsrfToken {
[key: string]: string;
}
const CSRF_ENDPOINT = "/actions/users/session-info";
/**
* Gets CSRF meta element from document head
*/
const getCsrfMetaEl = () =>
document.head.querySelector<HTMLMetaElement>("meta[csrf]");
/**
* Extracts CSRF token from meta element
*/
const getCsrfToken = (csrfMeta?: HTMLMetaElement | null): CsrfToken | null => {
const meta = csrfMeta ?? getCsrfMetaEl();
const tokenName = meta?.getAttribute("name");
const tokenValue = meta?.getAttribute("content");
return tokenName && tokenValue ? { [tokenName]: tokenValue } : null;
};
/**
* Fetches new CSRF token from server
* @throws Error if fetch fails or response is invalid
*/
const fetchNewCsrfToken = async (): Promise<SessionResponse> => {
try {
const response = await fetch(CSRF_ENDPOINT, {
headers: { Accept: "application/json" },
});
if (!response.ok) {
throw new Error(`HTTP error! status: ${response.status}`);
}
return await response.json();
} catch (error) {
throw new Error(
`Failed to fetch CSRF token: ${
error instanceof Error ? error.message : "Unknown error"
}`
);
}
};
const cUseForm = (props: FormProps) => {
let formObject = {
action: "",
...props,
};
const csrfMeta: HTMLMetaElement | null = getCsrfMetaEl();
if (csrfMeta !== null) {
formObject = {
...formObject,
...getCsrfToken(csrfMeta),
};
}
// Initialize form with action property
const form = useForm(formObject);
const originalPost = form.post.bind(form);
form.post = (url, options = {}) => {
// Set action to the URL (entries/save-entry)
form.action = url;
if (form.action.includes("login")) {
// Store user's onSuccess handler if it exists
const userOnSuccess = options.onSuccess;
options.onSuccess = (page) => {
if (userOnSuccess) {
// Call user's handler first
userOnSuccess(page);
}
const metaEl = getCsrfMetaEl();
fetchNewCsrfToken().then((newToken) => {
if (metaEl) {
metaEl.setAttribute("name", newToken.csrfTokenName);
metaEl.setAttribute("content", newToken.csrfTokenValue);
}
});
};
}
const mergedOptions = {
forceFormData: true,
...options,
};
// Call original post with empty URL
return originalPost("", mergedOptions);
};
return form;
};
export default cUseForm;