From 4e8018c6aca14ba830c6d806a4f71719a3bac826 Mon Sep 17 00:00:00 2001 From: Richard Weerasinghe Date: Sat, 24 Aug 2024 17:07:24 +1200 Subject: [PATCH] add support for projects endpoint --- openai/projects.go | 90 +++++++++++++++++++++++++++++++++++++++++ openai/projects_test.go | 70 ++++++++++++++++++++++++++++++++ 2 files changed, 160 insertions(+) create mode 100644 openai/projects.go create mode 100644 openai/projects_test.go diff --git a/openai/projects.go b/openai/projects.go new file mode 100644 index 0000000..675f352 --- /dev/null +++ b/openai/projects.go @@ -0,0 +1,90 @@ +package openai + +import ( + "fmt" + "net/url" +) + +const ProjectsEndpointPath = "/organization/projects" + +// ProjectsEndpoint - OpenAI Projects Endpoint +// +// List and describe the projects available. +// You can refer to the [Projects]: https://platform.openai.com/docs/api-reference/projects documentation. +type ProjectsEndpoint struct { + *organizationEndpoint +} + +// Projects - Projects Endpoint +func (c *Client) Projects() *ProjectsEndpoint { + return &ProjectsEndpoint{newOrganizationEndpoint(c, ProjectsEndpointPath)} +} + +// Project - OpenAPI Project. +type Project struct { + ID string `json:"id"` + Object string `json:"object"` + Name string `json:"name"` + CreatedAt int64 `json:"created_at"` + ArchivedAt int64 `json:"archived_at"` + Status string `json:"status"` +} + +type Projects struct { + Object string `json:"object"` + Data []Project `json:"data"` +} + +type ProjectRequest struct { + // The name of the assistant. The maximum length is 256 characters. + Name *string `json:"name"` +} + +// Lists the currently available projects, +// and provides basic information about each one. +// +// [OpenAI Documentation]: https://platform.openai.com/docs/api-reference/projects/list +func (e *ProjectsEndpoint) ListProjects() ([]Project, error) { + var projects Projects + err := e.do(e, "GET", "", nil, nil, &projects) + // TODO: This needs to move somewhere central + if err == nil && projects.Object != "list" { + err = fmt.Errorf("expected 'list' object type, got %s", projects.Object) + } + return projects.Data, err +} + +// Create a project. +// [OpenAI Documentation]: https://platform.openai.com/docs/api-reference/projects/create +func (e *ProjectsEndpoint) CreateProject(req *ProjectRequest) (*Project, error) { + var project Project + err := e.do(e, "POST", "", req, nil, &project) + return &project, err +} + +// Retrieves a project instance, +// providing basic information about the project. +// +// [OpenAI Documentation]: https://platform.openai.com/docs/api-reference/projects/retrieve +func (e *ProjectsEndpoint) RetrieveProject(id string) (*Project, error) { + var project Project + err := e.do(e, "GET", id, nil, nil, &project) + return &project, err +} + +// Modifies a project instance, +// +// [OpenAI Documentation]: https://platform.openai.com/docs/api-reference/projects/modify +func (e *ProjectsEndpoint) ModifyProject(id string, req ProjectRequest) (*Project, error) { + var project Project + err := e.do(e, "POST", id, req, nil, &project) + return &project, err +} + +// Archive a project. +// [OpenAI Documentation]: https://platform.openai.com/docs/api-reference/projects/archive +func (e *ProjectsEndpoint) ArchiveProject(id string) (*Project, error) { + var project Project + err := e.do(e, "POST", url.QueryEscape(id)+"/archive", nil, nil, &project) + return &project, err +} diff --git a/openai/projects_test.go b/openai/projects_test.go new file mode 100644 index 0000000..d7d3782 --- /dev/null +++ b/openai/projects_test.go @@ -0,0 +1,70 @@ +package openai_test + +import ( + "encoding/json" + "fmt" + "net/http" + "testing" + + "github.com/skyscrapr/openai-sdk-go/openai" + "github.com/skyscrapr/openai-sdk-go/openai/test" +) + +// TestListProjects Tests the Projects endpoint of the API using the mocked server. +func TestListProjects(t *testing.T) { + ts := openai_test.NewTestServer() + ts.RegisterHandler("/v1/organizations/projects", func(w http.ResponseWriter, _ *http.Request) { + resBytes, _ := json.Marshal(openai.Projects{Object: "list", Data: nil}) + fmt.Fprintln(w, string(resBytes)) + }) + ts.HTTPServer.Start() + defer ts.HTTPServer.Close() + + client := openai_test.NewTestClient(ts) + _, err := client.Projects().ListProjects() + t.Helper() + if err != nil { + t.Error(err, "TestListProjects error") + } +} + +func TestListProjectsInvalidObject(t *testing.T) { + expectedError := "expected 'list' object type, got project" + + ts := openai_test.NewTestServer() + ts.RegisterHandler("/v1/organizations/projects", func(w http.ResponseWriter, _ *http.Request) { + resBytes, _ := json.Marshal(openai.Projects{Object: "project", Data: nil}) + fmt.Fprintln(w, string(resBytes)) + }) + ts.HTTPServer.Start() + defer ts.HTTPServer.Close() + + client := openai_test.NewTestClient(ts) + _, err := client.Projects().ListProjects() + t.Helper() + if err != nil && err.Error() != expectedError { + t.Errorf("Unexpected error: %v , expected: %s", err, expectedError) + t.Fail() + } +} + +func TestRetrieveProject(t *testing.T) { + testProjectID := "testProjectID" + ts := openai_test.NewTestServer() + ts.RegisterHandler("/v1/organizations/projects/testProjectID", func(w http.ResponseWriter, _ *http.Request) { + resBytes, _ := json.Marshal(openai.Project{Object: "project", ID: testProjectID}) + fmt.Fprintln(w, string(resBytes)) + }) + ts.HTTPServer.Start() + defer ts.HTTPServer.Close() + + client := openai_test.NewTestClient(ts) + project, err := client.Projects().RetrieveProject(testProjectID) + t.Helper() + if err != nil { + t.Error(err, "GetProject error") + } + if project.ID != testProjectID { + t.Errorf("ProjectsEndpoint GetProject Project ID mismatch. Got %s. Expected %s", testProjectID, project.ID) + } +}