Skip to content

Commit e9971b1

Browse files
authored
feat(ui): paginate model gallery (mudler#4886)
Signed-off-by: Ettore Di Giacinto <mudler@localai.io>
1 parent 5b59b5e commit e9971b1

File tree

4 files changed

+117
-3
lines changed

4 files changed

+117
-3
lines changed

core/gallery/gallery.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -114,7 +114,7 @@ func FindModel(models []*GalleryModel, name string, basePath string) *GalleryMod
114114
// List available models
115115
// Models galleries are a list of yaml files that are hosted on a remote server (for example github).
116116
// Each yaml file contains a list of models that can be downloaded and optionally overrides to define a new model setting.
117-
func AvailableGalleryModels(galleries []config.Gallery, basePath string) ([]*GalleryModel, error) {
117+
func AvailableGalleryModels(galleries []config.Gallery, basePath string) (GalleryModels, error) {
118118
var models []*GalleryModel
119119

120120
// Get models from galleries

core/gallery/request.go

+12
Original file line numberDiff line numberDiff line change
@@ -62,3 +62,15 @@ func (gm GalleryModels) FindByName(name string) *GalleryModel {
6262
}
6363
return nil
6464
}
65+
66+
func (gm GalleryModels) Paginate(pageNum int, itemsNum int) GalleryModels {
67+
start := (pageNum - 1) * itemsNum
68+
end := start + itemsNum
69+
if start > len(gm) {
70+
start = len(gm)
71+
}
72+
if end > len(gm) {
73+
end = len(gm)
74+
}
75+
return gm[start:end]
76+
}

core/http/routes/ui.go

+75-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,9 @@ package routes
33
import (
44
"fmt"
55
"html/template"
6+
"math"
67
"sort"
8+
"strconv"
79
"strings"
810

911
"github.com/mudler/LocalAI/core/config"
@@ -126,6 +128,8 @@ func RegisterUIRoutes(app *fiber.App,
126128
// Show the Models page (all models)
127129
app.Get("/browse", func(c *fiber.Ctx) error {
128130
term := c.Query("term")
131+
page := c.Query("page")
132+
items := c.Query("items")
129133

130134
models, _ := gallery.AvailableGalleryModels(appConfig.Galleries, appConfig.ModelPath)
131135

@@ -164,13 +168,64 @@ func RegisterUIRoutes(app *fiber.App,
164168
// "ApplicationConfig": appConfig,
165169
}
166170

171+
if page == "" {
172+
page = "1"
173+
}
174+
175+
if page != "" {
176+
log.Debug().Msgf("page : %+v\n", page)
177+
// return a subset of the models
178+
pageNum, err := strconv.Atoi(page)
179+
if err != nil {
180+
return c.Status(fiber.StatusBadRequest).SendString("Invalid page number")
181+
}
182+
183+
if pageNum == 0 {
184+
return c.Render("views/models", summary)
185+
}
186+
187+
itemsNum, err := strconv.Atoi(items)
188+
if err != nil {
189+
itemsNum = 21
190+
}
191+
192+
totalPages := int(math.Ceil(float64(len(models)) / float64(itemsNum)))
193+
194+
models = models.Paginate(pageNum, itemsNum)
195+
196+
log.Debug().Msgf("number of models : %+v\n", len(models))
197+
prevPage := pageNum - 1
198+
nextPage := pageNum + 1
199+
if prevPage < 1 {
200+
prevPage = 1
201+
}
202+
if nextPage > totalPages {
203+
nextPage = totalPages
204+
}
205+
if prevPage != pageNum {
206+
summary["PrevPage"] = prevPage
207+
}
208+
summary["NextPage"] = nextPage
209+
summary["TotalPages"] = totalPages
210+
summary["CurrentPage"] = pageNum
211+
summary["Models"] = template.HTML(elements.ListModels(models, processingModels, galleryService))
212+
213+
log.Debug().Msgf("totalPages : %+v\n", totalPages)
214+
log.Debug().Msgf("prevPage : %+v\n", prevPage)
215+
log.Debug().Msgf("nextPage : %+v\n", nextPage)
216+
log.Debug().Msgf("CurrentPage : %+v\n", pageNum)
217+
}
218+
167219
// Render index
168220
return c.Render("views/models", summary)
169221
})
170222

171223
// Show the models, filtered from the user input
172224
// https://htmx.org/examples/active-search/
173225
app.Post("/browse/search/models", func(c *fiber.Ctx) error {
226+
page := c.Query("page")
227+
items := c.Query("items")
228+
174229
form := struct {
175230
Search string `form:"search"`
176231
}{}
@@ -180,7 +235,26 @@ func RegisterUIRoutes(app *fiber.App,
180235

181236
models, _ := gallery.AvailableGalleryModels(appConfig.Galleries, appConfig.ModelPath)
182237

183-
return c.SendString(elements.ListModels(gallery.GalleryModels(models).Search(form.Search), processingModels, galleryService))
238+
if page != "" {
239+
// return a subset of the models
240+
pageNum, err := strconv.Atoi(page)
241+
if err != nil {
242+
return c.Status(fiber.StatusBadRequest).SendString("Invalid page number")
243+
}
244+
245+
itemsNum, err := strconv.Atoi(items)
246+
if err != nil {
247+
itemsNum = 21
248+
}
249+
250+
models = models.Paginate(pageNum, itemsNum)
251+
}
252+
253+
if form.Search != "" {
254+
models = models.Search(form.Search)
255+
}
256+
257+
return c.SendString(elements.ListModels(models, processingModels, galleryService))
184258
})
185259

186260
/*

core/http/views/models.html

+29-1
Original file line numberDiff line numberDiff line change
@@ -75,9 +75,37 @@ <h2>Filter by type:</h2>
7575
hx-indicator=".htmx-indicator">
7676

7777
<div id="search-results">{{.Models}}</div>
78+
<!-- Pagination -->
79+
<div class="flex justify-center mt-5">
80+
<div class="flex items
81+
-center">
82+
<button onclick="window.location.href='browse?page={{.PrevPage}}'" class="bg-gray-800 text-gray-300 hover:bg-gray-700 hover:text-gray-200 px-3 py-1 rounded-l-md" {{if not .PrevPage}}disabled{{end}}
83+
><i class="fas fa-arrow-left"></i></button>
84+
<button onclick="window.location.href='browse?page={{.NextPage}}'" class="bg-gray-800 text-gray-300 hover:bg-gray-700 hover:text-gray-200 px-3 py-1 rounded-r-md" {{if not .NextPage}}disabled{{end}}
85+
><i class="fas fa-arrow-right"></i></button>
86+
<!--
87+
TODO: do not refresh the page, but use htmx.
88+
This however requires the page calculation to be here instead that in the golang backend.
89+
<button class="bg-gray-800 text-gray-300 hover:bg-gray-700 hover:text-gray-200 px-3 py-1 rounded-l-md"
90+
hx-post="browse/search/models?page={{.PrevPage}}"
91+
hx-target="#search-results"
92+
hx-indicator=".htmx-indicator"
93+
{{if not .PrevPage}}disabled{{end}}
94+
>
95+
<i class="fas fa-arrow-left"></i>
96+
</button>
97+
<button class="bg-gray-800 text-gray-300 hover:bg-gray-700 hover:text-gray-200 px-3 py-1 rounded-r-md"
98+
hx-post="browse/search/models?page={{.NextPage}}"
99+
hx-target="#search-results"
100+
hx-indicator=".htmx-indicator"
101+
{{if not .NextPage}}disabled{{end}}
102+
>
103+
<i class="fas fa-arrow-right"></i>
104+
</button>
105+
-->
106+
</div>
78107
</div>
79108
</div>
80-
81109
{{template "views/partials/footer" .}}
82110
</div>
83111

0 commit comments

Comments
 (0)