mirror of
https://github.com/pikami/cosmium.git
synced 2025-09-18 11:05:53 +01:00
Compare commits
No commits in common. "a3bea16a26d67bae51e7787523553bb15dddae58" and "51e3311ba4232e52fb1445893e313885fe8a2b46" have entirely different histories.
a3bea16a26
...
51e3311ba4
@ -15,7 +15,7 @@ jobs:
|
|||||||
uses: crazy-max/ghaction-xgo@e22d3c8b089adba750d5a74738b8e95d96f0c991 # v3.1.0
|
uses: crazy-max/ghaction-xgo@e22d3c8b089adba750d5a74738b8e95d96f0c991 # v3.1.0
|
||||||
with:
|
with:
|
||||||
xgo_version: latest
|
xgo_version: latest
|
||||||
go_version: 1.25.1
|
go_version: 1.24.0
|
||||||
dest: dist
|
dest: dist
|
||||||
pkg: sharedlibrary
|
pkg: sharedlibrary
|
||||||
prefix: cosmium
|
prefix: cosmium
|
||||||
|
4
.github/workflows/release.yml
vendored
4
.github/workflows/release.yml
vendored
@ -21,13 +21,13 @@ jobs:
|
|||||||
- name: Set up Go
|
- name: Set up Go
|
||||||
uses: actions/setup-go@v5
|
uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
go-version: 1.25.1
|
go-version: 1.24.0
|
||||||
|
|
||||||
- name: Cross-Compile with xgo
|
- name: Cross-Compile with xgo
|
||||||
uses: crazy-max/ghaction-xgo@e22d3c8b089adba750d5a74738b8e95d96f0c991 # v3.1.0
|
uses: crazy-max/ghaction-xgo@e22d3c8b089adba750d5a74738b8e95d96f0c991 # v3.1.0
|
||||||
with:
|
with:
|
||||||
xgo_version: latest
|
xgo_version: latest
|
||||||
go_version: 1.25.1
|
go_version: 1.24.0
|
||||||
dest: sharedlibrary_dist
|
dest: sharedlibrary_dist
|
||||||
pkg: sharedlibrary
|
pkg: sharedlibrary
|
||||||
prefix: cosmium
|
prefix: cosmium
|
||||||
|
2
Makefile
2
Makefile
@ -9,7 +9,7 @@ SERVER_LOCATION=./cmd/server
|
|||||||
SHARED_LIB_LOCATION=./sharedlibrary
|
SHARED_LIB_LOCATION=./sharedlibrary
|
||||||
SHARED_LIB_OPT=-buildmode=c-shared
|
SHARED_LIB_OPT=-buildmode=c-shared
|
||||||
XGO_TARGETS=linux/amd64,linux/arm64,windows/amd64,windows/arm64,darwin/amd64,darwin/arm64
|
XGO_TARGETS=linux/amd64,linux/arm64,windows/amd64,windows/arm64,darwin/amd64,darwin/arm64
|
||||||
GOVERSION=1.25.1
|
GOVERSION=1.24.0
|
||||||
|
|
||||||
DIST_DIR=dist
|
DIST_DIR=dist
|
||||||
|
|
||||||
|
@ -105,10 +105,10 @@ All mentioned arguments can also be set using environment variables:
|
|||||||
|
|
||||||
Cosmium supports multiple storage backends for saving, loading, and managing data at runtime.
|
Cosmium supports multiple storage backends for saving, loading, and managing data at runtime.
|
||||||
|
|
||||||
| Backend | Storage Location | Write Behavior | Memory Usage |
|
| Backend | Storage Location | Write Behavior | Memory Usage | Supports Initial JSON Load |
|
||||||
|----------|--------------------------|--------------------------|----------------------|
|
|----------|--------------------------|--------------------------|----------------------|----------------------------|
|
||||||
| `json` (default) | JSON file on disk 📄 | On application exit ⏳ | 🛑 More than Badger |
|
| `json` (default) | JSON file on disk 📄 | On application exit ⏳ | 🛑 More than Badger | ✅ Yes |
|
||||||
| `badger` | BadgerDB database on disk ⚡ | Immediately on write 🚀 | ✅ Less than JSON |
|
| `badger` | BadgerDB database on disk ⚡ | Immediately on write 🚀 | ✅ Less than JSON | ❌ No |
|
||||||
|
|
||||||
|
|
||||||
The `badger` backend is generally recommended as it uses less memory and writes data to disk immediately. However, if you need to load initial data from a JSON file, use the `json` backend.
|
The `badger` backend is generally recommended as it uses less memory and writes data to disk immediately. However, if you need to load initial data from a JSON file, use the `json` backend.
|
||||||
|
@ -94,6 +94,11 @@ func (c *ServerConfig) PopulateCalculatedFields() {
|
|||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if c.DataStore == DataStoreBadger && c.InitialDataFilePath != "" {
|
||||||
|
logger.ErrorLn("InitialData option is currently not supported with Badger data store")
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *ServerConfig) ApplyDefaultsToEmptyFields() {
|
func (c *ServerConfig) ApplyDefaultsToEmptyFields() {
|
||||||
|
@ -5,7 +5,6 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/pikami/cosmium/api/headers"
|
|
||||||
"github.com/pikami/cosmium/internal/constants"
|
"github.com/pikami/cosmium/internal/constants"
|
||||||
"github.com/pikami/cosmium/internal/datastore"
|
"github.com/pikami/cosmium/internal/datastore"
|
||||||
)
|
)
|
||||||
@ -17,7 +16,7 @@ func (h *Handlers) GetAllCollections(c *gin.Context) {
|
|||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
database, _ := h.dataStore.GetDatabase(databaseId)
|
database, _ := h.dataStore.GetDatabase(databaseId)
|
||||||
|
|
||||||
c.Header(headers.ItemCount, fmt.Sprintf("%d", len(collections)))
|
c.Header("x-ms-item-count", fmt.Sprintf("%d", len(collections)))
|
||||||
c.IndentedJSON(http.StatusOK, gin.H{
|
c.IndentedJSON(http.StatusOK, gin.H{
|
||||||
"_rid": database.ResourceID,
|
"_rid": database.ResourceID,
|
||||||
"DocumentCollections": collections,
|
"DocumentCollections": collections,
|
||||||
|
@ -5,7 +5,6 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/pikami/cosmium/api/headers"
|
|
||||||
"github.com/pikami/cosmium/internal/constants"
|
"github.com/pikami/cosmium/internal/constants"
|
||||||
"github.com/pikami/cosmium/internal/datastore"
|
"github.com/pikami/cosmium/internal/datastore"
|
||||||
)
|
)
|
||||||
@ -13,7 +12,7 @@ import (
|
|||||||
func (h *Handlers) GetAllDatabases(c *gin.Context) {
|
func (h *Handlers) GetAllDatabases(c *gin.Context) {
|
||||||
databases, status := h.dataStore.GetAllDatabases()
|
databases, status := h.dataStore.GetAllDatabases()
|
||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
c.Header(headers.ItemCount, fmt.Sprintf("%d", len(databases)))
|
c.Header("x-ms-item-count", fmt.Sprintf("%d", len(databases)))
|
||||||
c.IndentedJSON(http.StatusOK, gin.H{
|
c.IndentedJSON(http.StatusOK, gin.H{
|
||||||
"_rid": "",
|
"_rid": "",
|
||||||
"Databases": databases,
|
"Databases": databases,
|
||||||
|
@ -9,7 +9,6 @@ import (
|
|||||||
jsonpatch "github.com/cosmiumdev/json-patch/v5"
|
jsonpatch "github.com/cosmiumdev/json-patch/v5"
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
apimodels "github.com/pikami/cosmium/api/api_models"
|
apimodels "github.com/pikami/cosmium/api/api_models"
|
||||||
"github.com/pikami/cosmium/api/headers"
|
|
||||||
"github.com/pikami/cosmium/internal/constants"
|
"github.com/pikami/cosmium/internal/constants"
|
||||||
"github.com/pikami/cosmium/internal/converters"
|
"github.com/pikami/cosmium/internal/converters"
|
||||||
"github.com/pikami/cosmium/internal/datastore"
|
"github.com/pikami/cosmium/internal/datastore"
|
||||||
@ -27,7 +26,7 @@ func (h *Handlers) GetAllDocuments(c *gin.Context) {
|
|||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
collection, _ := h.dataStore.GetCollection(databaseId, collectionId)
|
collection, _ := h.dataStore.GetCollection(databaseId, collectionId)
|
||||||
|
|
||||||
c.Header(headers.ItemCount, fmt.Sprintf("%d", len(documents)))
|
c.Header("x-ms-item-count", fmt.Sprintf("%d", len(documents)))
|
||||||
c.IndentedJSON(http.StatusOK, gin.H{
|
c.IndentedJSON(http.StatusOK, gin.H{
|
||||||
"_rid": collection.ID,
|
"_rid": collection.ID,
|
||||||
"Documents": documents,
|
"Documents": documents,
|
||||||
@ -190,7 +189,7 @@ func (h *Handlers) DocumentsPost(c *gin.Context) {
|
|||||||
collectionId := c.Param("collId")
|
collectionId := c.Param("collId")
|
||||||
|
|
||||||
// Handle batch requests
|
// Handle batch requests
|
||||||
isBatchRequest, _ := strconv.ParseBool(c.GetHeader(headers.IsBatchRequest))
|
isBatchRequest, _ := strconv.ParseBool(c.GetHeader("x-ms-cosmos-is-batch-request"))
|
||||||
if isBatchRequest {
|
if isBatchRequest {
|
||||||
h.handleBatchRequest(c)
|
h.handleBatchRequest(c)
|
||||||
return
|
return
|
||||||
@ -202,17 +201,8 @@ func (h *Handlers) DocumentsPost(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle query plan requests
|
query := requestBody["query"]
|
||||||
isQueryPlanRequest, _ := strconv.ParseBool(c.GetHeader(headers.IsQueryPlanRequest))
|
if query != nil {
|
||||||
if isQueryPlanRequest {
|
|
||||||
c.IndentedJSON(http.StatusOK, constants.QueryPlanResponse)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle query requests
|
|
||||||
isQueryRequest, _ := strconv.ParseBool(c.GetHeader(headers.IsQuery))
|
|
||||||
isQueryRequestAltHeader, _ := strconv.ParseBool(c.GetHeader(headers.Query))
|
|
||||||
if isQueryRequest || isQueryRequestAltHeader {
|
|
||||||
h.handleDocumentQuery(c, requestBody)
|
h.handleDocumentQuery(c, requestBody)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -222,7 +212,7 @@ func (h *Handlers) DocumentsPost(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
isUpsert, _ := strconv.ParseBool(c.GetHeader(headers.IsUpsert))
|
isUpsert, _ := strconv.ParseBool(c.GetHeader("x-ms-documentdb-is-upsert"))
|
||||||
if isUpsert {
|
if isUpsert {
|
||||||
h.dataStore.DeleteDocument(databaseId, collectionId, requestBody["id"].(string))
|
h.dataStore.DeleteDocument(databaseId, collectionId, requestBody["id"].(string))
|
||||||
}
|
}
|
||||||
@ -257,6 +247,11 @@ func (h *Handlers) handleDocumentQuery(c *gin.Context, requestBody map[string]in
|
|||||||
databaseId := c.Param("databaseId")
|
databaseId := c.Param("databaseId")
|
||||||
collectionId := c.Param("collId")
|
collectionId := c.Param("collId")
|
||||||
|
|
||||||
|
if c.GetHeader("x-ms-cosmos-is-query-plan-request") != "" {
|
||||||
|
c.IndentedJSON(http.StatusOK, constants.QueryPlanResponse)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
var queryParameters map[string]interface{}
|
var queryParameters map[string]interface{}
|
||||||
if paramsArray, ok := requestBody["parameters"].([]interface{}); ok {
|
if paramsArray, ok := requestBody["parameters"].([]interface{}); ok {
|
||||||
queryParameters = parametersToMap(paramsArray)
|
queryParameters = parametersToMap(paramsArray)
|
||||||
@ -271,7 +266,7 @@ func (h *Handlers) handleDocumentQuery(c *gin.Context, requestBody map[string]in
|
|||||||
}
|
}
|
||||||
|
|
||||||
collection, _ := h.dataStore.GetCollection(databaseId, collectionId)
|
collection, _ := h.dataStore.GetCollection(databaseId, collectionId)
|
||||||
c.Header(headers.ItemCount, fmt.Sprintf("%d", len(docs)))
|
c.Header("x-ms-item-count", fmt.Sprintf("%d", len(docs)))
|
||||||
c.IndentedJSON(http.StatusOK, gin.H{
|
c.IndentedJSON(http.StatusOK, gin.H{
|
||||||
"_rid": collection.ResourceID,
|
"_rid": collection.ResourceID,
|
||||||
"Documents": docs,
|
"Documents": docs,
|
||||||
|
@ -6,7 +6,6 @@ import (
|
|||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/pikami/cosmium/api/config"
|
"github.com/pikami/cosmium/api/config"
|
||||||
"github.com/pikami/cosmium/api/headers"
|
|
||||||
"github.com/pikami/cosmium/internal/authentication"
|
"github.com/pikami/cosmium/internal/authentication"
|
||||||
"github.com/pikami/cosmium/internal/logger"
|
"github.com/pikami/cosmium/internal/logger"
|
||||||
)
|
)
|
||||||
@ -23,8 +22,8 @@ func Authentication(config *config.ServerConfig) gin.HandlerFunc {
|
|||||||
resourceType := urlToResourceType(requestUrl)
|
resourceType := urlToResourceType(requestUrl)
|
||||||
resourceId := requestToResourceId(c)
|
resourceId := requestToResourceId(c)
|
||||||
|
|
||||||
authHeader := c.Request.Header.Get(headers.Authorization)
|
authHeader := c.Request.Header.Get("authorization")
|
||||||
date := c.Request.Header.Get(headers.XDate)
|
date := c.Request.Header.Get("x-ms-date")
|
||||||
expectedSignature := authentication.GenerateSignature(
|
expectedSignature := authentication.GenerateSignature(
|
||||||
c.Request.Method, resourceType, resourceId, date, config.AccountKey)
|
c.Request.Method, resourceType, resourceId, date, config.AccountKey)
|
||||||
|
|
||||||
@ -86,7 +85,7 @@ func requestToResourceId(c *gin.Context) string {
|
|||||||
resourceId += "/udfs/" + udfId
|
resourceId += "/udfs/" + udfId
|
||||||
}
|
}
|
||||||
|
|
||||||
isFeed := c.Request.Header.Get(headers.AIM) == "Incremental Feed"
|
isFeed := c.Request.Header.Get("A-Im") == "Incremental Feed"
|
||||||
if resourceType == "pkranges" && isFeed {
|
if resourceType == "pkranges" && isFeed {
|
||||||
resourceId = collId
|
resourceId = collId
|
||||||
}
|
}
|
||||||
|
@ -4,11 +4,10 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/pikami/cosmium/api/headers"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func GetOffers(c *gin.Context) {
|
func GetOffers(c *gin.Context) {
|
||||||
c.Header(headers.ItemCount, "0")
|
c.Header("x-ms-item-count", "0")
|
||||||
c.IndentedJSON(http.StatusOK, gin.H{
|
c.IndentedJSON(http.StatusOK, gin.H{
|
||||||
"_rid": "",
|
"_rid": "",
|
||||||
"_count": 0,
|
"_count": 0,
|
||||||
|
@ -5,7 +5,6 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/pikami/cosmium/api/headers"
|
|
||||||
"github.com/pikami/cosmium/internal/constants"
|
"github.com/pikami/cosmium/internal/constants"
|
||||||
"github.com/pikami/cosmium/internal/datastore"
|
"github.com/pikami/cosmium/internal/datastore"
|
||||||
"github.com/pikami/cosmium/internal/resourceid"
|
"github.com/pikami/cosmium/internal/resourceid"
|
||||||
@ -15,18 +14,18 @@ func (h *Handlers) GetPartitionKeyRanges(c *gin.Context) {
|
|||||||
databaseId := c.Param("databaseId")
|
databaseId := c.Param("databaseId")
|
||||||
collectionId := c.Param("collId")
|
collectionId := c.Param("collId")
|
||||||
|
|
||||||
if c.Request.Header.Get(headers.IfNoneMatch) != "" {
|
if c.Request.Header.Get("if-none-match") != "" {
|
||||||
c.AbortWithStatus(http.StatusNotModified)
|
c.AbortWithStatus(http.StatusNotModified)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
partitionKeyRanges, status := h.dataStore.GetPartitionKeyRanges(databaseId, collectionId)
|
partitionKeyRanges, status := h.dataStore.GetPartitionKeyRanges(databaseId, collectionId)
|
||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
c.Header(headers.ETag, "\"420\"")
|
c.Header("etag", "\"420\"")
|
||||||
c.Header(headers.LSN, "420")
|
c.Header("lsn", "420")
|
||||||
c.Header(headers.CosmosLsn, "420")
|
c.Header("x-ms-cosmos-llsn", "420")
|
||||||
c.Header(headers.GlobalCommittedLsn, "420")
|
c.Header("x-ms-global-committed-lsn", "420")
|
||||||
c.Header(headers.ItemCount, fmt.Sprintf("%d", len(partitionKeyRanges)))
|
c.Header("x-ms-item-count", fmt.Sprintf("%d", len(partitionKeyRanges)))
|
||||||
|
|
||||||
collectionRid := collectionId
|
collectionRid := collectionId
|
||||||
collection, _ := h.dataStore.GetCollection(databaseId, collectionId)
|
collection, _ := h.dataStore.GetCollection(databaseId, collectionId)
|
||||||
|
@ -5,7 +5,6 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/pikami/cosmium/api/headers"
|
|
||||||
"github.com/pikami/cosmium/internal/constants"
|
"github.com/pikami/cosmium/internal/constants"
|
||||||
"github.com/pikami/cosmium/internal/datastore"
|
"github.com/pikami/cosmium/internal/datastore"
|
||||||
)
|
)
|
||||||
@ -17,7 +16,7 @@ func (h *Handlers) GetAllStoredProcedures(c *gin.Context) {
|
|||||||
sps, status := h.dataStore.GetAllStoredProcedures(databaseId, collectionId)
|
sps, status := h.dataStore.GetAllStoredProcedures(databaseId, collectionId)
|
||||||
|
|
||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
c.Header(headers.ItemCount, fmt.Sprintf("%d", len(sps)))
|
c.Header("x-ms-item-count", fmt.Sprintf("%d", len(sps)))
|
||||||
c.IndentedJSON(http.StatusOK, gin.H{"_rid": "", "StoredProcedures": sps, "_count": len(sps)})
|
c.IndentedJSON(http.StatusOK, gin.H{"_rid": "", "StoredProcedures": sps, "_count": len(sps)})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,6 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/pikami/cosmium/api/headers"
|
|
||||||
"github.com/pikami/cosmium/internal/constants"
|
"github.com/pikami/cosmium/internal/constants"
|
||||||
"github.com/pikami/cosmium/internal/datastore"
|
"github.com/pikami/cosmium/internal/datastore"
|
||||||
)
|
)
|
||||||
@ -17,7 +16,7 @@ func (h *Handlers) GetAllTriggers(c *gin.Context) {
|
|||||||
triggers, status := h.dataStore.GetAllTriggers(databaseId, collectionId)
|
triggers, status := h.dataStore.GetAllTriggers(databaseId, collectionId)
|
||||||
|
|
||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
c.Header(headers.ItemCount, fmt.Sprintf("%d", len(triggers)))
|
c.Header("x-ms-item-count", fmt.Sprintf("%d", len(triggers)))
|
||||||
c.IndentedJSON(http.StatusOK, gin.H{"_rid": "", "Triggers": triggers, "_count": len(triggers)})
|
c.IndentedJSON(http.StatusOK, gin.H{"_rid": "", "Triggers": triggers, "_count": len(triggers)})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,6 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
"github.com/pikami/cosmium/api/headers"
|
|
||||||
"github.com/pikami/cosmium/internal/constants"
|
"github.com/pikami/cosmium/internal/constants"
|
||||||
"github.com/pikami/cosmium/internal/datastore"
|
"github.com/pikami/cosmium/internal/datastore"
|
||||||
)
|
)
|
||||||
@ -17,7 +16,7 @@ func (h *Handlers) GetAllUserDefinedFunctions(c *gin.Context) {
|
|||||||
udfs, status := h.dataStore.GetAllUserDefinedFunctions(databaseId, collectionId)
|
udfs, status := h.dataStore.GetAllUserDefinedFunctions(databaseId, collectionId)
|
||||||
|
|
||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
c.Header(headers.ItemCount, fmt.Sprintf("%d", len(udfs)))
|
c.Header("x-ms-item-count", fmt.Sprintf("%d", len(udfs)))
|
||||||
c.IndentedJSON(http.StatusOK, gin.H{"_rid": "", "UserDefinedFunctions": udfs, "_count": len(udfs)})
|
c.IndentedJSON(http.StatusOK, gin.H{"_rid": "", "UserDefinedFunctions": udfs, "_count": len(udfs)})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -1,20 +0,0 @@
|
|||||||
package headers
|
|
||||||
|
|
||||||
const (
|
|
||||||
AIM = "A-Im"
|
|
||||||
Authorization = "authorization"
|
|
||||||
CosmosLsn = "x-ms-cosmos-llsn"
|
|
||||||
ETag = "etag"
|
|
||||||
GlobalCommittedLsn = "x-ms-global-committed-lsn"
|
|
||||||
IfNoneMatch = "if-none-match"
|
|
||||||
IsBatchRequest = "x-ms-cosmos-is-batch-request"
|
|
||||||
IsQueryPlanRequest = "x-ms-cosmos-is-query-plan-request"
|
|
||||||
IsUpsert = "x-ms-documentdb-is-upsert"
|
|
||||||
ItemCount = "x-ms-item-count"
|
|
||||||
LSN = "lsn"
|
|
||||||
XDate = "x-ms-date"
|
|
||||||
|
|
||||||
// Kinda retarded, but what can I do ¯\_(ツ)_/¯
|
|
||||||
IsQuery = "x-ms-documentdb-isquery" // Sent from python sdk and web explorer
|
|
||||||
Query = "x-ms-documentdb-query" // Sent from Go sdk
|
|
||||||
)
|
|
@ -8,7 +8,6 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/pikami/cosmium/api/config"
|
"github.com/pikami/cosmium/api/config"
|
||||||
"github.com/pikami/cosmium/api/headers"
|
|
||||||
"github.com/pikami/cosmium/internal/authentication"
|
"github.com/pikami/cosmium/internal/authentication"
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
)
|
)
|
||||||
@ -27,8 +26,8 @@ func Test_Documents_Read_Trailing_Slash(t *testing.T) {
|
|||||||
signature := authentication.GenerateSignature("GET", "docs", path, date, config.DefaultAccountKey)
|
signature := authentication.GenerateSignature("GET", "docs", path, date, config.DefaultAccountKey)
|
||||||
httpClient := &http.Client{}
|
httpClient := &http.Client{}
|
||||||
req, _ := http.NewRequest("GET", testUrl, nil)
|
req, _ := http.NewRequest("GET", testUrl, nil)
|
||||||
req.Header.Add(headers.XDate, date)
|
req.Header.Add("x-ms-date", date)
|
||||||
req.Header.Add(headers.Authorization, "sig="+url.QueryEscape(signature))
|
req.Header.Add("authorization", "sig="+url.QueryEscape(signature))
|
||||||
res, err := httpClient.Do(req)
|
res, err := httpClient.Do(req)
|
||||||
|
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
@ -20,7 +20,6 @@ func main() {
|
|||||||
switch configuration.DataStore {
|
switch configuration.DataStore {
|
||||||
case config.DataStoreBadger:
|
case config.DataStoreBadger:
|
||||||
dataStore = badgerdatastore.NewBadgerDataStore(badgerdatastore.BadgerDataStoreOptions{
|
dataStore = badgerdatastore.NewBadgerDataStore(badgerdatastore.BadgerDataStoreOptions{
|
||||||
InitialDataFilePath: configuration.InitialDataFilePath,
|
|
||||||
PersistDataFilePath: configuration.PersistDataFilePath,
|
PersistDataFilePath: configuration.PersistDataFilePath,
|
||||||
})
|
})
|
||||||
logger.InfoLn("Using Badger data store")
|
logger.InfoLn("Using Badger data store")
|
||||||
|
35
go.mod
35
go.mod
@ -1,31 +1,30 @@
|
|||||||
module github.com/pikami/cosmium
|
module github.com/pikami/cosmium
|
||||||
|
|
||||||
go 1.25.1
|
go 1.24.0
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.19.1
|
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.18.2
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/data/azcosmos v1.4.1
|
github.com/Azure/azure-sdk-for-go/sdk/data/azcosmos v1.4.0
|
||||||
github.com/cosmiumdev/json-patch/v5 v5.9.11
|
github.com/cosmiumdev/json-patch/v5 v5.9.11
|
||||||
github.com/dgraph-io/badger/v4 v4.8.0
|
github.com/dgraph-io/badger/v4 v4.8.0
|
||||||
github.com/gin-gonic/gin v1.10.1
|
github.com/gin-gonic/gin v1.10.1
|
||||||
github.com/google/uuid v1.6.0
|
github.com/google/uuid v1.6.0
|
||||||
github.com/stretchr/testify v1.11.1
|
github.com/stretchr/testify v1.10.0
|
||||||
github.com/vmihailenco/msgpack/v5 v5.4.1
|
github.com/vmihailenco/msgpack/v5 v5.4.1
|
||||||
golang.org/x/exp v0.0.0-20250911091902-df9299821621
|
golang.org/x/exp v0.0.0-20250819193227-8b4c13bb791b
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/Azure/azure-sdk-for-go v68.0.0+incompatible // indirect
|
github.com/Azure/azure-sdk-for-go v68.0.0+incompatible // indirect
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.2 // indirect
|
github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.2 // indirect
|
||||||
github.com/bytedance/gopkg v0.1.3 // indirect
|
github.com/bytedance/sonic v1.14.0 // indirect
|
||||||
github.com/bytedance/sonic v1.14.1 // indirect
|
|
||||||
github.com/bytedance/sonic/loader v0.3.0 // indirect
|
github.com/bytedance/sonic/loader v0.3.0 // indirect
|
||||||
github.com/cespare/xxhash/v2 v2.3.0 // indirect
|
github.com/cespare/xxhash/v2 v2.3.0 // indirect
|
||||||
github.com/cloudwego/base64x v0.1.6 // indirect
|
github.com/cloudwego/base64x v0.1.6 // indirect
|
||||||
github.com/davecgh/go-spew v1.1.1 // indirect
|
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||||
github.com/dgraph-io/ristretto/v2 v2.3.0 // indirect
|
github.com/dgraph-io/ristretto/v2 v2.3.0 // indirect
|
||||||
github.com/dustin/go-humanize v1.0.1 // indirect
|
github.com/dustin/go-humanize v1.0.1 // indirect
|
||||||
github.com/gabriel-vasile/mimetype v1.4.10 // indirect
|
github.com/gabriel-vasile/mimetype v1.4.9 // indirect
|
||||||
github.com/gin-contrib/sse v1.1.0 // indirect
|
github.com/gin-contrib/sse v1.1.0 // indirect
|
||||||
github.com/go-logr/logr v1.4.3 // indirect
|
github.com/go-logr/logr v1.4.3 // indirect
|
||||||
github.com/go-logr/stdr v1.2.2 // indirect
|
github.com/go-logr/stdr v1.2.2 // indirect
|
||||||
@ -46,15 +45,15 @@ require (
|
|||||||
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
||||||
github.com/ugorji/go/codec v1.3.0 // indirect
|
github.com/ugorji/go/codec v1.3.0 // indirect
|
||||||
github.com/vmihailenco/tagparser/v2 v2.0.0 // indirect
|
github.com/vmihailenco/tagparser/v2 v2.0.0 // indirect
|
||||||
go.opentelemetry.io/auto/sdk v1.2.1 // indirect
|
go.opentelemetry.io/auto/sdk v1.1.0 // indirect
|
||||||
go.opentelemetry.io/otel v1.38.0 // indirect
|
go.opentelemetry.io/otel v1.37.0 // indirect
|
||||||
go.opentelemetry.io/otel/metric v1.38.0 // indirect
|
go.opentelemetry.io/otel/metric v1.37.0 // indirect
|
||||||
go.opentelemetry.io/otel/trace v1.38.0 // indirect
|
go.opentelemetry.io/otel/trace v1.37.0 // indirect
|
||||||
golang.org/x/arch v0.21.0 // indirect
|
golang.org/x/arch v0.20.0 // indirect
|
||||||
golang.org/x/crypto v0.42.0 // indirect
|
golang.org/x/crypto v0.41.0 // indirect
|
||||||
golang.org/x/net v0.44.0 // indirect
|
golang.org/x/net v0.43.0 // indirect
|
||||||
golang.org/x/sys v0.36.0 // indirect
|
golang.org/x/sys v0.35.0 // indirect
|
||||||
golang.org/x/text v0.29.0 // indirect
|
golang.org/x/text v0.28.0 // indirect
|
||||||
google.golang.org/protobuf v1.36.9 // indirect
|
google.golang.org/protobuf v1.36.7 // indirect
|
||||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
)
|
)
|
||||||
|
70
go.sum
70
go.sum
@ -1,19 +1,17 @@
|
|||||||
github.com/Azure/azure-sdk-for-go v68.0.0+incompatible h1:fcYLmCpyNYRnvJbPerq7U0hS+6+I79yEDJBqVNcqUzU=
|
github.com/Azure/azure-sdk-for-go v68.0.0+incompatible h1:fcYLmCpyNYRnvJbPerq7U0hS+6+I79yEDJBqVNcqUzU=
|
||||||
github.com/Azure/azure-sdk-for-go v68.0.0+incompatible/go.mod h1:9XXNKU+eRnpl9moKnB4QOLf1HestfXbmab5FXxiDBjc=
|
github.com/Azure/azure-sdk-for-go v68.0.0+incompatible/go.mod h1:9XXNKU+eRnpl9moKnB4QOLf1HestfXbmab5FXxiDBjc=
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.19.1 h1:5YTBM8QDVIBN3sxBil89WfdAAqDZbyJTgh688DSxX5w=
|
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.18.2 h1:Hr5FTipp7SL07o2FvoVOX9HRiRH3CR3Mj8pxqCcdD5A=
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.19.1/go.mod h1:YD5h/ldMsG0XiIw7PdyNhLxaM317eFh5yNLccNfGdyw=
|
github.com/Azure/azure-sdk-for-go/sdk/azcore v1.18.2/go.mod h1:QyVsSSN64v5TGltphKLQ2sQxe4OBQg0J1eKRcVBnfgE=
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.10.1 h1:B+blDbyVIG3WaikNxPnhPiJ1MThR03b3vKGtER95TP4=
|
github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.10.1 h1:B+blDbyVIG3WaikNxPnhPiJ1MThR03b3vKGtER95TP4=
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.10.1/go.mod h1:JdM5psgjfBf5fo2uWOZhflPWyDBZ/O/CNAH9CtsuZE4=
|
github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.10.1/go.mod h1:JdM5psgjfBf5fo2uWOZhflPWyDBZ/O/CNAH9CtsuZE4=
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/data/azcosmos v1.4.1 h1:ToPLhnXvatKVN4ZkcxLOwcXOJhdu4iQl8w0efeuDz9Y=
|
github.com/Azure/azure-sdk-for-go/sdk/data/azcosmos v1.4.0 h1:TSaH6Lj0m8bDr4vX1+LC1KLQTnLzZb3tOxrx/PLqw+c=
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/data/azcosmos v1.4.1/go.mod h1:Krtog/7tz27z75TwM5cIS8bxEH4dcBUezcq+kGVeZEo=
|
github.com/Azure/azure-sdk-for-go/sdk/data/azcosmos v1.4.0/go.mod h1:Krtog/7tz27z75TwM5cIS8bxEH4dcBUezcq+kGVeZEo=
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.2 h1:9iefClla7iYpfYWdzPCRDozdmndjTm8DXdpCzPajMgA=
|
github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.2 h1:9iefClla7iYpfYWdzPCRDozdmndjTm8DXdpCzPajMgA=
|
||||||
github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.2/go.mod h1:XtLgD3ZD34DAaVIIAyG3objl5DynM3CQ/vMcbBNJZGI=
|
github.com/Azure/azure-sdk-for-go/sdk/internal v1.11.2/go.mod h1:XtLgD3ZD34DAaVIIAyG3objl5DynM3CQ/vMcbBNJZGI=
|
||||||
github.com/AzureAD/microsoft-authentication-library-for-go v1.4.2 h1:oygO0locgZJe7PpYPXT5A29ZkwJaPqcva7BVeemZOZs=
|
github.com/AzureAD/microsoft-authentication-library-for-go v1.4.2 h1:oygO0locgZJe7PpYPXT5A29ZkwJaPqcva7BVeemZOZs=
|
||||||
github.com/AzureAD/microsoft-authentication-library-for-go v1.4.2/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI=
|
github.com/AzureAD/microsoft-authentication-library-for-go v1.4.2/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI=
|
||||||
github.com/bytedance/gopkg v0.1.3 h1:TPBSwH8RsouGCBcMBktLt1AymVo2TVsBVCY4b6TnZ/M=
|
github.com/bytedance/sonic v1.14.0 h1:/OfKt8HFw0kh2rj8N0F6C/qPGRESq0BbaNZgcNXXzQQ=
|
||||||
github.com/bytedance/gopkg v0.1.3/go.mod h1:576VvJ+eJgyCzdjS+c4+77QF3p7ubbtiKARP3TxducM=
|
github.com/bytedance/sonic v1.14.0/go.mod h1:WoEbx8WTcFJfzCe0hbmyTGrfjt8PzNEBdxlNUO24NhA=
|
||||||
github.com/bytedance/sonic v1.14.1 h1:FBMC0zVz5XUmE4z9wF4Jey0An5FueFvOsTKKKtwIl7w=
|
|
||||||
github.com/bytedance/sonic v1.14.1/go.mod h1:gi6uhQLMbTdeP0muCnrjHLeCUPyb70ujhnNlhOylAFc=
|
|
||||||
github.com/bytedance/sonic/loader v0.3.0 h1:dskwH8edlzNMctoruo8FPTJDF3vLtDT0sXZwvZJyqeA=
|
github.com/bytedance/sonic/loader v0.3.0 h1:dskwH8edlzNMctoruo8FPTJDF3vLtDT0sXZwvZJyqeA=
|
||||||
github.com/bytedance/sonic/loader v0.3.0/go.mod h1:N8A3vUdtUebEY2/VQC0MyhYeKUFosQU6FxH2JmUe6VI=
|
github.com/bytedance/sonic/loader v0.3.0/go.mod h1:N8A3vUdtUebEY2/VQC0MyhYeKUFosQU6FxH2JmUe6VI=
|
||||||
github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs=
|
github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs=
|
||||||
@ -33,8 +31,8 @@ github.com/dgryski/go-farm v0.0.0-20240924180020-3414d57e47da h1:aIftn67I1fkbMa5
|
|||||||
github.com/dgryski/go-farm v0.0.0-20240924180020-3414d57e47da/go.mod h1:SqUrOPUnsFjfmXRMNPybcSiG0BgUW2AuFH8PAnS2iTw=
|
github.com/dgryski/go-farm v0.0.0-20240924180020-3414d57e47da/go.mod h1:SqUrOPUnsFjfmXRMNPybcSiG0BgUW2AuFH8PAnS2iTw=
|
||||||
github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY=
|
github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY=
|
||||||
github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto=
|
github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto=
|
||||||
github.com/gabriel-vasile/mimetype v1.4.10 h1:zyueNbySn/z8mJZHLt6IPw0KoZsiQNszIpU+bX4+ZK0=
|
github.com/gabriel-vasile/mimetype v1.4.9 h1:5k+WDwEsD9eTLL8Tz3L0VnmVh9QxGjRmjBvAG7U/oYY=
|
||||||
github.com/gabriel-vasile/mimetype v1.4.10/go.mod h1:d+9Oxyo1wTzWdyVUPMmXFvp4F9tea18J8ufA774AB3s=
|
github.com/gabriel-vasile/mimetype v1.4.9/go.mod h1:WnSQhFKJuBlRyLiKohA/2DtIlPFAbguNaG7QCHcyGok=
|
||||||
github.com/gin-contrib/sse v1.1.0 h1:n0w2GMuUpWDVp7qSpvze6fAu9iRxJY4Hmj6AmBOU05w=
|
github.com/gin-contrib/sse v1.1.0 h1:n0w2GMuUpWDVp7qSpvze6fAu9iRxJY4Hmj6AmBOU05w=
|
||||||
github.com/gin-contrib/sse v1.1.0/go.mod h1:hxRZ5gVpWMT7Z0B0gSNYqqsSCNIJMjzvm6fqCz9vjwM=
|
github.com/gin-contrib/sse v1.1.0/go.mod h1:hxRZ5gVpWMT7Z0B0gSNYqqsSCNIJMjzvm6fqCz9vjwM=
|
||||||
github.com/gin-gonic/gin v1.10.1 h1:T0ujvqyCSqRopADpgPgiTT63DUQVSfojyME59Ei63pQ=
|
github.com/gin-gonic/gin v1.10.1 h1:T0ujvqyCSqRopADpgPgiTT63DUQVSfojyME59Ei63pQ=
|
||||||
@ -90,8 +88,8 @@ github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c h1:+mdjkGKdHQG3305AYmd
|
|||||||
github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c/go.mod h1:7rwL4CYBLnjLxUqIJNnCWiEdr3bn6IUYi15bNlnbCCU=
|
github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c/go.mod h1:7rwL4CYBLnjLxUqIJNnCWiEdr3bn6IUYi15bNlnbCCU=
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/rogpeppe/go-internal v1.14.1 h1:UQB4HGPB6osV0SQTLymcB4TgvyWu6ZyliaW0tI/otEQ=
|
github.com/rogpeppe/go-internal v1.13.1 h1:KvO1DLK/DRN07sQ1LQKScxyZJuNnedQ5/wKSR38lUII=
|
||||||
github.com/rogpeppe/go-internal v1.14.1/go.mod h1:MaRKkUm5W0goXpeCfT7UZI6fk/L7L7so1lCWt35ZSgc=
|
github.com/rogpeppe/go-internal v1.13.1/go.mod h1:uMEvuHeurkdAXX61udpOXGD/AzZDWNMNyH2VO9fmH0o=
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
|
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
|
||||||
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
|
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
|
||||||
@ -99,8 +97,8 @@ github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UV
|
|||||||
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||||
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
||||||
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||||
github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U=
|
github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA=
|
||||||
github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U=
|
github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI=
|
github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI=
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08=
|
github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08=
|
||||||
github.com/ugorji/go/codec v1.3.0 h1:Qd2W2sQawAfG8XSvzwhBeoGq71zXOC/Q1E9y/wUcsUA=
|
github.com/ugorji/go/codec v1.3.0 h1:Qd2W2sQawAfG8XSvzwhBeoGq71zXOC/Q1E9y/wUcsUA=
|
||||||
@ -109,29 +107,29 @@ github.com/vmihailenco/msgpack/v5 v5.4.1 h1:cQriyiUvjTwOHg8QZaPihLWeRAAVoCpE00IU
|
|||||||
github.com/vmihailenco/msgpack/v5 v5.4.1/go.mod h1:GaZTsDaehaPpQVyxrf5mtQlH+pc21PIudVV/E3rRQok=
|
github.com/vmihailenco/msgpack/v5 v5.4.1/go.mod h1:GaZTsDaehaPpQVyxrf5mtQlH+pc21PIudVV/E3rRQok=
|
||||||
github.com/vmihailenco/tagparser/v2 v2.0.0 h1:y09buUbR+b5aycVFQs/g70pqKVZNBmxwAhO7/IwNM9g=
|
github.com/vmihailenco/tagparser/v2 v2.0.0 h1:y09buUbR+b5aycVFQs/g70pqKVZNBmxwAhO7/IwNM9g=
|
||||||
github.com/vmihailenco/tagparser/v2 v2.0.0/go.mod h1:Wri+At7QHww0WTrCBeu4J6bNtoV6mEfg5OIWRZA9qds=
|
github.com/vmihailenco/tagparser/v2 v2.0.0/go.mod h1:Wri+At7QHww0WTrCBeu4J6bNtoV6mEfg5OIWRZA9qds=
|
||||||
go.opentelemetry.io/auto/sdk v1.2.1 h1:jXsnJ4Lmnqd11kwkBV2LgLoFMZKizbCi5fNZ/ipaZ64=
|
go.opentelemetry.io/auto/sdk v1.1.0 h1:cH53jehLUN6UFLY71z+NDOiNJqDdPRaXzTel0sJySYA=
|
||||||
go.opentelemetry.io/auto/sdk v1.2.1/go.mod h1:KRTj+aOaElaLi+wW1kO/DZRXwkF4C5xPbEe3ZiIhN7Y=
|
go.opentelemetry.io/auto/sdk v1.1.0/go.mod h1:3wSPjt5PWp2RhlCcmmOial7AvC4DQqZb7a7wCow3W8A=
|
||||||
go.opentelemetry.io/otel v1.38.0 h1:RkfdswUDRimDg0m2Az18RKOsnI8UDzppJAtj01/Ymk8=
|
go.opentelemetry.io/otel v1.37.0 h1:9zhNfelUvx0KBfu/gb+ZgeAfAgtWrfHJZcAqFC228wQ=
|
||||||
go.opentelemetry.io/otel v1.38.0/go.mod h1:zcmtmQ1+YmQM9wrNsTGV/q/uyusom3P8RxwExxkZhjM=
|
go.opentelemetry.io/otel v1.37.0/go.mod h1:ehE/umFRLnuLa/vSccNq9oS1ErUlkkK71gMcN34UG8I=
|
||||||
go.opentelemetry.io/otel/metric v1.38.0 h1:Kl6lzIYGAh5M159u9NgiRkmoMKjvbsKtYRwgfrA6WpA=
|
go.opentelemetry.io/otel/metric v1.37.0 h1:mvwbQS5m0tbmqML4NqK+e3aDiO02vsf/WgbsdpcPoZE=
|
||||||
go.opentelemetry.io/otel/metric v1.38.0/go.mod h1:kB5n/QoRM8YwmUahxvI3bO34eVtQf2i4utNVLr9gEmI=
|
go.opentelemetry.io/otel/metric v1.37.0/go.mod h1:04wGrZurHYKOc+RKeye86GwKiTb9FKm1WHtO+4EVr2E=
|
||||||
go.opentelemetry.io/otel/trace v1.38.0 h1:Fxk5bKrDZJUH+AMyyIXGcFAPah0oRcT+LuNtJrmcNLE=
|
go.opentelemetry.io/otel/trace v1.37.0 h1:HLdcFNbRQBE2imdSEgm/kwqmQj1Or1l/7bW6mxVK7z4=
|
||||||
go.opentelemetry.io/otel/trace v1.38.0/go.mod h1:j1P9ivuFsTceSWe1oY+EeW3sc+Pp42sO++GHkg4wwhs=
|
go.opentelemetry.io/otel/trace v1.37.0/go.mod h1:TlgrlQ+PtQO5XFerSPUYG0JSgGyryXewPGyayAWSBS0=
|
||||||
golang.org/x/arch v0.21.0 h1:iTC9o7+wP6cPWpDWkivCvQFGAHDQ59SrSxsLPcnkArw=
|
golang.org/x/arch v0.20.0 h1:dx1zTU0MAE98U+TQ8BLl7XsJbgze2WnNKF/8tGp/Q6c=
|
||||||
golang.org/x/arch v0.21.0/go.mod h1:dNHoOeKiyja7GTvF9NJS1l3Z2yntpQNzgrjh1cU103A=
|
golang.org/x/arch v0.20.0/go.mod h1:bdwinDaKcfZUGpH09BB7ZmOfhalA8lQdzl62l8gGWsk=
|
||||||
golang.org/x/crypto v0.42.0 h1:chiH31gIWm57EkTXpwnqf8qeuMUi0yekh6mT2AvFlqI=
|
golang.org/x/crypto v0.41.0 h1:WKYxWedPGCTVVl5+WHSSrOBT0O8lx32+zxmHxijgXp4=
|
||||||
golang.org/x/crypto v0.42.0/go.mod h1:4+rDnOTJhQCx2q7/j6rAN5XDw8kPjeaXEUR2eL94ix8=
|
golang.org/x/crypto v0.41.0/go.mod h1:pO5AFd7FA68rFak7rOAGVuygIISepHftHnr8dr6+sUc=
|
||||||
golang.org/x/exp v0.0.0-20250911091902-df9299821621 h1:2id6c1/gto0kaHYyrixvknJ8tUK/Qs5IsmBtrc+FtgU=
|
golang.org/x/exp v0.0.0-20250819193227-8b4c13bb791b h1:DXr+pvt3nC887026GRP39Ej11UATqWDmWuS99x26cD0=
|
||||||
golang.org/x/exp v0.0.0-20250911091902-df9299821621/go.mod h1:TwQYMMnGpvZyc+JpB/UAuTNIsVJifOlSkrZkhcvpVUk=
|
golang.org/x/exp v0.0.0-20250819193227-8b4c13bb791b/go.mod h1:4QTo5u+SEIbbKW1RacMZq1YEfOBqeXa19JeshGi+zc4=
|
||||||
golang.org/x/net v0.44.0 h1:evd8IRDyfNBMBTTY5XRF1vaZlD+EmWx6x8PkhR04H/I=
|
golang.org/x/net v0.43.0 h1:lat02VYK2j4aLzMzecihNvTlJNQUq316m2Mr9rnM6YE=
|
||||||
golang.org/x/net v0.44.0/go.mod h1:ECOoLqd5U3Lhyeyo/QDCEVQ4sNgYsqvCZ722XogGieY=
|
golang.org/x/net v0.43.0/go.mod h1:vhO1fvI4dGsIjh73sWfUVjj3N7CA9WkKJNQm2svM6Jg=
|
||||||
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.36.0 h1:KVRy2GtZBrk1cBYA7MKu5bEZFxQk4NIDV6RLVcC8o0k=
|
golang.org/x/sys v0.35.0 h1:vz1N37gP5bs89s7He8XuIYXpyY0+QlsKmzipCbUtyxI=
|
||||||
golang.org/x/sys v0.36.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
|
golang.org/x/sys v0.35.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k=
|
||||||
golang.org/x/text v0.29.0 h1:1neNs90w9YzJ9BocxfsQNHKuAT4pkghyXc4nhZ6sJvk=
|
golang.org/x/text v0.28.0 h1:rhazDwis8INMIwQ4tpjLDzUhx6RlXqZNPEM0huQojng=
|
||||||
golang.org/x/text v0.29.0/go.mod h1:7MhJOA9CD2qZyOKYazxdYMF85OwPdEr9jTtBpO7ydH4=
|
golang.org/x/text v0.28.0/go.mod h1:U8nCwOR8jO/marOQ0QbDiOngZVEBB7MAiitBuMjXiNU=
|
||||||
google.golang.org/protobuf v1.36.9 h1:w2gp2mA27hUeUzj9Ex9FBjsBm40zfaDtEWow293U7Iw=
|
google.golang.org/protobuf v1.36.7 h1:IgrO7UwFQGJdRNXH/sQux4R1Dj1WAKcLElzeeRaXV2A=
|
||||||
google.golang.org/protobuf v1.36.9/go.mod h1:fuxRtAxBytpl4zzqUh6/eyUujkJdNiuEkXntxiD/uRU=
|
google.golang.org/protobuf v1.36.7/go.mod h1:jduwjTPXsFjZGTmRluh+L6NjiWu7pchiJ2/5YcXBHnY=
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
|
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
|
||||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
|
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
|
||||||
|
@ -1,13 +1,9 @@
|
|||||||
package badgerdatastore
|
package badgerdatastore
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/dgraph-io/badger/v4"
|
"github.com/dgraph-io/badger/v4"
|
||||||
"github.com/pikami/cosmium/internal/datastore"
|
|
||||||
"github.com/pikami/cosmium/internal/logger"
|
"github.com/pikami/cosmium/internal/logger"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -17,7 +13,6 @@ type BadgerDataStore struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type BadgerDataStoreOptions struct {
|
type BadgerDataStoreOptions struct {
|
||||||
InitialDataFilePath string
|
|
||||||
PersistDataFilePath string
|
PersistDataFilePath string
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -40,8 +35,6 @@ func NewBadgerDataStore(options BadgerDataStoreOptions) *BadgerDataStore {
|
|||||||
gcTicker: gcTicker,
|
gcTicker: gcTicker,
|
||||||
}
|
}
|
||||||
|
|
||||||
ds.initializeDataStore(options.InitialDataFilePath)
|
|
||||||
|
|
||||||
go ds.runGarbageCollector()
|
go ds.runGarbageCollector()
|
||||||
|
|
||||||
return ds
|
return ds
|
||||||
@ -71,53 +64,3 @@ func (r *BadgerDataStore) runGarbageCollector() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *BadgerDataStore) initializeDataStore(initialDataFilePath string) {
|
|
||||||
if initialDataFilePath == "" {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
stat, err := os.Stat(initialDataFilePath)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if stat.IsDir() {
|
|
||||||
logger.ErrorLn("Argument '-Persist' must be a path to file, not a directory.")
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
jsonData, err := os.ReadFile(initialDataFilePath)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Error reading state JSON file: %v", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
var state datastore.InitialDataModel
|
|
||||||
if err := json.Unmarshal([]byte(jsonData), &state); err != nil {
|
|
||||||
log.Fatalf("Error parsing state JSON file: %v", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
for dbName, dbModel := range state.Databases {
|
|
||||||
r.CreateDatabase(dbModel)
|
|
||||||
for colName, colModel := range state.Collections[dbName] {
|
|
||||||
r.CreateCollection(dbName, colModel)
|
|
||||||
for _, docModel := range state.Documents[dbName][colName] {
|
|
||||||
r.CreateDocument(dbName, colName, docModel)
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, triggerModel := range state.Triggers[dbName][colName] {
|
|
||||||
r.CreateTrigger(dbName, colName, triggerModel)
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, spModel := range state.StoredProcedures[dbName][colName] {
|
|
||||||
r.CreateStoredProcedure(dbName, colName, spModel)
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, udfModel := range state.UserDefinedFunctions[dbName][colName] {
|
|
||||||
r.CreateUserDefinedFunction(dbName, colName, udfModel)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -22,8 +22,7 @@ func (r *BadgerDataStore) GetAllCollections(databaseId string) ([]datastore.Coll
|
|||||||
return nil, datastore.StatusNotFound
|
return nil, datastore.StatusNotFound
|
||||||
}
|
}
|
||||||
|
|
||||||
prefix := generateKey(resourceid.ResourceTypeCollection, databaseId, "", "") + "/"
|
colls, status := listByPrefix[datastore.Collection](r.db, generateKey(resourceid.ResourceTypeCollection, databaseId, "", ""))
|
||||||
colls, status := listByPrefix[datastore.Collection](r.db, prefix)
|
|
||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
return colls, datastore.StatusOk
|
return colls, datastore.StatusOk
|
||||||
}
|
}
|
||||||
@ -50,10 +49,10 @@ func (r *BadgerDataStore) DeleteCollection(databaseId string, collectionId strin
|
|||||||
defer txn.Discard()
|
defer txn.Discard()
|
||||||
|
|
||||||
prefixes := []string{
|
prefixes := []string{
|
||||||
generateKey(resourceid.ResourceTypeDocument, databaseId, collectionId, "") + "/",
|
generateKey(resourceid.ResourceTypeDocument, databaseId, collectionId, ""),
|
||||||
generateKey(resourceid.ResourceTypeTrigger, databaseId, collectionId, "") + "/",
|
generateKey(resourceid.ResourceTypeTrigger, databaseId, collectionId, ""),
|
||||||
generateKey(resourceid.ResourceTypeStoredProcedure, databaseId, collectionId, "") + "/",
|
generateKey(resourceid.ResourceTypeStoredProcedure, databaseId, collectionId, ""),
|
||||||
generateKey(resourceid.ResourceTypeUserDefinedFunction, databaseId, collectionId, "") + "/",
|
generateKey(resourceid.ResourceTypeUserDefinedFunction, databaseId, collectionId, ""),
|
||||||
collectionKey,
|
collectionKey,
|
||||||
}
|
}
|
||||||
for _, prefix := range prefixes {
|
for _, prefix := range prefixes {
|
||||||
|
@ -38,11 +38,11 @@ func (r *BadgerDataStore) DeleteDatabase(id string) datastore.DataStoreStatus {
|
|||||||
defer txn.Discard()
|
defer txn.Discard()
|
||||||
|
|
||||||
prefixes := []string{
|
prefixes := []string{
|
||||||
generateKey(resourceid.ResourceTypeCollection, id, "", "") + "/",
|
generateKey(resourceid.ResourceTypeCollection, id, "", ""),
|
||||||
generateKey(resourceid.ResourceTypeDocument, id, "", "") + "/",
|
generateKey(resourceid.ResourceTypeDocument, id, "", ""),
|
||||||
generateKey(resourceid.ResourceTypeTrigger, id, "", "") + "/",
|
generateKey(resourceid.ResourceTypeTrigger, id, "", ""),
|
||||||
generateKey(resourceid.ResourceTypeStoredProcedure, id, "", "") + "/",
|
generateKey(resourceid.ResourceTypeStoredProcedure, id, "", ""),
|
||||||
generateKey(resourceid.ResourceTypeUserDefinedFunction, id, "", "") + "/",
|
generateKey(resourceid.ResourceTypeUserDefinedFunction, id, "", ""),
|
||||||
databaseKey,
|
databaseKey,
|
||||||
}
|
}
|
||||||
for _, prefix := range prefixes {
|
for _, prefix := range prefixes {
|
||||||
|
@ -24,8 +24,7 @@ func (r *BadgerDataStore) GetAllDocuments(databaseId string, collectionId string
|
|||||||
return nil, datastore.StatusNotFound
|
return nil, datastore.StatusNotFound
|
||||||
}
|
}
|
||||||
|
|
||||||
prefix := generateKey(resourceid.ResourceTypeDocument, databaseId, collectionId, "") + "/"
|
docs, status := listByPrefix[datastore.Document](r.db, generateKey(resourceid.ResourceTypeDocument, databaseId, collectionId, ""))
|
||||||
docs, status := listByPrefix[datastore.Document](r.db, prefix)
|
|
||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
return docs, datastore.StatusOk
|
return docs, datastore.StatusOk
|
||||||
}
|
}
|
||||||
@ -46,8 +45,7 @@ func (r *BadgerDataStore) GetDocumentIterator(databaseId string, collectionId st
|
|||||||
return nil, datastore.StatusNotFound
|
return nil, datastore.StatusNotFound
|
||||||
}
|
}
|
||||||
|
|
||||||
prefix := generateKey(resourceid.ResourceTypeDocument, databaseId, collectionId, "") + "/"
|
iter := NewBadgerDocumentIterator(txn, generateKey(resourceid.ResourceTypeDocument, databaseId, collectionId, ""))
|
||||||
iter := NewBadgerDocumentIterator(txn, prefix)
|
|
||||||
return iter, datastore.StatusOk
|
return iter, datastore.StatusOk
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,8 +24,7 @@ func (r *BadgerDataStore) GetAllStoredProcedures(databaseId string, collectionId
|
|||||||
return nil, datastore.StatusNotFound
|
return nil, datastore.StatusNotFound
|
||||||
}
|
}
|
||||||
|
|
||||||
prefix := generateKey(resourceid.ResourceTypeStoredProcedure, databaseId, collectionId, "") + "/"
|
storedProcedures, status := listByPrefix[datastore.StoredProcedure](r.db, generateKey(resourceid.ResourceTypeStoredProcedure, databaseId, collectionId, ""))
|
||||||
storedProcedures, status := listByPrefix[datastore.StoredProcedure](r.db, prefix)
|
|
||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
return storedProcedures, datastore.StatusOk
|
return storedProcedures, datastore.StatusOk
|
||||||
}
|
}
|
||||||
|
@ -24,8 +24,7 @@ func (r *BadgerDataStore) GetAllTriggers(databaseId string, collectionId string)
|
|||||||
return nil, datastore.StatusNotFound
|
return nil, datastore.StatusNotFound
|
||||||
}
|
}
|
||||||
|
|
||||||
prefix := generateKey(resourceid.ResourceTypeTrigger, databaseId, collectionId, "") + "/"
|
triggers, status := listByPrefix[datastore.Trigger](r.db, generateKey(resourceid.ResourceTypeTrigger, databaseId, collectionId, ""))
|
||||||
triggers, status := listByPrefix[datastore.Trigger](r.db, prefix)
|
|
||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
return triggers, datastore.StatusOk
|
return triggers, datastore.StatusOk
|
||||||
}
|
}
|
||||||
|
@ -24,8 +24,7 @@ func (r *BadgerDataStore) GetAllUserDefinedFunctions(databaseId string, collecti
|
|||||||
return nil, datastore.StatusNotFound
|
return nil, datastore.StatusNotFound
|
||||||
}
|
}
|
||||||
|
|
||||||
prefix := generateKey(resourceid.ResourceTypeUserDefinedFunction, databaseId, collectionId, "") + "/"
|
udfs, status := listByPrefix[datastore.UserDefinedFunction](r.db, generateKey(resourceid.ResourceTypeUserDefinedFunction, databaseId, collectionId, ""))
|
||||||
udfs, status := listByPrefix[datastore.UserDefinedFunction](r.db, prefix)
|
|
||||||
if status == datastore.StatusOk {
|
if status == datastore.StatusOk {
|
||||||
return udfs, datastore.StatusOk
|
return udfs, datastore.StatusOk
|
||||||
}
|
}
|
||||||
|
@ -1,21 +0,0 @@
|
|||||||
package datastore
|
|
||||||
|
|
||||||
type InitialDataModel struct {
|
|
||||||
// Map databaseId -> Database
|
|
||||||
Databases map[string]Database `json:"databases"`
|
|
||||||
|
|
||||||
// Map databaseId -> collectionId -> Collection
|
|
||||||
Collections map[string]map[string]Collection `json:"collections"`
|
|
||||||
|
|
||||||
// Map databaseId -> collectionId -> documentId -> Documents
|
|
||||||
Documents map[string]map[string]map[string]Document `json:"documents"`
|
|
||||||
|
|
||||||
// Map databaseId -> collectionId -> triggerId -> Trigger
|
|
||||||
Triggers map[string]map[string]map[string]Trigger `json:"triggers"`
|
|
||||||
|
|
||||||
// Map databaseId -> collectionId -> spId -> StoredProcedure
|
|
||||||
StoredProcedures map[string]map[string]map[string]StoredProcedure `json:"sprocs"`
|
|
||||||
|
|
||||||
// Map databaseId -> collectionId -> udfId -> UserDefinedFunction
|
|
||||||
UserDefinedFunctions map[string]map[string]map[string]UserDefinedFunction `json:"udfs"`
|
|
||||||
}
|
|
@ -38,7 +38,6 @@ func CreateServerInstance(serverName *C.char, configurationJSON *C.char) int {
|
|||||||
switch configuration.DataStore {
|
switch configuration.DataStore {
|
||||||
case config.DataStoreBadger:
|
case config.DataStoreBadger:
|
||||||
dataStore = badgerdatastore.NewBadgerDataStore(badgerdatastore.BadgerDataStoreOptions{
|
dataStore = badgerdatastore.NewBadgerDataStore(badgerdatastore.BadgerDataStoreOptions{
|
||||||
InitialDataFilePath: configuration.InitialDataFilePath,
|
|
||||||
PersistDataFilePath: configuration.PersistDataFilePath,
|
PersistDataFilePath: configuration.PersistDataFilePath,
|
||||||
})
|
})
|
||||||
default:
|
default:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user