jbilcke-hf HF staff commited on
Commit
f39b1f2
·
1 Parent(s): 74224a2

working to add the firehose

Browse files
.env CHANGED
@@ -10,4 +10,11 @@ VIDEOCHAIN_API_TOKEN=
10
  # Not supported yet
11
  REPLICATE_API_TOKEN=
12
  REPLICATE_API_MODEL="lucataco/sdxl-panoramic"
13
- REPLICATE_API_MODEL_VERSION="76acc4075d0633dcb3823c1fed0419de21d42001b65c816c7b5b9beff30ec8cd"
 
 
 
 
 
 
 
 
10
  # Not supported yet
11
  REPLICATE_API_TOKEN=
12
  REPLICATE_API_MODEL="lucataco/sdxl-panoramic"
13
+ REPLICATE_API_MODEL_VERSION="76acc4075d0633dcb3823c1fed0419de21d42001b65c816c7b5b9beff30ec8cd"
14
+
15
+ # ----------- COMMUNITY SHARING (OPTIONAL, YOU DON'T NEED THIS IN LOCAL) -----------
16
+ # You don't need those community sharing options to run Panoremix
17
+ # locally or on your own server (they are meant to be used by the Hugging Face team)
18
+ COMMUNITY_API_URL=
19
+ COMMUNITY_API_TOKEN=
20
+ COMMUNITY_API_ID=
src/app/engine/community.ts ADDED
@@ -0,0 +1,127 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ "use server"
2
+
3
+ import { v4 as uuidv4 } from "uuid"
4
+
5
+ import { CreatePostResponse, GetAppPostsResponse, Post } from "@/types"
6
+
7
+ const apiUrl = `${process.env.COMMUNITY_API_URL || ""}`
8
+ const apiToken = `${process.env.COMMUNITY_API_TOKEN || ""}`
9
+ const appId = `${process.env.APP_ID || ""}`
10
+
11
+ export async function postToCommunity({
12
+ prompt,
13
+ assetUrl,
14
+ }: {
15
+ prompt: string
16
+ assetUrl: string
17
+ }): Promise<Post> {
18
+ // if the community API is disabled,
19
+ // we don't fail, we just mock
20
+ if (!apiUrl) {
21
+ const mockPost: Post = {
22
+ postId: uuidv4(),
23
+ appId: "mock",
24
+ prompt,
25
+ previewUrl: assetUrl,
26
+ assetUrl,
27
+ createdAt: new Date().toISOString(),
28
+ upvotes: 0,
29
+ downvotes: 0
30
+ }
31
+ return mockPost
32
+ }
33
+
34
+ if (!prompt) {
35
+ console.error(`cannot call the community API without a prompt, aborting..`)
36
+ throw new Error(`cannot call the community API without a prompt, aborting..`)
37
+ }
38
+ if (!assetUrl) {
39
+ console.error(`cannot call the community API without an assetUrl, aborting..`)
40
+ throw new Error(`cannot call the community API without an assetUrl, aborting..`)
41
+ }
42
+
43
+ try {
44
+ console.log(`calling POST ${apiUrl}/post with prompt: ${prompt}`)
45
+
46
+ const postId = uuidv4()
47
+
48
+ const post: Partial<Post> = { postId, appId, prompt, assetUrl }
49
+
50
+ console.table(post)
51
+
52
+ const res = await fetch(`${apiUrl}/post`, {
53
+ method: "POST",
54
+ headers: {
55
+ Accept: "application/json",
56
+ "Content-Type": "application/json",
57
+ Authorization: `Bearer ${apiToken}`,
58
+ },
59
+ body: JSON.stringify(post),
60
+ cache: 'no-store',
61
+ // we can also use this (see https://vercel.com/blog/vercel-cache-api-nextjs-cache)
62
+ // next: { revalidate: 1 }
63
+ })
64
+
65
+ // console.log("res:", res)
66
+ // The return value is *not* serialized
67
+ // You can return Date, Map, Set, etc.
68
+
69
+ // Recommendation: handle errors
70
+ if (res.status !== 200) {
71
+ // This will activate the closest `error.js` Error Boundary
72
+ throw new Error('Failed to fetch data')
73
+ }
74
+
75
+ const response = (await res.json()) as CreatePostResponse
76
+ // console.log("response:", response)
77
+ return response.post
78
+ } catch (err) {
79
+ const error = `failed to post to community: ${err}`
80
+ console.error(error)
81
+ throw new Error(error)
82
+ }
83
+ }
84
+
85
+ export async function getLatestPosts(): Promise<Post[]> {
86
+
87
+ let posts: Post[] = []
88
+
89
+ // if the community API is disabled we don't fail,
90
+ // we just mock
91
+ if (!apiUrl) {
92
+ return posts
93
+ }
94
+
95
+ try {
96
+ // console.log(`calling GET ${apiUrl}/posts with renderId: ${renderId}`)
97
+ const res = await fetch(`${apiUrl}/posts/${appId}`, {
98
+ method: "GET",
99
+ headers: {
100
+ Accept: "application/json",
101
+ "Content-Type": "application/json",
102
+ Authorization: `Bearer ${apiToken}`,
103
+ },
104
+ cache: 'no-store',
105
+ // we can also use this (see https://vercel.com/blog/vercel-cache-api-nextjs-cache)
106
+ // next: { revalidate: 1 }
107
+ })
108
+
109
+ // console.log("res:", res)
110
+ // The return value is *not* serialized
111
+ // You can return Date, Map, Set, etc.
112
+
113
+ // Recommendation: handle errors
114
+ if (res.status !== 200) {
115
+ // This will activate the closest `error.js` Error Boundary
116
+ throw new Error('Failed to fetch data')
117
+ }
118
+
119
+ const response = (await res.json()) as GetAppPostsResponse
120
+ // console.log("response:", response)
121
+ return Array.isArray(response?.posts) ? response?.posts : []
122
+ } catch (err) {
123
+ const error = `failed to get posts: ${err}`
124
+ console.error(error)
125
+ throw new Error(error)
126
+ }
127
+ }
src/app/engine/render.ts CHANGED
@@ -29,10 +29,10 @@ export async function newRender({
29
  }
30
 
31
  prompt = [
32
- `hdri view`,
33
- `highly detailed`,
34
- `intricate details`,
35
- prompt
36
  ].join(', ')
37
 
38
  // return await Gorgon.get(cacheKey, async () => {
 
29
  }
30
 
31
  prompt = [
32
+ `hdri view`,
33
+ `highly detailed`,
34
+ `intricate details`,
35
+ prompt
36
  ].join(', ')
37
 
38
  // return await Gorgon.get(cacheKey, async () => {
src/app/interface/top-menu/index.tsx CHANGED
@@ -45,7 +45,7 @@ export function TopMenu() {
45
  )}>
46
  <div className="flex flex-row flex-grow w-full">
47
  <Input
48
- placeholder={`Type a location name eg. "Spaceport in Mos Eisley", "Jurassic Park entrance", "Street at night, Metropolis"..`}
49
  className="w-full bg-neutral-300 text-neutral-800 dark:bg-neutral-300 dark:text-neutral-800 rounded-r-none"
50
  // disabled={atLeastOnePanelIsBusy}
51
  onChange={(e) => {
 
45
  )}>
46
  <div className="flex flex-row flex-grow w-full">
47
  <Input
48
+ placeholder={`Type a location e.g. "Jurassic Park entrance", "Spaceport in Mos Eisley"..`}
49
  className="w-full bg-neutral-300 text-neutral-800 dark:bg-neutral-300 dark:text-neutral-800 rounded-r-none"
50
  // disabled={atLeastOnePanelIsBusy}
51
  onChange={(e) => {
src/app/main.tsx CHANGED
@@ -12,7 +12,7 @@ import { SphericalImage } from "./interface/spherical-image"
12
  import { getRender, newRender } from "./engine/render"
13
  import { RenderedScene } from "@/types"
14
 
15
- export default function Main() {
16
  const [_isPending, startTransition] = useTransition()
17
 
18
  const prompt = useStore(state => state.prompt)
 
12
  import { getRender, newRender } from "./engine/render"
13
  import { RenderedScene } from "@/types"
14
 
15
+ export default function Generator() {
16
  const [_isPending, startTransition] = useTransition()
17
 
18
  const prompt = useStore(state => state.prompt)
src/app/todo.tsx ADDED
@@ -0,0 +1,21 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ "use client"
2
+
3
+ import { useState, useTransition } from "react"
4
+
5
+ import { Post } from "@/types"
6
+
7
+ export default function Main() {
8
+ const [_isPending, startTransition] = useTransition()
9
+ const posts = useState<Post[]>([])
10
+
11
+ return (
12
+ <div>
13
+ <h1>Panoremix</h1>
14
+ <h2>Generate 360° panoramas from text!</h2>
15
+
16
+ <h2>Explore latent locations discovered by the community</h2>
17
+
18
+
19
+ </div>
20
+ )
21
+ }
src/types.ts CHANGED
@@ -86,4 +86,28 @@ export interface ImageAnalysisResponse {
86
  export type RenderingEngine =
87
  | "VIDEOCHAIN"
88
  | "OPENAI"
89
- | "REPLICATE"
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
86
  export type RenderingEngine =
87
  | "VIDEOCHAIN"
88
  | "OPENAI"
89
+ | "REPLICATE"
90
+
91
+ export type Post = {
92
+ postId: string
93
+ appId: string
94
+ prompt: string
95
+ previewUrl: string
96
+ assetUrl: string
97
+ createdAt: string
98
+ upvotes: number
99
+ downvotes: number
100
+ }
101
+
102
+ export type CreatePostResponse = {
103
+ success?: boolean
104
+ error?: string
105
+ post: Post
106
+ }
107
+
108
+ export type GetAppPostsResponse = {
109
+ success?: boolean
110
+ error?: string
111
+ posts: Post[]
112
+ }
113
+