mirror of
https://github.com/alexmickelson/canvasManagement.git
synced 2026-03-25 23:28:33 -06:00
am calling canvas api
This commit is contained in:
@@ -1,6 +1,13 @@
|
|||||||
/** @type {import('next').NextConfig} */
|
/** @type {import('next').NextConfig} */
|
||||||
const nextConfig = {
|
const nextConfig = {
|
||||||
|
async rewrites() {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
source: '/api/canvas/:rest*',
|
||||||
|
destination: 'https://snow.instructure.com/api/v1/:rest*',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
export default nextConfig;
|
export default nextConfig;
|
||||||
|
|||||||
@@ -21,7 +21,6 @@ export default function EditAssignment({
|
|||||||
const [assignmentText, setAssignmentText] = useState(
|
const [assignmentText, setAssignmentText] = useState(
|
||||||
localAssignmentMarkdown.toMarkdown(assignment)
|
localAssignmentMarkdown.toMarkdown(assignment)
|
||||||
);
|
);
|
||||||
console.log(assignmentText);
|
|
||||||
const [error, setError] = useState("");
|
const [error, setError] = useState("");
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -69,7 +68,7 @@ export default function EditAssignment({
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="p-5">
|
<div className="p-5">
|
||||||
<button className="bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded">
|
<button>
|
||||||
Add to canvas....
|
Add to canvas....
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,10 +1,15 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { MonacoEditor } from "@/components/editor/MonacoEditor";
|
import { MonacoEditor } from "@/components/editor/MonacoEditor";
|
||||||
import { usePageQuery } from "@/hooks/localCourse/pageHooks";
|
import {
|
||||||
|
usePageQuery,
|
||||||
|
useUpdatePageMutation,
|
||||||
|
} from "@/hooks/localCourse/pageHooks";
|
||||||
import { localPageMarkdownUtils } from "@/models/local/page/localCoursePage";
|
import { localPageMarkdownUtils } from "@/models/local/page/localCoursePage";
|
||||||
import { useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import PagePreview from "./PagePreview";
|
import PagePreview from "./PagePreview";
|
||||||
|
import { useLocalCourseSettingsQuery } from "@/hooks/localCourse/localCoursesHooks";
|
||||||
|
import { useCanvasPagesQuery } from "@/hooks/canvas/canvasPageHooks";
|
||||||
|
|
||||||
export default function EditPage({
|
export default function EditPage({
|
||||||
moduleName,
|
moduleName,
|
||||||
@@ -14,11 +19,43 @@ export default function EditPage({
|
|||||||
moduleName: string;
|
moduleName: string;
|
||||||
}) {
|
}) {
|
||||||
const { data: page } = usePageQuery(moduleName, pageName);
|
const { data: page } = usePageQuery(moduleName, pageName);
|
||||||
|
const updatePage = useUpdatePageMutation();
|
||||||
const [pageText, setPageText] = useState(
|
const [pageText, setPageText] = useState(
|
||||||
localPageMarkdownUtils.toMarkdown(page)
|
localPageMarkdownUtils.toMarkdown(page)
|
||||||
);
|
);
|
||||||
const [error, setError] = useState("");
|
const [error, setError] = useState("");
|
||||||
|
|
||||||
|
const { data: settings } = useLocalCourseSettingsQuery();
|
||||||
|
const { data: canvasPages } = useCanvasPagesQuery(settings.canvasId ?? 0);
|
||||||
|
console.log("canvas pages", canvasPages);
|
||||||
|
const pageInCanvas = canvasPages?.find((p) => p.title === pageName);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const delay = 500;
|
||||||
|
const handler = setTimeout(() => {
|
||||||
|
const updatedPage = localPageMarkdownUtils.parseMarkdown(pageText);
|
||||||
|
if (
|
||||||
|
localPageMarkdownUtils.toMarkdown(page) !==
|
||||||
|
localPageMarkdownUtils.toMarkdown(updatedPage)
|
||||||
|
) {
|
||||||
|
console.log("updating assignment");
|
||||||
|
try {
|
||||||
|
updatePage.mutate({
|
||||||
|
page: updatedPage,
|
||||||
|
moduleName,
|
||||||
|
pageName,
|
||||||
|
});
|
||||||
|
} catch (e: any) {
|
||||||
|
setError(e.toString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, delay);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
clearTimeout(handler);
|
||||||
|
};
|
||||||
|
}, [moduleName, page, pageName, pageText, updatePage]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="h-full flex flex-col">
|
<div className="h-full flex flex-col">
|
||||||
<div className="columns-2 min-h-0 flex-1">
|
<div className="columns-2 min-h-0 flex-1">
|
||||||
@@ -27,13 +64,22 @@ export default function EditPage({
|
|||||||
</div>
|
</div>
|
||||||
<div className="h-full">
|
<div className="h-full">
|
||||||
<div className="text-red-300">{error && error}</div>
|
<div className="text-red-300">{error && error}</div>
|
||||||
<PagePreview page={page} />
|
<div className="h-full overflow-y-auto">
|
||||||
|
<br />
|
||||||
|
<PagePreview page={page} />
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="p-5">
|
<div className="p-5 flex flex-row">
|
||||||
<button className="bg-blue-500 hover:bg-blue-700 text-white font-bold py-2 px-4 rounded">
|
{pageInCanvas && (
|
||||||
Add to canvas....
|
<a
|
||||||
</button>
|
target="_blank"
|
||||||
|
href={`https://snow.instructure.com/courses/${settings.canvasId}/pages/${pageInCanvas.page_id}`}
|
||||||
|
>
|
||||||
|
View Page In Canvas
|
||||||
|
</a>
|
||||||
|
)}
|
||||||
|
<button>Add to canvas</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
@import url('https://fonts.googleapis.com/css2?family=DM+Sans:ital,opsz,wght@0,9..40,100..1000;1,9..40,100..1000&display=swap');
|
@import url("https://fonts.googleapis.com/css2?family=DM+Sans:ital,opsz,wght@0,9..40,100..1000;1,9..40,100..1000&display=swap");
|
||||||
|
|
||||||
@tailwind base;
|
@tailwind base;
|
||||||
@tailwind components;
|
@tailwind components;
|
||||||
@@ -12,36 +12,37 @@
|
|||||||
|
|
||||||
:root {
|
:root {
|
||||||
font-family: "DM Sans", sans-serif;
|
font-family: "DM Sans", sans-serif;
|
||||||
|
@apply text-lg;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* monaco editor */
|
/* monaco editor */
|
||||||
.monaco-editor-background,
|
.monaco-editor-background,
|
||||||
.monaco-editor .margin {
|
.monaco-editor .margin {
|
||||||
background-color: black !important;
|
background-color: #18181b !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
h1 {
|
h1 {
|
||||||
@apply text-4xl font-bold;
|
@apply text-4xl font-bold my-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
h2 {
|
h2 {
|
||||||
@apply text-3xl font-semibold;
|
@apply text-3xl font-semibold my-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
h3 {
|
h3 {
|
||||||
@apply text-2xl font-semibold;
|
@apply text-2xl font-semibold my-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
h4 {
|
h4 {
|
||||||
@apply text-xl font-medium;
|
@apply text-xl font-medium my-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
h5 {
|
h5 {
|
||||||
@apply text-lg font-medium;
|
@apply text-lg font-medium my-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
h6 {
|
h6 {
|
||||||
@apply text-base font-medium;
|
@apply text-base font-medium my-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
strong {
|
strong {
|
||||||
@@ -62,5 +63,16 @@ hr {
|
|||||||
}
|
}
|
||||||
|
|
||||||
blockquote {
|
blockquote {
|
||||||
@apply border-l-4 border-gray-300 pl-4 italic text-gray-700;
|
@apply border-l-4 border-gray-300 pl-4 italic text-gray-400 m-5 pe-3;
|
||||||
|
}
|
||||||
|
|
||||||
|
code {
|
||||||
|
@apply font-mono text-sm bg-gray-800 px-1;
|
||||||
|
}
|
||||||
|
p {
|
||||||
|
@apply mb-3;
|
||||||
|
}
|
||||||
|
|
||||||
|
button {
|
||||||
|
@apply bg-blue-900 hover:bg-blue-700 text-blue-50 font-bold py-1 px-3 rounded transition-all duration-200;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,16 +26,6 @@ export default async function RootLayout({
|
|||||||
<head>
|
<head>
|
||||||
</head>
|
</head>
|
||||||
<body className="bg-slate-900 h-screen p-1 text-slate-300">
|
<body className="bg-slate-900 h-screen p-1 text-slate-300">
|
||||||
{/* <Toaster
|
|
||||||
containerClassName=""
|
|
||||||
toastOptions={{
|
|
||||||
className: "bg-slate-900",
|
|
||||||
error: { duration: Infinity },
|
|
||||||
style: {
|
|
||||||
// backgroundColor: "black",
|
|
||||||
},
|
|
||||||
}}
|
|
||||||
/> */}
|
|
||||||
<MyToaster />
|
<MyToaster />
|
||||||
<Suspense>
|
<Suspense>
|
||||||
<Providers>
|
<Providers>
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ export default function Providers({ children }: { children: ReactNode }) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<QueryClientProvider client={queryClient}>
|
<QueryClientProvider client={queryClient}>
|
||||||
<ReactQueryDevtools initialIsOpen={false} />
|
{/* <ReactQueryDevtools initialIsOpen={false} /> */}
|
||||||
{children}
|
{children}
|
||||||
</QueryClientProvider>
|
</QueryClientProvider>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -2,7 +2,8 @@ import { canvasService } from "@/services/canvas/canvasService";
|
|||||||
import { useSuspenseQuery } from "@tanstack/react-query";
|
import { useSuspenseQuery } from "@tanstack/react-query";
|
||||||
|
|
||||||
export const canvasCourseKeys = {
|
export const canvasCourseKeys = {
|
||||||
courseDetails: (canavasId: number) => ["canvas course", canavasId] as const,
|
courseDetails: (canavasId: number) =>
|
||||||
|
["canvas", canavasId, "course details"] as const,
|
||||||
};
|
};
|
||||||
|
|
||||||
export const useCanvasCourseQuery = (canvasId: number) =>
|
export const useCanvasCourseQuery = (canvasId: number) =>
|
||||||
16
nextjs/src/hooks/canvas/canvasPageHooks.ts
Normal file
16
nextjs/src/hooks/canvas/canvasPageHooks.ts
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
import { canvasPageService } from "@/services/canvas/canvasPageService";
|
||||||
|
import { useQuery } from "@tanstack/react-query";
|
||||||
|
|
||||||
|
export const canvasPageKeys = {
|
||||||
|
pagesInCourse: (canvasCourseId: number) => [
|
||||||
|
"canvas",
|
||||||
|
canvasCourseId,
|
||||||
|
"pages",
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
export const useCanvasPagesQuery = (canvasCourseId: number) =>
|
||||||
|
useQuery({
|
||||||
|
queryKey: canvasPageKeys.pagesInCourse(canvasCourseId),
|
||||||
|
queryFn: async () => await canvasPageService.getAll(canvasCourseId),
|
||||||
|
});
|
||||||
@@ -1,8 +1,26 @@
|
|||||||
|
import { isServer } from "@tanstack/react-query";
|
||||||
import axios, { AxiosInstance, AxiosError } from "axios";
|
import axios, { AxiosInstance, AxiosError } from "axios";
|
||||||
import toast from "react-hot-toast";
|
import toast from "react-hot-toast";
|
||||||
|
|
||||||
export const axiosClient: AxiosInstance = axios.create();
|
export const axiosClient: AxiosInstance = axios.create();
|
||||||
|
|
||||||
|
if (!isServer) {
|
||||||
|
console.log("not on the server, setting up interceptor");
|
||||||
|
axiosClient.interceptors.request.use((config) => {
|
||||||
|
if (
|
||||||
|
config.url &&
|
||||||
|
config.url.startsWith("https://snow.instructure.com/api/v1/")
|
||||||
|
) {
|
||||||
|
const newUrl = config.url.replace(
|
||||||
|
"https://snow.instructure.com/api/v1/",
|
||||||
|
"/api/canvas/"
|
||||||
|
);
|
||||||
|
config.url = newUrl;
|
||||||
|
}
|
||||||
|
return config;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
axiosClient.interceptors.response.use(
|
axiosClient.interceptors.response.use(
|
||||||
(response) => response,
|
(response) => response,
|
||||||
(error: AxiosError) => {
|
(error: AxiosError) => {
|
||||||
|
|||||||
65
nextjs/src/services/canvas/canvasPageService.ts
Normal file
65
nextjs/src/services/canvas/canvasPageService.ts
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
import { CanvasPage } from "@/models/canvas/pages/canvasPageModel";
|
||||||
|
import { LocalCoursePage } from "@/models/local/page/localCoursePage";
|
||||||
|
import { canvasServiceUtils } from "./canvasServiceUtils";
|
||||||
|
import { webRequestor } from "./webRequestor";
|
||||||
|
|
||||||
|
const baseCanvasUrl = "https://snow.instructure.com/api/v1";
|
||||||
|
|
||||||
|
export const canvasPageService = {
|
||||||
|
async getAll(courseId: number): Promise<CanvasPage[]> {
|
||||||
|
const url = `${baseCanvasUrl}/courses/${courseId}/pages`;
|
||||||
|
const pages = await canvasServiceUtils.paginatedRequest<CanvasPage[]>({
|
||||||
|
url,
|
||||||
|
});
|
||||||
|
return pages.flatMap((pageList) => pageList);
|
||||||
|
},
|
||||||
|
|
||||||
|
// async create(canvasCourseId: number, localCourse: LocalCoursePage): Promise<CanvasPage> {
|
||||||
|
// console.log(`Creating course page: ${localCourse.name}`);
|
||||||
|
// const url = `courses/${canvasCourseId}/pages`;
|
||||||
|
// const body = {
|
||||||
|
// wiki_page: {
|
||||||
|
// title: localCourse.name,
|
||||||
|
// body: localCourse.getBodyHtml(),
|
||||||
|
// },
|
||||||
|
// };
|
||||||
|
|
||||||
|
// const { canvasPage, response } = await webRequestor.post<CanvasPage>({
|
||||||
|
// url,
|
||||||
|
// body,
|
||||||
|
// });
|
||||||
|
|
||||||
|
// if (!canvasPage) {
|
||||||
|
// throw new Error("Created canvas course page was null");
|
||||||
|
// }
|
||||||
|
|
||||||
|
// return canvasPage;
|
||||||
|
// },
|
||||||
|
|
||||||
|
// async update(courseId: number, canvasPageId: number, localCoursePage: LocalCoursePage): Promise<void> {
|
||||||
|
// console.log(`Updating course page: ${localCoursePage.name}`);
|
||||||
|
// const url = `courses/${courseId}/pages/${canvasPageId}`;
|
||||||
|
// const body = {
|
||||||
|
// wiki_page: {
|
||||||
|
// title: localCoursePage.name,
|
||||||
|
// body: localCoursePage.getBodyHtml(),
|
||||||
|
// },
|
||||||
|
// };
|
||||||
|
|
||||||
|
// await webRequestor.put({
|
||||||
|
// url,
|
||||||
|
// body,
|
||||||
|
// });
|
||||||
|
// },
|
||||||
|
|
||||||
|
// async delete(courseId: number, canvasPageId: number): Promise<void> {
|
||||||
|
// console.log(`Deleting page from canvas ${canvasPageId}`);
|
||||||
|
// const url = `courses/${courseId}/pages/${canvasPageId}`;
|
||||||
|
// const response = await webRequestor.delete({ url });
|
||||||
|
|
||||||
|
// if (!response.isSuccessful) {
|
||||||
|
// console.error(url);
|
||||||
|
// throw new Error("Failed to delete canvas course page");
|
||||||
|
// }
|
||||||
|
// },
|
||||||
|
};
|
||||||
@@ -29,13 +29,6 @@ export const canvasService = {
|
|||||||
async getCourse(courseId: number): Promise<CanvasCourseModel> {
|
async getCourse(courseId: number): Promise<CanvasCourseModel> {
|
||||||
const url = `courses/${courseId}`;
|
const url = `courses/${courseId}`;
|
||||||
const { data, response } = await webRequestor.get<CanvasCourseModel>(url);
|
const { data, response } = await webRequestor.get<CanvasCourseModel>(url);
|
||||||
|
|
||||||
if (!data) {
|
|
||||||
console.error((await response?.text()) ?? "");
|
|
||||||
console.error(response?.url ?? "");
|
|
||||||
throw new Error("Error getting course from Canvas");
|
|
||||||
}
|
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@@ -1,11 +1,16 @@
|
|||||||
// services/canvasServiceUtils.ts
|
// services/canvasServiceUtils.ts
|
||||||
|
|
||||||
|
import { AxiosResponseHeaders, RawAxiosResponseHeaders } from "axios";
|
||||||
import { webRequestor } from "./webRequestor";
|
import { webRequestor } from "./webRequestor";
|
||||||
|
|
||||||
const BASE_URL = "https://snow.instructure.com/api/v1/";
|
const getNextUrl = (
|
||||||
|
headers: AxiosResponseHeaders | RawAxiosResponseHeaders
|
||||||
|
): string | undefined => {
|
||||||
|
const linkHeader: string | undefined =
|
||||||
|
typeof headers.get === "function"
|
||||||
|
? (headers.get("link") as string)
|
||||||
|
: ((headers as RawAxiosResponseHeaders)["link"] as string);
|
||||||
|
|
||||||
const getNextUrl = (headers: Headers): string | undefined => {
|
|
||||||
const linkHeader = headers.get("Link");
|
|
||||||
if (!linkHeader) return undefined;
|
if (!linkHeader) return undefined;
|
||||||
|
|
||||||
const links = linkHeader.split(",").map((link) => link.trim());
|
const links = linkHeader.split(",").map((link) => link.trim());
|
||||||
@@ -14,33 +19,24 @@ const getNextUrl = (headers: Headers): string | undefined => {
|
|||||||
if (!nextLink) return undefined;
|
if (!nextLink) return undefined;
|
||||||
|
|
||||||
const nextUrl = nextLink.split(";")[0].trim().slice(1, -1);
|
const nextUrl = nextLink.split(";")[0].trim().slice(1, -1);
|
||||||
return nextUrl.replace(BASE_URL, "").trim();
|
return nextUrl;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const canvasServiceUtils = {
|
export const canvasServiceUtils = {
|
||||||
async paginatedRequest<T>(request: { url: string }): Promise<T[]> {
|
async paginatedRequest<T>(request: { url: string }): Promise<T[]> {
|
||||||
var requestCount = 1;
|
var requestCount = 1;
|
||||||
const url = new URL(request.url!, BASE_URL);
|
const url = new URL(request.url);
|
||||||
url.searchParams.set("per_page", "100");
|
url.searchParams.set("per_page", "100");
|
||||||
|
|
||||||
const { data: firstData, response: firstResponse } =
|
const { data: firstData, response: firstResponse } =
|
||||||
await webRequestor.get<T>(url.toString());
|
await webRequestor.get<T>(url.toString());
|
||||||
|
|
||||||
if (!firstResponse.ok) {
|
|
||||||
console.error(
|
|
||||||
"error in response",
|
|
||||||
firstResponse.statusText,
|
|
||||||
firstResponse.body
|
|
||||||
);
|
|
||||||
throw new Error("error in response");
|
|
||||||
}
|
|
||||||
|
|
||||||
var returnData: T[] = firstData ? [firstData] : [];
|
var returnData: T[] = firstData ? [firstData] : [];
|
||||||
var nextUrl = getNextUrl(firstResponse.headers);
|
var nextUrl = getNextUrl(firstResponse.headers);
|
||||||
|
|
||||||
while (nextUrl) {
|
while (nextUrl) {
|
||||||
requestCount += 1;
|
requestCount += 1;
|
||||||
const {data, response} = await webRequestor.get<T>(nextUrl);
|
const { data, response } = await webRequestor.get<T>(nextUrl);
|
||||||
if (data) {
|
if (data) {
|
||||||
returnData = [...returnData, data];
|
returnData = [...returnData, data];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
|
import { axiosClient } from "../axiosUtils";
|
||||||
|
|
||||||
type FetchOptions = Omit<RequestInit, "method">;
|
type FetchOptions = Omit<RequestInit, "method">;
|
||||||
|
|
||||||
const token = process.env.CANVAS_TOKEN;
|
const token = process.env.NEXT_PUBLIC_CANVAS_TOKEN;
|
||||||
if (!token) {
|
if (!token) {
|
||||||
throw new Error("CANVAS_TOKEN not in environment");
|
throw new Error("CANVAS_TOKEN not in environment");
|
||||||
}
|
}
|
||||||
|
|
||||||
const baseUrl = `${process.env.CANVAS_URL}/api/v1/`;
|
|
||||||
const rateLimitRetryCount = 6;
|
const rateLimitRetryCount = 6;
|
||||||
const rateLimitSleepInterval = 1000;
|
const rateLimitSleepInterval = 1000;
|
||||||
|
|
||||||
@@ -112,27 +113,21 @@ const recursiveDelete = async (
|
|||||||
};
|
};
|
||||||
export const webRequestor = {
|
export const webRequestor = {
|
||||||
getMany: async <T>(url: string, options: FetchOptions = {}) => {
|
getMany: async <T>(url: string, options: FetchOptions = {}) => {
|
||||||
const response = await fetch(url, {
|
const response = await axiosClient.get<T[]>(url, {
|
||||||
...options,
|
|
||||||
method: "GET",
|
|
||||||
headers: {
|
headers: {
|
||||||
...options.headers,
|
|
||||||
Authorization: `Bearer ${token}`,
|
Authorization: `Bearer ${token}`,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
return { data: await deserialize<T[]>(response), response };
|
return { data: response.data, response };
|
||||||
},
|
},
|
||||||
|
|
||||||
get: async <T>(url: string, options: FetchOptions = {}) => {
|
get: async <T>(url: string) => {
|
||||||
const response = await fetch(url, {
|
const response = await axiosClient.get<T>(url, {
|
||||||
...options,
|
|
||||||
method: "GET",
|
|
||||||
headers: {
|
headers: {
|
||||||
...options.headers,
|
|
||||||
Authorization: `Bearer ${token}`,
|
Authorization: `Bearer ${token}`,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
return { data: await deserialize<T>(response), response };
|
return { data: response.data, response };
|
||||||
},
|
},
|
||||||
|
|
||||||
post: async (url: string, body: any) => {
|
post: async (url: string, body: any) => {
|
||||||
|
|||||||
Reference in New Issue
Block a user