functions.py 6.84 KB
Newer Older
1
from pydantic import BaseModel, ConfigDict
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
2
3
4
from typing import List, Union, Optional
import time
import logging
5
6
7

from sqlalchemy import Column, String, Text, BigInteger, Boolean

8
from apps.webui.internal.db import JSONField, Base, Session
Timothy J. Baek's avatar
Timothy J. Baek committed
9
from apps.webui.models.users import Users
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
10
11

import json
12
13
import copy

Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
14
15
16
17
18
19
20
21
22
23
24

from config import SRC_LOG_LEVELS

log = logging.getLogger(__name__)
log.setLevel(SRC_LOG_LEVELS["MODELS"])

####################
# Functions DB Schema
####################


25
26
class Function(Base):
    __tablename__ = "function"
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
27

28
29
30
31
32
33
34
35
36
37
    id = Column(String, primary_key=True)
    user_id = Column(String)
    name = Column(Text)
    type = Column(Text)
    content = Column(Text)
    meta = Column(JSONField)
    valves = Column(JSONField)
    is_active = Column(Boolean)
    updated_at = Column(BigInteger)
    created_at = Column(BigInteger)
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
38
39
40
41


class FunctionMeta(BaseModel):
    description: Optional[str] = None
42
    manifest: Optional[dict] = {}
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
43
44
45
46
47
48
49
50
51


class FunctionModel(BaseModel):
    id: str
    user_id: str
    name: str
    type: str
    content: str
    meta: FunctionMeta
52
    is_active: bool = False
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
53
54
55
    updated_at: int  # timestamp in epoch
    created_at: int  # timestamp in epoch

56
57
    model_config = ConfigDict(from_attributes=True)

Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
58
59
60
61
62
63
64
65
66

####################
# Forms
####################


class FunctionResponse(BaseModel):
    id: str
    user_id: str
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
67
    type: str
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
68
69
    name: str
    meta: FunctionMeta
Timothy J. Baek's avatar
Timothy J. Baek committed
70
    is_active: bool
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
71
72
73
74
75
76
77
78
79
80
81
    updated_at: int  # timestamp in epoch
    created_at: int  # timestamp in epoch


class FunctionForm(BaseModel):
    id: str
    name: str
    content: str
    meta: FunctionMeta


Timothy J. Baek's avatar
Timothy J. Baek committed
82
83
84
85
class FunctionValves(BaseModel):
    valves: Optional[dict] = None


Timothy J. Baek's avatar
Timothy J. Baek committed
86
class FunctionsTable:
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
87
88

    def insert_new_function(
89
        self, user_id: str, type: str, form_data: FunctionForm
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
90
91
92
93
94
    ) -> Optional[FunctionModel]:
        function = FunctionModel(
            **{
                **form_data.model_dump(),
                "user_id": user_id,
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
95
                "type": type,
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
96
97
98
99
100
101
                "updated_at": int(time.time()),
                "created_at": int(time.time()),
            }
        )

        try:
102
103
104
105
106
107
108
109
            result = Function(**function.model_dump())
            Session.add(result)
            Session.commit()
            Session.refresh(result)
            if result:
                return FunctionModel.model_validate(result)
            else:
                return None
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
110
111
112
113
        except Exception as e:
            print(f"Error creating tool: {e}")
            return None

114
    def get_function_by_id(self, id: str) -> Optional[FunctionModel]:
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
115
        try:
116
117
            function = Session.get(Function, id)
            return FunctionModel.model_validate(function)
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
118
119
120
        except:
            return None

121
122
    def get_functions(self, active_only=False) -> List[FunctionModel]:
        if active_only:
123
124
125
126
            return [
                FunctionModel.model_validate(function)
                for function in Session.query(Function).filter_by(is_active=True).all()
            ]
127
        else:
128
129
130
131
            return [
                FunctionModel.model_validate(function)
                for function in Session.query(Function).all()
            ]
132
133
134
135
136

    def get_functions_by_type(
        self, type: str, active_only=False
    ) -> List[FunctionModel]:
        if active_only:
137
138
139
140
141
142
            return [
                FunctionModel.model_validate(function)
                for function in Session.query(Function)
                .filter_by(type=type, is_active=True)
                .all()
            ]
143
        else:
144
145
146
147
            return [
                FunctionModel.model_validate(function)
                for function in Session.query(Function).filter_by(type=type).all()
            ]
148

Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
149
    def get_function_valves_by_id(self, id: str) -> Optional[dict]:
150
        try:
151
152
            function = Session.get(Function, id)
            return function.valves if function.valves else {}
153
154
155
156
157
158
159
160
        except Exception as e:
            print(f"An error occurred: {e}")
            return None

    def update_function_valves_by_id(
        self, id: str, valves: dict
    ) -> Optional[FunctionValves]:
        try:
161
162
163
164
165
166
            function = Session.get(Function, id)
            function.valves = valves
            function.updated_at = int(time.time())
            Session.commit()
            Session.refresh(function)
            return self.get_function_by_id(id)
167
168
        except:
            return None
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
169

Timothy J. Baek's avatar
Timothy J. Baek committed
170
171
172
173
174
    def get_user_valves_by_id_and_user_id(
        self, id: str, user_id: str
    ) -> Optional[dict]:
        try:
            user = Users.get_user_by_id(user_id)
175
            user_settings = user.settings.model_dump()
Timothy J. Baek's avatar
Timothy J. Baek committed
176
177

            # Check if user has "functions" and "valves" settings
178
179
180
181
            if "functions" not in user_settings:
                user_settings["functions"] = {}
            if "valves" not in user_settings["functions"]:
                user_settings["functions"]["valves"] = {}
Timothy J. Baek's avatar
Timothy J. Baek committed
182

183
            return user_settings["functions"]["valves"].get(id, {})
Timothy J. Baek's avatar
Timothy J. Baek committed
184
185
186
187
188
189
190
191
192
        except Exception as e:
            print(f"An error occurred: {e}")
            return None

    def update_user_valves_by_id_and_user_id(
        self, id: str, user_id: str, valves: dict
    ) -> Optional[dict]:
        try:
            user = Users.get_user_by_id(user_id)
193
            user_settings = user.settings.model_dump()
Timothy J. Baek's avatar
Timothy J. Baek committed
194
195

            # Check if user has "functions" and "valves" settings
196
197
198
199
            if "functions" not in user_settings:
                user_settings["functions"] = {}
            if "valves" not in user_settings["functions"]:
                user_settings["functions"]["valves"] = {}
Timothy J. Baek's avatar
Timothy J. Baek committed
200

201
            user_settings["functions"]["valves"][id] = valves
Timothy J. Baek's avatar
Timothy J. Baek committed
202
203

            # Update the user settings in the database
204
            query = Users.update_user_by_id(user_id, {"settings": user_settings})
Timothy J. Baek's avatar
Timothy J. Baek committed
205
206
            query.execute()

207
            return user_settings["functions"]["valves"][id]
Timothy J. Baek's avatar
Timothy J. Baek committed
208
209
210
211
        except Exception as e:
            print(f"An error occurred: {e}")
            return None

Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
212
213
    def update_function_by_id(self, id: str, updated: dict) -> Optional[FunctionModel]:
        try:
214
215
216
217
218
219
220
221
            Session.query(Function).filter_by(id=id).update(
                {
                    **updated,
                    "updated_at": int(time.time()),
                }
            )
            Session.commit()
            return self.get_function_by_id(id)
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
222
223
224
        except:
            return None

Timothy J. Baek's avatar
Timothy J. Baek committed
225
226
    def deactivate_all_functions(self) -> Optional[bool]:
        try:
227
228
229
230
231
232
233
            Session.query(Function).update(
                {
                    "is_active": False,
                    "updated_at": int(time.time()),
                }
            )
            Session.commit()
Timothy J. Baek's avatar
Timothy J. Baek committed
234
235
236
237
            return True
        except:
            return None

238
    def delete_function_by_id(self, id: str) -> bool:
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
239
        try:
240
            Session.query(Function).filter_by(id=id).delete()
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
241
242
243
244
245
            return True
        except:
            return False


246
Functions = FunctionsTable()