Skip to content

Latest commit

 

History

History
210 lines (165 loc) · 5.29 KB

README.md

File metadata and controls

210 lines (165 loc) · 5.29 KB

goutils

some useful go utils

Usage

import "github.com/lyricat/goutils"

packages

Qdrant

	qd := qdrant.New(qdrant.Config{
		Addr:	"Qdrant.Addr",
		APIKey: "Qdrant.APIKey",
	})
	if _, err := qd.Check(); err != nil {
		slog.Error("[index] qdrant check failed", "error", err)
	}

Insert

import (
  lqdrant "github.com/lyricat/goutils/qdrant"
)

func UpsertToQdrant(ctx context.Context) error {
	text := "The grapes are innocent, please do not ban the planting of grapes because eating grapes can lead to death."
	params := lqdrant.UpsertPointsParams{}
	params.CollectionName = "QdrantCollectionName"
	input := []string{text}
	vec, err := client.CreateEmbeddingAzureOpenAI(ctx, input)
	if err != nil {
		return err
	}
	now := time.Now()
	expiry := now.Add(24 * time.Hour)
	params.PointUUID = uuid.MD5(...)
	params.Vector = vec
	params.Payload = map[string]qdrant.UpsertPointPayloadItem{
		"text": {Type: "text", Value: text},
	}
	if err := qd.UpsertPoints(ctx, params); err != nil {
		return err
	}

	return nil
}

Delete

func (tc *TweetCache) DeleteExpiredFromQdrant(ctx context.Context) error {
	currentTime := time.Now()
	unixNow := float64(currentTime.Unix())
	filter := &qdrant.Filter{
		Must: []*qdrant.Condition{
			{
				ConditionOneOf: &qdrant.Condition_Field{
					Field: &qdrant.FieldCondition{
						// ...,
					},
				},
			},
		},
	}

	params := lqdrant.DeletePointsParams{}
	params.CollectionName = "QdrantCollectionName"
	params.Selector = qdrant.NewPointsSelectorFilter(filter)

	return tc.qd.DeletePoints(ctx, params)
}

AI

The ai package provides useful functions for AI related tasks, and it supports multiple AI providers:

  • OpenAI
  • OpenAI on Azure
  • Claude AI on AWS Bedrock
import "github.com/lyricat/goutils/ai"

	clientAzure := ai.New(ai.Config{
		AzureOpenAIApiKey:                "...",
		AzureOpenAIEndpoint:              "https://abc.openai.azure.com/",
		AzureOpenAIGptDeploymentID:       "gpt-4o-mini-0718",
		AzureOpenAIEmbeddingDeploymentID: "text-embedding-3-small-latest",
		Provider:                         "azure",

		Debug: true,
	})

	clientOpenAI := ai.New(ai.Config{
		OpenAIGptModel:       "gpt-4o-mini-2024-07-18",
		OpenAIEmbeddingModel: "text-embedding-3-small",

		OpenAIApiKey: "sk-...",
		Provider:     "openai",
		Debug:        true,
	})

	clientBedrock := ai.New(ai.Config{
		AwsKey:    "...",
		AwsSecret: "...",

		AwsBedrockModelArn:          "arn:aws:...",
		AwsBedrockEmbeddingModelArn: "arn:aws:...",
		Provider:                    "bedrock",
		Debug:                       true,
	})

One time API call for JSON response

Here is an example of how to use the OneTimeRequest function to check if a given content is spam:

	inst := fmt.Sprintf(`Is it a spam ?:

	%s

	Output the probability as json, from 0 to 1.
	example:
	{ "probability": float_value, "is_spam": true_or_false}
	`, content)

	ret, err := client.OneTimeRequestJson(ctx, inst)
	if err != nil {
		slog.Error("failed to send one time request", "error", err)
	}

	spam, ok := ret["is_spam"].(bool)
	if !ok {
		slog.Error("no `is_spam` key in response")
	}

Multiple API calls for TEXT response

MultipleSteps function is used to send multiple requests to the AI provider, and the response of each request is used as the input of the next request.

Here is an example of how to use the MultipleSteps function to translate a given content to a specific language. In this case, we use text as the format of the response, because we want to get the translated text directly.

	content := "The grapes are innocent, please do not ban the planting of grapes because eating grapes can lead to death."
	lang := "Japanese"
	// a timeout of 1 minutes context
	ctx, cancel := context.WithTimeout(ctx, 1*time.Minute)
	defer cancel()

	inst := fmt.Sprintf(`You are an expert linguist, specializing in %s language.
Provide the %s translation for above text by ensuring:

1. accuracy (by correcting errors of addition, mistranslation, omission, or untranslated text),
2. fluency (by applying %s grammar, spelling and punctuation rules and ensuring there are no unnecessary repetitions),
3. style (by ensuring the translations always following the style of the original source text)
4. terminology (by ensuring terminology use is consistent and reflects the source text domain; and by only ensuring you use equivalent idioms of %s)
5. translation must be markdown format.

Do not provide any explanations or text apart from the translation result.
`, lang, lang, lang, lang)

	ret, err := client.MultipleSteps(ctx, ai.ChainParams{
		Format: "text",
		Steps: []ai.ChainParamsStep{
			{Input: content},
			{Instruction: inst},
		},
	})
	if err != nil {
		return "", err
	}

	slog.Info("translate result", "ret", ret.Text)

Get Text Embedding

	vector, err := a.client.GetEmbeddings(ctx, []string{content})
	if err != nil {
		slog.Error("failed to get embeddings", "error", err)
	}

then you may use the vector to search at qdrant:

import 	"github.com/lyricat/goutils/qdrant"

  // ...
	params := qdrant.SearchPointsParams{}
	params.CollectionName = "QdrantCollectionName"
	params.Vector = vector
	params.TopK = uint64(limit)
	searchResult, err := qd.SearchPointsWithFilter(ctx, params)
	if err != nil {
		slog.Error("failed to search points with filter", "error", err)
	}