Spaces:
Runtime error
Runtime error
jiangjiechen
commited on
Commit
·
8acb22e
1
Parent(s):
fbb1938
init app
Browse files- .gitignore +234 -0
- LICENSE +201 -0
- app.py +396 -0
- app_modules/overwrites.py +57 -0
- app_modules/presets.py +78 -0
- app_modules/utils.py +375 -0
- assets/Kelpy-Codos.js +76 -0
- assets/custom.css +191 -0
- assets/custom.js +1 -0
- assets/favicon.ico +0 -0
- assets/totopower-removebg.png +0 -0
- auction_workflow.py +293 -0
- data/bidders_demo.jsonl +4 -0
- data/items_demo.jsonl +26 -0
- requirements.txt +20 -0
- src/auctioneer_base.py +259 -0
- src/bidder_base.py +1031 -0
- src/human_bidder.py +137 -0
- src/item_base.py +50 -0
- src/prompt_base.py +349 -0
- utils.py +73 -0
.gitignore
ADDED
@@ -0,0 +1,234 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# Created by .ignore support plugin (hsz.mobi)
|
2 |
+
### macOS template
|
3 |
+
# General
|
4 |
+
.DS_Store
|
5 |
+
.AppleDouble
|
6 |
+
.LSOverride
|
7 |
+
|
8 |
+
# Icon must end with two \r
|
9 |
+
Icon
|
10 |
+
|
11 |
+
# Thumbnails
|
12 |
+
._*
|
13 |
+
|
14 |
+
# Files that might appear in the root of a volume
|
15 |
+
.DocumentRevisions-V100
|
16 |
+
.fseventsd
|
17 |
+
.Spotlight-V100
|
18 |
+
.TemporaryItems
|
19 |
+
.Trashes
|
20 |
+
.VolumeIcon.icns
|
21 |
+
.com.apple.timemachine.donotpresent
|
22 |
+
|
23 |
+
# Directories potentially created on remote AFP share
|
24 |
+
.AppleDB
|
25 |
+
.AppleDesktop
|
26 |
+
Network Trash Folder
|
27 |
+
Temporary Items
|
28 |
+
.apdisk
|
29 |
+
### Python template
|
30 |
+
# Byte-compiled / optimized / DLL files
|
31 |
+
__pycache__/
|
32 |
+
*.py[cod]
|
33 |
+
*$py.class
|
34 |
+
|
35 |
+
# C extensions
|
36 |
+
*.so
|
37 |
+
|
38 |
+
# Distribution / packaging
|
39 |
+
.Python
|
40 |
+
build/
|
41 |
+
develop-eggs/
|
42 |
+
dist/
|
43 |
+
downloads/
|
44 |
+
eggs/
|
45 |
+
.eggs/
|
46 |
+
lib/
|
47 |
+
lib64/
|
48 |
+
parts/
|
49 |
+
sdist/
|
50 |
+
var/
|
51 |
+
wheels/
|
52 |
+
*.egg-info/
|
53 |
+
.installed.cfg
|
54 |
+
*.egg
|
55 |
+
MANIFEST
|
56 |
+
|
57 |
+
# PyInstaller
|
58 |
+
# Usually these files are written by a python script from a template
|
59 |
+
# before PyInstaller builds the exe, so as to inject date/other infos into it.
|
60 |
+
*.manifest
|
61 |
+
*.spec
|
62 |
+
|
63 |
+
# Installer logs
|
64 |
+
pip-log.txt
|
65 |
+
pip-delete-this-directory.txt
|
66 |
+
|
67 |
+
# Unit test / coverage reports
|
68 |
+
htmlcov/
|
69 |
+
.tox/
|
70 |
+
.coverage
|
71 |
+
.coverage.*
|
72 |
+
.cache
|
73 |
+
nosetests.xml
|
74 |
+
coverage.xml
|
75 |
+
*.cover
|
76 |
+
.hypothesis/
|
77 |
+
.pytest_cache/
|
78 |
+
|
79 |
+
# Translations
|
80 |
+
*.mo
|
81 |
+
*.pot
|
82 |
+
|
83 |
+
# Django stuff:
|
84 |
+
*.log
|
85 |
+
local_settings.py
|
86 |
+
db.sqlite3
|
87 |
+
|
88 |
+
# Flask stuff:
|
89 |
+
instance/
|
90 |
+
.webassets-cache
|
91 |
+
|
92 |
+
# Scrapy stuff:
|
93 |
+
.scrapy
|
94 |
+
|
95 |
+
# Sphinx documentation
|
96 |
+
docs/_build/
|
97 |
+
|
98 |
+
# PyBuilder
|
99 |
+
target/
|
100 |
+
|
101 |
+
# Jupyter Notebook
|
102 |
+
.ipynb_checkpoints
|
103 |
+
|
104 |
+
# pyenv
|
105 |
+
.python-version
|
106 |
+
|
107 |
+
# celery beat schedule file
|
108 |
+
celerybeat-schedule
|
109 |
+
|
110 |
+
# SageMath parsed files
|
111 |
+
*.sage.py
|
112 |
+
|
113 |
+
# Environments
|
114 |
+
.env
|
115 |
+
.venv
|
116 |
+
env/
|
117 |
+
venv/
|
118 |
+
ENV/
|
119 |
+
env.bak/
|
120 |
+
venv.bak/
|
121 |
+
|
122 |
+
# Spyder project settings
|
123 |
+
.spyderproject
|
124 |
+
.spyproject
|
125 |
+
|
126 |
+
# Rope project settings
|
127 |
+
.ropeproject
|
128 |
+
|
129 |
+
# mkdocs documentation
|
130 |
+
/site
|
131 |
+
|
132 |
+
# mypy
|
133 |
+
.mypy_cache/
|
134 |
+
### JetBrains template
|
135 |
+
# Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio and WebStorm
|
136 |
+
# Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839
|
137 |
+
|
138 |
+
# User-specific stuff
|
139 |
+
.idea/**/workspace.xml
|
140 |
+
.idea/**/tasks.xml
|
141 |
+
.idea/**/usage.statistics.xml
|
142 |
+
.idea/**/dictionaries
|
143 |
+
.idea/**/shelf
|
144 |
+
|
145 |
+
# Sensitive or high-churn files
|
146 |
+
.idea/**/dataSources/
|
147 |
+
.idea/**/dataSources.ids
|
148 |
+
.idea/**/dataSources.local.xml
|
149 |
+
.idea/**/sqlDataSources.xml
|
150 |
+
.idea/**/dynamic.xml
|
151 |
+
.idea/**/uiDesigner.xml
|
152 |
+
.idea/**/dbnavigator.xml
|
153 |
+
|
154 |
+
# Gradle
|
155 |
+
.idea/**/gradle.xml
|
156 |
+
.idea/**/libraries
|
157 |
+
|
158 |
+
# Gradle and Maven with auto-import
|
159 |
+
# When using Gradle or Maven with auto-import, you should exclude module files,
|
160 |
+
# since they will be recreated, and may cause churn. Uncomment if using
|
161 |
+
# auto-import.
|
162 |
+
# .idea/modules.xml
|
163 |
+
# .idea/*.iml
|
164 |
+
# .idea/modules
|
165 |
+
|
166 |
+
# CMake
|
167 |
+
cmake-build-*/
|
168 |
+
|
169 |
+
# Mongo Explorer plugin
|
170 |
+
.idea/**/mongoSettings.xml
|
171 |
+
|
172 |
+
# File-based project format
|
173 |
+
*.iws
|
174 |
+
|
175 |
+
# IntelliJ
|
176 |
+
out/
|
177 |
+
|
178 |
+
# mpeltonen/sbt-idea plugin
|
179 |
+
.idea_modules/
|
180 |
+
|
181 |
+
# JIRA plugin
|
182 |
+
atlassian-ide-plugin.xml
|
183 |
+
|
184 |
+
# Cursive Clojure plugin
|
185 |
+
.idea/replstate.xml
|
186 |
+
|
187 |
+
# Crashlytics plugin (for Android Studio and IntelliJ)
|
188 |
+
com_crashlytics_export_strings.xml
|
189 |
+
crashlytics.properties
|
190 |
+
crashlytics-build.properties
|
191 |
+
fabric.properties
|
192 |
+
|
193 |
+
# Editor-based Rest Client
|
194 |
+
.idea/httpRequests
|
195 |
+
### VirtualEnv template
|
196 |
+
# Virtualenv
|
197 |
+
# http://iamzed.com/2009/05/07/a-primer-on-virtualenv/
|
198 |
+
.Python
|
199 |
+
[Bb]in
|
200 |
+
[Ii]nclude
|
201 |
+
[Ll]ib
|
202 |
+
[Ll]ib64
|
203 |
+
[Ll]ocal
|
204 |
+
pyvenv.cfg
|
205 |
+
.venv
|
206 |
+
pip-selfcheck.json
|
207 |
+
|
208 |
+
.idea/
|
209 |
+
eden.py
|
210 |
+
backup/
|
211 |
+
raw/
|
212 |
+
runs
|
213 |
+
*nohup*
|
214 |
+
*.pt
|
215 |
+
*.out
|
216 |
+
/nlgeval/
|
217 |
+
*.pkl
|
218 |
+
*.db
|
219 |
+
/cache/
|
220 |
+
_archived/
|
221 |
+
output/
|
222 |
+
models/
|
223 |
+
*_proc
|
224 |
+
lightning_logs/
|
225 |
+
wandb/
|
226 |
+
.lock
|
227 |
+
cjjpy.py
|
228 |
+
logs/
|
229 |
+
exp/
|
230 |
+
*.ipynb
|
231 |
+
docs/
|
232 |
+
.bashrc
|
233 |
+
google-cloud-sdk/
|
234 |
+
scripts
|
LICENSE
ADDED
@@ -0,0 +1,201 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
Apache License
|
2 |
+
Version 2.0, January 2004
|
3 |
+
http://www.apache.org/licenses/
|
4 |
+
|
5 |
+
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
6 |
+
|
7 |
+
1. Definitions.
|
8 |
+
|
9 |
+
"License" shall mean the terms and conditions for use, reproduction,
|
10 |
+
and distribution as defined by Sections 1 through 9 of this document.
|
11 |
+
|
12 |
+
"Licensor" shall mean the copyright owner or entity authorized by
|
13 |
+
the copyright owner that is granting the License.
|
14 |
+
|
15 |
+
"Legal Entity" shall mean the union of the acting entity and all
|
16 |
+
other entities that control, are controlled by, or are under common
|
17 |
+
control with that entity. For the purposes of this definition,
|
18 |
+
"control" means (i) the power, direct or indirect, to cause the
|
19 |
+
direction or management of such entity, whether by contract or
|
20 |
+
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
21 |
+
outstanding shares, or (iii) beneficial ownership of such entity.
|
22 |
+
|
23 |
+
"You" (or "Your") shall mean an individual or Legal Entity
|
24 |
+
exercising permissions granted by this License.
|
25 |
+
|
26 |
+
"Source" form shall mean the preferred form for making modifications,
|
27 |
+
including but not limited to software source code, documentation
|
28 |
+
source, and configuration files.
|
29 |
+
|
30 |
+
"Object" form shall mean any form resulting from mechanical
|
31 |
+
transformation or translation of a Source form, including but
|
32 |
+
not limited to compiled object code, generated documentation,
|
33 |
+
and conversions to other media types.
|
34 |
+
|
35 |
+
"Work" shall mean the work of authorship, whether in Source or
|
36 |
+
Object form, made available under the License, as indicated by a
|
37 |
+
copyright notice that is included in or attached to the work
|
38 |
+
(an example is provided in the Appendix below).
|
39 |
+
|
40 |
+
"Derivative Works" shall mean any work, whether in Source or Object
|
41 |
+
form, that is based on (or derived from) the Work and for which the
|
42 |
+
editorial revisions, annotations, elaborations, or other modifications
|
43 |
+
represent, as a whole, an original work of authorship. For the purposes
|
44 |
+
of this License, Derivative Works shall not include works that remain
|
45 |
+
separable from, or merely link (or bind by name) to the interfaces of,
|
46 |
+
the Work and Derivative Works thereof.
|
47 |
+
|
48 |
+
"Contribution" shall mean any work of authorship, including
|
49 |
+
the original version of the Work and any modifications or additions
|
50 |
+
to that Work or Derivative Works thereof, that is intentionally
|
51 |
+
submitted to Licensor for inclusion in the Work by the copyright owner
|
52 |
+
or by an individual or Legal Entity authorized to submit on behalf of
|
53 |
+
the copyright owner. For the purposes of this definition, "submitted"
|
54 |
+
means any form of electronic, verbal, or written communication sent
|
55 |
+
to the Licensor or its representatives, including but not limited to
|
56 |
+
communication on electronic mailing lists, source code control systems,
|
57 |
+
and issue tracking systems that are managed by, or on behalf of, the
|
58 |
+
Licensor for the purpose of discussing and improving the Work, but
|
59 |
+
excluding communication that is conspicuously marked or otherwise
|
60 |
+
designated in writing by the copyright owner as "Not a Contribution."
|
61 |
+
|
62 |
+
"Contributor" shall mean Licensor and any individual or Legal Entity
|
63 |
+
on behalf of whom a Contribution has been received by Licensor and
|
64 |
+
subsequently incorporated within the Work.
|
65 |
+
|
66 |
+
2. Grant of Copyright License. Subject to the terms and conditions of
|
67 |
+
this License, each Contributor hereby grants to You a perpetual,
|
68 |
+
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
69 |
+
copyright license to reproduce, prepare Derivative Works of,
|
70 |
+
publicly display, publicly perform, sublicense, and distribute the
|
71 |
+
Work and such Derivative Works in Source or Object form.
|
72 |
+
|
73 |
+
3. Grant of Patent License. Subject to the terms and conditions of
|
74 |
+
this License, each Contributor hereby grants to You a perpetual,
|
75 |
+
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
76 |
+
(except as stated in this section) patent license to make, have made,
|
77 |
+
use, offer to sell, sell, import, and otherwise transfer the Work,
|
78 |
+
where such license applies only to those patent claims licensable
|
79 |
+
by such Contributor that are necessarily infringed by their
|
80 |
+
Contribution(s) alone or by combination of their Contribution(s)
|
81 |
+
with the Work to which such Contribution(s) was submitted. If You
|
82 |
+
institute patent litigation against any entity (including a
|
83 |
+
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
84 |
+
or a Contribution incorporated within the Work constitutes direct
|
85 |
+
or contributory patent infringement, then any patent licenses
|
86 |
+
granted to You under this License for that Work shall terminate
|
87 |
+
as of the date such litigation is filed.
|
88 |
+
|
89 |
+
4. Redistribution. You may reproduce and distribute copies of the
|
90 |
+
Work or Derivative Works thereof in any medium, with or without
|
91 |
+
modifications, and in Source or Object form, provided that You
|
92 |
+
meet the following conditions:
|
93 |
+
|
94 |
+
(a) You must give any other recipients of the Work or
|
95 |
+
Derivative Works a copy of this License; and
|
96 |
+
|
97 |
+
(b) You must cause any modified files to carry prominent notices
|
98 |
+
stating that You changed the files; and
|
99 |
+
|
100 |
+
(c) You must retain, in the Source form of any Derivative Works
|
101 |
+
that You distribute, all copyright, patent, trademark, and
|
102 |
+
attribution notices from the Source form of the Work,
|
103 |
+
excluding those notices that do not pertain to any part of
|
104 |
+
the Derivative Works; and
|
105 |
+
|
106 |
+
(d) If the Work includes a "NOTICE" text file as part of its
|
107 |
+
distribution, then any Derivative Works that You distribute must
|
108 |
+
include a readable copy of the attribution notices contained
|
109 |
+
within such NOTICE file, excluding those notices that do not
|
110 |
+
pertain to any part of the Derivative Works, in at least one
|
111 |
+
of the following places: within a NOTICE text file distributed
|
112 |
+
as part of the Derivative Works; within the Source form or
|
113 |
+
documentation, if provided along with the Derivative Works; or,
|
114 |
+
within a display generated by the Derivative Works, if and
|
115 |
+
wherever such third-party notices normally appear. The contents
|
116 |
+
of the NOTICE file are for informational purposes only and
|
117 |
+
do not modify the License. You may add Your own attribution
|
118 |
+
notices within Derivative Works that You distribute, alongside
|
119 |
+
or as an addendum to the NOTICE text from the Work, provided
|
120 |
+
that such additional attribution notices cannot be construed
|
121 |
+
as modifying the License.
|
122 |
+
|
123 |
+
You may add Your own copyright statement to Your modifications and
|
124 |
+
may provide additional or different license terms and conditions
|
125 |
+
for use, reproduction, or distribution of Your modifications, or
|
126 |
+
for any such Derivative Works as a whole, provided Your use,
|
127 |
+
reproduction, and distribution of the Work otherwise complies with
|
128 |
+
the conditions stated in this License.
|
129 |
+
|
130 |
+
5. Submission of Contributions. Unless You explicitly state otherwise,
|
131 |
+
any Contribution intentionally submitted for inclusion in the Work
|
132 |
+
by You to the Licensor shall be under the terms and conditions of
|
133 |
+
this License, without any additional terms or conditions.
|
134 |
+
Notwithstanding the above, nothing herein shall supersede or modify
|
135 |
+
the terms of any separate license agreement you may have executed
|
136 |
+
with Licensor regarding such Contributions.
|
137 |
+
|
138 |
+
6. Trademarks. This License does not grant permission to use the trade
|
139 |
+
names, trademarks, service marks, or product names of the Licensor,
|
140 |
+
except as required for reasonable and customary use in describing the
|
141 |
+
origin of the Work and reproducing the content of the NOTICE file.
|
142 |
+
|
143 |
+
7. Disclaimer of Warranty. Unless required by applicable law or
|
144 |
+
agreed to in writing, Licensor provides the Work (and each
|
145 |
+
Contributor provides its Contributions) on an "AS IS" BASIS,
|
146 |
+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
147 |
+
implied, including, without limitation, any warranties or conditions
|
148 |
+
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
149 |
+
PARTICULAR PURPOSE. You are solely responsible for determining the
|
150 |
+
appropriateness of using or redistributing the Work and assume any
|
151 |
+
risks associated with Your exercise of permissions under this License.
|
152 |
+
|
153 |
+
8. Limitation of Liability. In no event and under no legal theory,
|
154 |
+
whether in tort (including negligence), contract, or otherwise,
|
155 |
+
unless required by applicable law (such as deliberate and grossly
|
156 |
+
negligent acts) or agreed to in writing, shall any Contributor be
|
157 |
+
liable to You for damages, including any direct, indirect, special,
|
158 |
+
incidental, or consequential damages of any character arising as a
|
159 |
+
result of this License or out of the use or inability to use the
|
160 |
+
Work (including but not limited to damages for loss of goodwill,
|
161 |
+
work stoppage, computer failure or malfunction, or any and all
|
162 |
+
other commercial damages or losses), even if such Contributor
|
163 |
+
has been advised of the possibility of such damages.
|
164 |
+
|
165 |
+
9. Accepting Warranty or Additional Liability. While redistributing
|
166 |
+
the Work or Derivative Works thereof, You may choose to offer,
|
167 |
+
and charge a fee for, acceptance of support, warranty, indemnity,
|
168 |
+
or other liability obligations and/or rights consistent with this
|
169 |
+
License. However, in accepting such obligations, You may act only
|
170 |
+
on Your own behalf and on Your sole responsibility, not on behalf
|
171 |
+
of any other Contributor, and only if You agree to indemnify,
|
172 |
+
defend, and hold each Contributor harmless for any liability
|
173 |
+
incurred by, or claims asserted against, such Contributor by reason
|
174 |
+
of your accepting any such warranty or additional liability.
|
175 |
+
|
176 |
+
END OF TERMS AND CONDITIONS
|
177 |
+
|
178 |
+
APPENDIX: How to apply the Apache License to your work.
|
179 |
+
|
180 |
+
To apply the Apache License to your work, attach the following
|
181 |
+
boilerplate notice, with the fields enclosed by brackets "[]"
|
182 |
+
replaced with your own identifying information. (Don't include
|
183 |
+
the brackets!) The text should be enclosed in the appropriate
|
184 |
+
comment syntax for the file format. We also recommend that a
|
185 |
+
file or class name and description of purpose be included on the
|
186 |
+
same "printed page" as the copyright notice for easier
|
187 |
+
identification within third-party archives.
|
188 |
+
|
189 |
+
Copyright [yyyy] [name of copyright owner]
|
190 |
+
|
191 |
+
Licensed under the Apache License, Version 2.0 (the "License");
|
192 |
+
you may not use this file except in compliance with the License.
|
193 |
+
You may obtain a copy of the License at
|
194 |
+
|
195 |
+
http://www.apache.org/licenses/LICENSE-2.0
|
196 |
+
|
197 |
+
Unless required by applicable law or agreed to in writing, software
|
198 |
+
distributed under the License is distributed on an "AS IS" BASIS,
|
199 |
+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
200 |
+
See the License for the specific language governing permissions and
|
201 |
+
limitations under the License.
|
app.py
ADDED
@@ -0,0 +1,396 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import gradio as gr
|
3 |
+
from app_modules.presets import *
|
4 |
+
from app_modules.overwrites import *
|
5 |
+
from app_modules.utils import *
|
6 |
+
from src.item_base import create_items
|
7 |
+
from src.bidder_base import Bidder
|
8 |
+
from src.human_bidder import HumanBidder
|
9 |
+
from src.auctioneer_base import Auctioneer
|
10 |
+
from auction_workflow import run_auction, make_auction_hash
|
11 |
+
from utils import chunks, reset_state_list
|
12 |
+
|
13 |
+
|
14 |
+
BIDDER_NUM = 4
|
15 |
+
items = create_items('data/items_demo.jsonl')
|
16 |
+
|
17 |
+
def auction_loop_app(*args):
|
18 |
+
global items
|
19 |
+
|
20 |
+
bidder_list = args[0] # gr.State() -> session state
|
21 |
+
items_id = args[1]
|
22 |
+
os.environ['OPENAI_API_KEY'] = args[2] if args[2] != '' else os.environ.get('OPENAI_API_KEY', '')
|
23 |
+
os.environ['ANTHROPIC_API_KEY'] = args[3] if args[3] != '' else os.environ.get('ANTHROPIC_API_KEY', '')
|
24 |
+
thread_num = args[4]
|
25 |
+
item_shuffle = args[5]
|
26 |
+
enable_discount = args[6]
|
27 |
+
min_markup_pct = args[7]
|
28 |
+
args = args[8:]
|
29 |
+
auction_hash = make_auction_hash()
|
30 |
+
|
31 |
+
items_to_bid = [items[i] for i in items_id]
|
32 |
+
|
33 |
+
auctioneer = Auctioneer(enable_discount=enable_discount, min_markup_pct=min_markup_pct)
|
34 |
+
auctioneer.init_items(items_to_bid)
|
35 |
+
if item_shuffle:
|
36 |
+
auctioneer.shuffle_items()
|
37 |
+
|
38 |
+
# must correspond to the order in app's parameters
|
39 |
+
input_keys = [
|
40 |
+
'chatbot',
|
41 |
+
'model_name',
|
42 |
+
'desire',
|
43 |
+
'plan_strategy',
|
44 |
+
'budget',
|
45 |
+
'correct_belief',
|
46 |
+
'enable_learning',
|
47 |
+
'temperature',
|
48 |
+
'overestimate_percent',
|
49 |
+
]
|
50 |
+
|
51 |
+
# convert flatten list into a json list
|
52 |
+
input_jsl = []
|
53 |
+
for i, chunk in enumerate(chunks(args, len(input_keys))):
|
54 |
+
js = {'name': f"Bidder {i+1}", 'auction_hash': auction_hash}
|
55 |
+
for k, v in zip(input_keys, chunk):
|
56 |
+
js[k] = v
|
57 |
+
input_jsl.append(js)
|
58 |
+
|
59 |
+
for js in input_jsl:
|
60 |
+
js.pop('chatbot')
|
61 |
+
if 'human' in js['model_name']:
|
62 |
+
bidder_list.append(HumanBidder.create(**js))
|
63 |
+
else:
|
64 |
+
bidder_list.append(Bidder.create(**js))
|
65 |
+
|
66 |
+
yield from run_auction(auction_hash, auctioneer, bidder_list, thread_num, yield_for_demo=True)
|
67 |
+
|
68 |
+
|
69 |
+
with open("assets/custom.css", "r", encoding="utf-8") as f:
|
70 |
+
customCSS = f.read()
|
71 |
+
|
72 |
+
with gr.Blocks(css=customCSS, theme=small_and_beautiful_theme) as demo:
|
73 |
+
with gr.Row():
|
74 |
+
gr.HTML(title)
|
75 |
+
|
76 |
+
gr.Markdown(description_top)
|
77 |
+
|
78 |
+
with gr.Row():
|
79 |
+
with gr.Column(scale=6):
|
80 |
+
# item_file = gr.File(label="Upload Item File", file_types=[".jsonl"])
|
81 |
+
items_checkbox = gr.CheckboxGroup(
|
82 |
+
choices=[item.info() for item in items[:20]],
|
83 |
+
label="Items in Auction",
|
84 |
+
info="Select the items you want to include in the auction.",
|
85 |
+
value=[item.info() for item in items[:8]],
|
86 |
+
type="index",
|
87 |
+
)
|
88 |
+
|
89 |
+
with gr.Column(scale=4):
|
90 |
+
with gr.Row():
|
91 |
+
openai_key = gr.Textbox(label="OpenAI API Key", value="", type="password", placeholder="sk-..")
|
92 |
+
anthropic_key = gr.Textbox(label="Anthropic API Key", value="", type="password", placeholder="sk-ant-..")
|
93 |
+
|
94 |
+
with gr.Row():
|
95 |
+
with gr.Row():
|
96 |
+
item_shuffle = gr.Checkbox(
|
97 |
+
label="Shuffle Items",
|
98 |
+
value=False,
|
99 |
+
info='Shuffle the order of items in the auction.')
|
100 |
+
enable_discount = gr.Checkbox(
|
101 |
+
label="Enable Discount",
|
102 |
+
value=False,
|
103 |
+
info='When an item fails to sell at auction, it can be auctioned again at a reduced price.')
|
104 |
+
|
105 |
+
with gr.Column():
|
106 |
+
min_markup_pct = gr.Slider(
|
107 |
+
minimum=0.1,
|
108 |
+
maximum=0.5,
|
109 |
+
value=0.1,
|
110 |
+
step=0.1,
|
111 |
+
interactive=True,
|
112 |
+
label='Min Increase',
|
113 |
+
info="The minimum percentage to increase a bid.",
|
114 |
+
)
|
115 |
+
|
116 |
+
thread_num = gr.Slider(
|
117 |
+
minimum=1,
|
118 |
+
maximum=BIDDER_NUM,
|
119 |
+
value=min(5, BIDDER_NUM),
|
120 |
+
step=1,
|
121 |
+
interactive=True,
|
122 |
+
label='Thread Number',
|
123 |
+
info="More threads, faster bidding, but will run into RateLimitError quicker."
|
124 |
+
)
|
125 |
+
|
126 |
+
with gr.Row():
|
127 |
+
bidder_info_gr = []
|
128 |
+
chatbots = []
|
129 |
+
monitors = []
|
130 |
+
textbox_list = []
|
131 |
+
for i in range(BIDDER_NUM):
|
132 |
+
with gr.Tab(label=f"Bidder {i+1}"):
|
133 |
+
with gr.Row().style(equal_height=True):
|
134 |
+
with gr.Column(scale=6):
|
135 |
+
with gr.Row():
|
136 |
+
chatbot = gr.Chatbot(elem_id="chuanhu_chatbot", height=600, label='Auction Log')
|
137 |
+
input_box = gr.Textbox(label="Human Bidder Input", interactive=False, placeholder="Please wait a moment before engaging in the auction.", visible=False)
|
138 |
+
chatbots.append(chatbot)
|
139 |
+
textbox_list.append(input_box)
|
140 |
+
with gr.Column(scale=4):
|
141 |
+
with gr.Tab(label=f'Parameters'):
|
142 |
+
model_name = gr.Dropdown(
|
143 |
+
choices=[
|
144 |
+
'rule',
|
145 |
+
'human',
|
146 |
+
'gpt-3.5-turbo-0613',
|
147 |
+
'gpt-3.5-turbo-16k-0613',
|
148 |
+
'gpt-4-0613',
|
149 |
+
# 'claude-instant-1.1',
|
150 |
+
'claude-instant-1.2',
|
151 |
+
# 'claude-1.3',
|
152 |
+
'claude-2.0',
|
153 |
+
# 'chat-bison-001',
|
154 |
+
],
|
155 |
+
value='gpt-3.5-turbo-16k-0613',
|
156 |
+
label="Model Selection",
|
157 |
+
)
|
158 |
+
budget = gr.Number(
|
159 |
+
value=10000,
|
160 |
+
label='Budget ($)'
|
161 |
+
)
|
162 |
+
with gr.Row():
|
163 |
+
plan_strategy = gr.Dropdown(
|
164 |
+
choices=[
|
165 |
+
'none',
|
166 |
+
'static',
|
167 |
+
'adaptive',
|
168 |
+
],
|
169 |
+
value='adaptive',
|
170 |
+
label='Planning Strategy',
|
171 |
+
info='None: no plan. Static: plan only once. Adaptive: replan for the remaining items.'
|
172 |
+
)
|
173 |
+
desire = gr.Dropdown(
|
174 |
+
choices=[
|
175 |
+
# 'default',
|
176 |
+
'maximize_profit',
|
177 |
+
'maximize_items',
|
178 |
+
# 'specific_items',
|
179 |
+
],
|
180 |
+
value='maximize_profit',
|
181 |
+
label='Desire',
|
182 |
+
info='Default desires: spending all the budget, stay within budget. All desires include the default one.',
|
183 |
+
)
|
184 |
+
overestimate_percent = gr.Slider(
|
185 |
+
minimum=-100,
|
186 |
+
maximum=100,
|
187 |
+
value=10,
|
188 |
+
step=10,
|
189 |
+
interactive=True,
|
190 |
+
label='Overestimate Percent (%)',
|
191 |
+
info="Overestimate the true value of items by this percentage.",
|
192 |
+
)
|
193 |
+
with gr.Row():
|
194 |
+
correct_belief = gr.Checkbox(
|
195 |
+
label='Correct Wrong Beliefs',
|
196 |
+
value=True,
|
197 |
+
info='Forceful beliefs correction about self and others.',
|
198 |
+
)
|
199 |
+
enable_learning = gr.Checkbox(
|
200 |
+
label='Enable Learning',
|
201 |
+
value=False,
|
202 |
+
info='Learn from past auctions for future guidance. Only for adaptive bidder.',
|
203 |
+
visible=False
|
204 |
+
)
|
205 |
+
temperature = gr.Slider(
|
206 |
+
minimum=0.,
|
207 |
+
maximum=2.0,
|
208 |
+
value=0.7,
|
209 |
+
step=0.1,
|
210 |
+
interactive=True,
|
211 |
+
label="Temperature",
|
212 |
+
)
|
213 |
+
|
214 |
+
# deprecated
|
215 |
+
# special_items = gr.CheckboxGroup(
|
216 |
+
# value = [],
|
217 |
+
# label='Special Items',
|
218 |
+
# info='Special items add 20% value for you personally.',
|
219 |
+
# visible=False,
|
220 |
+
# )
|
221 |
+
# hedge_percent = gr.Slider(
|
222 |
+
# minimum=0,
|
223 |
+
# maximum=100,
|
224 |
+
# value=90,
|
225 |
+
# step=1,
|
226 |
+
# interactive=True,
|
227 |
+
# label='Strategy (Hedging %)',
|
228 |
+
# info="The maximum percentage of the estimated value to bid on an item.",
|
229 |
+
# visible=False
|
230 |
+
# )
|
231 |
+
|
232 |
+
with gr.Tab(label='Monitors'):
|
233 |
+
with gr.Row():
|
234 |
+
budget_monitor = gr.Number(label='Budget Left ($)', interactive=False)
|
235 |
+
profit_monitor = gr.Number(label='Profit ($)', interactive=False)
|
236 |
+
|
237 |
+
with gr.Row():
|
238 |
+
engagement_monitor = gr.Number(
|
239 |
+
label='Engagement',
|
240 |
+
interactive=False,
|
241 |
+
info='The number of times the bidder has bid.'
|
242 |
+
)
|
243 |
+
failure_monitor = gr.Number(
|
244 |
+
label='Failed Bids',
|
245 |
+
info='Out-of-budget, or less than the previous highest bid.',
|
246 |
+
interactive=False
|
247 |
+
)
|
248 |
+
|
249 |
+
items_own_monitor = gr.DataFrame(
|
250 |
+
label='Items Owned',
|
251 |
+
headers=['Item', 'Bid ($)', 'Value ($)'],
|
252 |
+
datatype=['str', 'number', 'number'],
|
253 |
+
interactive=False,
|
254 |
+
)
|
255 |
+
|
256 |
+
with gr.Row():
|
257 |
+
tokens_monitor = gr.Number(
|
258 |
+
label='Token Used',
|
259 |
+
interactive=False,
|
260 |
+
info='Tokens used in the last call.'
|
261 |
+
)
|
262 |
+
money_monitor = gr.Number(
|
263 |
+
label='API Cost ($)',
|
264 |
+
info='Only OpenAI cost for now.',
|
265 |
+
interactive=False
|
266 |
+
)
|
267 |
+
|
268 |
+
plan_change_monitor = gr.DataFrame(
|
269 |
+
label='Plan Changes',
|
270 |
+
headers=['Round', 'Changed', 'New Plan'],
|
271 |
+
datatype=['str', 'bool', 'str'],
|
272 |
+
interactive=False,
|
273 |
+
)
|
274 |
+
|
275 |
+
plot_monitor = gr.Plot(
|
276 |
+
label='Budget-Profit Plot',
|
277 |
+
interactive=False
|
278 |
+
)
|
279 |
+
|
280 |
+
with gr.Tab(label='Belief Errors'):
|
281 |
+
with gr.Row():
|
282 |
+
self_belief_error_cnt_monitor = gr.Number(
|
283 |
+
label='Wrong Beliefs of Self',
|
284 |
+
info='Not knowing its own budget, bid items, or won items.',
|
285 |
+
interactive=False,
|
286 |
+
)
|
287 |
+
other_belief_error_cnt_monitor = gr.Number(
|
288 |
+
label='Wrong Beliefs of Others',
|
289 |
+
info='Not knowing other bidders\' profits.',
|
290 |
+
interactive=False,
|
291 |
+
)
|
292 |
+
budget_belief_monitor = gr.DataFrame(
|
293 |
+
label='Wrong Belief of Budget ($)',
|
294 |
+
headers=['Round', 'Belief', 'Truth'],
|
295 |
+
datatype=['str', 'number', 'number'],
|
296 |
+
interactive=False,
|
297 |
+
)
|
298 |
+
profit_belief_monitor = gr.DataFrame(
|
299 |
+
label='Wrong Belief of Profit ($)',
|
300 |
+
headers=['Bidder (Round)', 'Belief', 'Truth'],
|
301 |
+
datatype=['str', 'number', 'number'],
|
302 |
+
interactive=False,
|
303 |
+
)
|
304 |
+
win_bid_belief_monitor = gr.DataFrame(
|
305 |
+
label='Wrong Belief of Items Won',
|
306 |
+
headers=['Bidder (Round)',
|
307 |
+
'Belief', 'Truth'],
|
308 |
+
datatype=['str', 'str', 'str'],
|
309 |
+
interactive=False,
|
310 |
+
)
|
311 |
+
|
312 |
+
monitors += [
|
313 |
+
budget_monitor,
|
314 |
+
profit_monitor,
|
315 |
+
items_own_monitor,
|
316 |
+
tokens_monitor,
|
317 |
+
money_monitor,
|
318 |
+
failure_monitor,
|
319 |
+
self_belief_error_cnt_monitor,
|
320 |
+
other_belief_error_cnt_monitor,
|
321 |
+
engagement_monitor,
|
322 |
+
plot_monitor,
|
323 |
+
plan_change_monitor,
|
324 |
+
budget_belief_monitor,
|
325 |
+
profit_belief_monitor,
|
326 |
+
win_bid_belief_monitor,
|
327 |
+
]
|
328 |
+
|
329 |
+
bidder_info_gr += [
|
330 |
+
chatbot,
|
331 |
+
model_name,
|
332 |
+
desire,
|
333 |
+
plan_strategy,
|
334 |
+
budget,
|
335 |
+
correct_belief,
|
336 |
+
enable_learning,
|
337 |
+
temperature,
|
338 |
+
overestimate_percent,
|
339 |
+
]
|
340 |
+
|
341 |
+
with gr.Row():
|
342 |
+
with gr.Column():
|
343 |
+
startBtn = gr.Button('Start Bidding', variant='primary', interactive=True)
|
344 |
+
with gr.Column():
|
345 |
+
clearBtn = gr.Button('New Auction', variant='secondary', interactive=False)
|
346 |
+
btn_list = [startBtn, clearBtn]
|
347 |
+
|
348 |
+
with gr.Accordion(label='Bidding Log (click to open)', open=True):
|
349 |
+
with gr.Row():
|
350 |
+
bidding_log = gr.Markdown(value="")
|
351 |
+
|
352 |
+
gr.Markdown(description)
|
353 |
+
|
354 |
+
bidder_list_state = gr.State([]) # session state
|
355 |
+
|
356 |
+
start_args = dict(
|
357 |
+
fn=auction_loop_app,
|
358 |
+
inputs=[bidder_list_state, items_checkbox, openai_key, anthropic_key, thread_num, item_shuffle, enable_discount, min_markup_pct] + bidder_info_gr,
|
359 |
+
outputs=[bidder_list_state] + chatbots + monitors + [bidding_log] + btn_list + textbox_list, # TODO: handle textbox_list interactivity
|
360 |
+
show_progress=True,
|
361 |
+
)
|
362 |
+
start_event = startBtn.click(**start_args)
|
363 |
+
|
364 |
+
def bot(user_message, bidder_list, id):
|
365 |
+
if len(bidder_list) > 0:
|
366 |
+
bidder = bidder_list[int(id)]
|
367 |
+
if bidder.need_input:
|
368 |
+
bidder.input_box = user_message
|
369 |
+
bidder.semaphore += 1
|
370 |
+
return '', bidder_list
|
371 |
+
|
372 |
+
# handle user input from time to time
|
373 |
+
for i in range(len(textbox_list)):
|
374 |
+
_dummy_id = gr.Number(i, visible=False, interactive=False)
|
375 |
+
textbox_list[i].submit(
|
376 |
+
bot,
|
377 |
+
[textbox_list[i], bidder_list_state, _dummy_id],
|
378 |
+
[textbox_list[i], bidder_list_state])
|
379 |
+
|
380 |
+
clearBtn.click(reset_state_list,
|
381 |
+
inputs=[bidder_list_state] + chatbots + monitors + [bidding_log],
|
382 |
+
outputs=[bidder_list_state] + chatbots + monitors + [bidding_log],
|
383 |
+
show_progress=True).then(lambda: gr.update(interactive=True), outputs=[startBtn])
|
384 |
+
|
385 |
+
demo.title = 'Auction Arena'
|
386 |
+
|
387 |
+
|
388 |
+
demo.queue(max_size=64, concurrency_count=16).launch(
|
389 |
+
# server_name='0.0.0.0',
|
390 |
+
# ssl_verify=False,
|
391 |
+
# share=True,
|
392 |
+
# debug=True,
|
393 |
+
show_api=False,
|
394 |
+
)
|
395 |
+
|
396 |
+
demo.close()
|
app_modules/overwrites.py
ADDED
@@ -0,0 +1,57 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from __future__ import annotations
|
2 |
+
import logging
|
3 |
+
|
4 |
+
# from llama_index import Prompt
|
5 |
+
from typing import List, Tuple
|
6 |
+
# import mdtex2html
|
7 |
+
|
8 |
+
from app_modules.presets import *
|
9 |
+
from app_modules.utils import *
|
10 |
+
|
11 |
+
def compact_text_chunks(self, prompt: Prompt, text_chunks: List[str]) -> List[str]:
|
12 |
+
logging.debug("Compacting text chunks...🚀🚀🚀")
|
13 |
+
combined_str = [c.strip() for c in text_chunks if c.strip()]
|
14 |
+
combined_str = [f"[{index+1}] {c}" for index, c in enumerate(combined_str)]
|
15 |
+
combined_str = "\n\n".join(combined_str)
|
16 |
+
# resplit based on self.max_chunk_overlap
|
17 |
+
text_splitter = self.get_text_splitter_given_prompt(prompt, 1, padding=1)
|
18 |
+
return text_splitter.split_text(combined_str)
|
19 |
+
|
20 |
+
|
21 |
+
def postprocess(
|
22 |
+
self, y: List[Tuple[str | None, str | None]]
|
23 |
+
) -> List[Tuple[str | None, str | None]]:
|
24 |
+
"""
|
25 |
+
Parameters:
|
26 |
+
y: List of tuples representing the message and response pairs. Each message and response should be a string, which may be in Markdown format.
|
27 |
+
Returns:
|
28 |
+
List of tuples representing the message and response. Each message and response will be a string of HTML.
|
29 |
+
"""
|
30 |
+
if y is None or y == []:
|
31 |
+
return []
|
32 |
+
temp = []
|
33 |
+
for x in y:
|
34 |
+
user, bot = x
|
35 |
+
if not detect_converted_mark(user):
|
36 |
+
user = convert_asis(user)
|
37 |
+
if not detect_converted_mark(bot):
|
38 |
+
bot = convert_mdtext(bot)
|
39 |
+
temp.append((user, bot))
|
40 |
+
return temp
|
41 |
+
|
42 |
+
with open("./assets/custom.js", "r", encoding="utf-8") as f, open("./assets/Kelpy-Codos.js", "r", encoding="utf-8") as f2:
|
43 |
+
customJS = f.read()
|
44 |
+
kelpyCodos = f2.read()
|
45 |
+
|
46 |
+
def reload_javascript():
|
47 |
+
print("Reloading javascript...")
|
48 |
+
js = f'<script>{customJS}</script><script>{kelpyCodos}</script>'
|
49 |
+
def template_response(*args, **kwargs):
|
50 |
+
res = GradioTemplateResponseOriginal(*args, **kwargs)
|
51 |
+
res.body = res.body.replace(b'</html>', f'{js}</html>'.encode("utf8"))
|
52 |
+
res.init_headers()
|
53 |
+
return res
|
54 |
+
|
55 |
+
gr.routes.templates.TemplateResponse = template_response
|
56 |
+
|
57 |
+
GradioTemplateResponseOriginal = gr.routes.templates.TemplateResponse
|
app_modules/presets.py
ADDED
@@ -0,0 +1,78 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import gradio as gr
|
2 |
+
|
3 |
+
|
4 |
+
title = """<h1 align="center" style="min-width:200px; margin-top:0;"> <img src="https://raw.githubusercontent.com/jiangjiechen/jiangjiechen.github.io/1f23a6b72b7e0b57a54e31a583c9f668a2b8b4b6/media/icon_hua9f9b78e35233aa477f7219cbf68418f_67044_512x512_fill_lanczos_center_3.png" width="32px" style="display: inline"> Auction Arena </h1>"""
|
5 |
+
|
6 |
+
description_top = """\
|
7 |
+
<div align="center">
|
8 |
+
<p>
|
9 |
+
An interactive demo for this paper: <a href="https://auction-arena.github.io">Put Your Money Where Your Mouth Is: Evaluating Strategic Planning and Execution of LLM Agents in an Auction Arena</a>. Details of this work can be found at <a href="https://auction-arena.github.io">this page</a>. You can watch AI vs AI in this auction arena, or you can set `model_name=human` to engage in the arena personally. Please enter your API key before start, otherwise you will have errors (please refresh the page if you do). Feel free to <a href="mailto:jjchen19@fudan.edu.cn">contact us</a> if you have any questions!
|
10 |
+
</p >
|
11 |
+
</div>
|
12 |
+
"""
|
13 |
+
description = """\
|
14 |
+
<div align="center" style="margin:16px 0">
|
15 |
+
The demo is built on <a href="https://github.com/GaiZhenbiao/ChuanhuChatGPT">ChuanhuChatGPT</a>.
|
16 |
+
</div>
|
17 |
+
"""
|
18 |
+
|
19 |
+
small_and_beautiful_theme = gr.themes.Soft(
|
20 |
+
primary_hue=gr.themes.Color(
|
21 |
+
c50="#02C160",
|
22 |
+
c100="rgba(2, 193, 96, 0.2)",
|
23 |
+
c200="#02C160",
|
24 |
+
c300="rgba(2, 193, 96, 0.32)",
|
25 |
+
c400="rgba(2, 193, 96, 0.32)",
|
26 |
+
c500="rgba(2, 193, 96, 1.0)",
|
27 |
+
c600="rgba(2, 193, 96, 1.0)",
|
28 |
+
c700="rgba(2, 193, 96, 0.32)",
|
29 |
+
c800="rgba(2, 193, 96, 0.32)",
|
30 |
+
c900="#02C160",
|
31 |
+
c950="#02C160",
|
32 |
+
),
|
33 |
+
secondary_hue=gr.themes.Color(
|
34 |
+
c50="#576b95",
|
35 |
+
c100="#576b95",
|
36 |
+
c200="#576b95",
|
37 |
+
c300="#576b95",
|
38 |
+
c400="#576b95",
|
39 |
+
c500="#576b95",
|
40 |
+
c600="#576b95",
|
41 |
+
c700="#576b95",
|
42 |
+
c800="#576b95",
|
43 |
+
c900="#576b95",
|
44 |
+
c950="#576b95",
|
45 |
+
),
|
46 |
+
neutral_hue=gr.themes.Color(
|
47 |
+
name="gray",
|
48 |
+
c50="#f9fafb",
|
49 |
+
c100="#f3f4f6",
|
50 |
+
c200="#e5e7eb",
|
51 |
+
c300="#d1d5db",
|
52 |
+
c400="#B2B2B2",
|
53 |
+
c500="#808080",
|
54 |
+
c600="#636363",
|
55 |
+
c700="#515151",
|
56 |
+
c800="#393939",
|
57 |
+
c900="#272727",
|
58 |
+
c950="#171717",
|
59 |
+
),
|
60 |
+
radius_size=gr.themes.sizes.radius_sm,
|
61 |
+
).set(
|
62 |
+
button_primary_background_fill="#06AE56",
|
63 |
+
button_primary_background_fill_dark="#06AE56",
|
64 |
+
button_primary_background_fill_hover="#07C863",
|
65 |
+
button_primary_border_color="#06AE56",
|
66 |
+
button_primary_border_color_dark="#06AE56",
|
67 |
+
button_primary_text_color="#FFFFFF",
|
68 |
+
button_primary_text_color_dark="#FFFFFF",
|
69 |
+
button_secondary_background_fill="#F2F2F2",
|
70 |
+
button_secondary_background_fill_dark="#2B2B2B",
|
71 |
+
button_secondary_text_color="#393939",
|
72 |
+
button_secondary_text_color_dark="#FFFFFF",
|
73 |
+
# background_fill_primary="#F7F7F7",
|
74 |
+
# background_fill_primary_dark="#1F1F1F",
|
75 |
+
block_title_text_color="*primary_500",
|
76 |
+
block_title_background_fill="*primary_100",
|
77 |
+
input_background_fill="#F6F6F6",
|
78 |
+
)
|
app_modules/utils.py
ADDED
@@ -0,0 +1,375 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# -*- coding:utf-8 -*-
|
2 |
+
from __future__ import annotations
|
3 |
+
from typing import TYPE_CHECKING, Any, Callable, Dict, List, Tuple, Type
|
4 |
+
import logging
|
5 |
+
# import json
|
6 |
+
# import os
|
7 |
+
# import datetime
|
8 |
+
# import hashlib
|
9 |
+
# import csv
|
10 |
+
# import requests
|
11 |
+
import re
|
12 |
+
import html
|
13 |
+
# import markdown2
|
14 |
+
import torch
|
15 |
+
import sys
|
16 |
+
import gc
|
17 |
+
from pygments.lexers import guess_lexer, ClassNotFound
|
18 |
+
|
19 |
+
import gradio as gr
|
20 |
+
# from pypinyin import lazy_pinyin
|
21 |
+
# import tiktoken
|
22 |
+
# import mdtex2html
|
23 |
+
# from markdown import markdown
|
24 |
+
from pygments import highlight
|
25 |
+
from pygments.lexers import guess_lexer,get_lexer_by_name
|
26 |
+
from pygments.formatters import HtmlFormatter
|
27 |
+
# import transformers
|
28 |
+
# from peft import PeftModel
|
29 |
+
# from transformers import GenerationConfig, LlamaForCausalLM, LlamaTokenizer
|
30 |
+
|
31 |
+
from app_modules.presets import *
|
32 |
+
|
33 |
+
# logging.basicConfig(
|
34 |
+
# level=logging.INFO,
|
35 |
+
# format="%(asctime)s [%(levelname)s] [%(filename)s:%(lineno)d] %(message)s",
|
36 |
+
# )
|
37 |
+
|
38 |
+
|
39 |
+
def markdown_to_html_with_syntax_highlight(md_str):
|
40 |
+
def replacer(match):
|
41 |
+
lang = match.group(1) or "text"
|
42 |
+
code = match.group(2)
|
43 |
+
lang = lang.strip()
|
44 |
+
#print(1,lang)
|
45 |
+
if lang=="text":
|
46 |
+
lexer = guess_lexer(code)
|
47 |
+
lang = lexer.name
|
48 |
+
#print(2,lang)
|
49 |
+
try:
|
50 |
+
lexer = get_lexer_by_name(lang, stripall=True)
|
51 |
+
except ValueError:
|
52 |
+
lexer = get_lexer_by_name("python", stripall=True)
|
53 |
+
formatter = HtmlFormatter()
|
54 |
+
#print(3,lexer.name)
|
55 |
+
highlighted_code = highlight(code, lexer, formatter)
|
56 |
+
|
57 |
+
return f'<pre><code class="{lang}">{highlighted_code}</code></pre>'
|
58 |
+
|
59 |
+
code_block_pattern = r"```(\w+)?\n([\s\S]+?)\n```"
|
60 |
+
md_str = re.sub(code_block_pattern, replacer, md_str, flags=re.MULTILINE)
|
61 |
+
|
62 |
+
html_str = markdown(md_str)
|
63 |
+
return html_str
|
64 |
+
|
65 |
+
|
66 |
+
def normalize_markdown(md_text: str) -> str:
|
67 |
+
lines = md_text.split("\n")
|
68 |
+
normalized_lines = []
|
69 |
+
inside_list = False
|
70 |
+
|
71 |
+
for i, line in enumerate(lines):
|
72 |
+
if re.match(r"^(\d+\.|-|\*|\+)\s", line.strip()):
|
73 |
+
if not inside_list and i > 0 and lines[i - 1].strip() != "":
|
74 |
+
normalized_lines.append("")
|
75 |
+
inside_list = True
|
76 |
+
normalized_lines.append(line)
|
77 |
+
elif inside_list and line.strip() == "":
|
78 |
+
if i < len(lines) - 1 and not re.match(
|
79 |
+
r"^(\d+\.|-|\*|\+)\s", lines[i + 1].strip()
|
80 |
+
):
|
81 |
+
normalized_lines.append(line)
|
82 |
+
continue
|
83 |
+
else:
|
84 |
+
inside_list = False
|
85 |
+
normalized_lines.append(line)
|
86 |
+
|
87 |
+
return "\n".join(normalized_lines)
|
88 |
+
|
89 |
+
|
90 |
+
def convert_mdtext(md_text):
|
91 |
+
code_block_pattern = re.compile(r"```(.*?)(?:```|$)", re.DOTALL)
|
92 |
+
inline_code_pattern = re.compile(r"`(.*?)`", re.DOTALL)
|
93 |
+
code_blocks = code_block_pattern.findall(md_text)
|
94 |
+
non_code_parts = code_block_pattern.split(md_text)[::2]
|
95 |
+
|
96 |
+
result = []
|
97 |
+
for non_code, code in zip(non_code_parts, code_blocks + [""]):
|
98 |
+
if non_code.strip():
|
99 |
+
non_code = normalize_markdown(non_code)
|
100 |
+
if inline_code_pattern.search(non_code):
|
101 |
+
result.append(markdown(non_code, extensions=["tables"]))
|
102 |
+
else:
|
103 |
+
result.append(mdtex2html.convert(non_code, extensions=["tables"]))
|
104 |
+
if code.strip():
|
105 |
+
code = f"\n```{code}\n\n```"
|
106 |
+
code = markdown_to_html_with_syntax_highlight(code)
|
107 |
+
result.append(code)
|
108 |
+
result = "".join(result)
|
109 |
+
result += ALREADY_CONVERTED_MARK
|
110 |
+
return result
|
111 |
+
|
112 |
+
def convert_asis(userinput):
|
113 |
+
return f"<p style=\"white-space:pre-wrap;\">{html.escape(userinput)}</p>"+ALREADY_CONVERTED_MARK
|
114 |
+
|
115 |
+
def detect_converted_mark(userinput):
|
116 |
+
if userinput.endswith(ALREADY_CONVERTED_MARK):
|
117 |
+
return True
|
118 |
+
else:
|
119 |
+
return False
|
120 |
+
|
121 |
+
|
122 |
+
|
123 |
+
def detect_language(code):
|
124 |
+
if code.startswith("\n"):
|
125 |
+
first_line = ""
|
126 |
+
else:
|
127 |
+
first_line = code.strip().split("\n", 1)[0]
|
128 |
+
language = first_line.lower() if first_line else ""
|
129 |
+
code_without_language = code[len(first_line) :].lstrip() if first_line else code
|
130 |
+
return language, code_without_language
|
131 |
+
|
132 |
+
def convert_to_markdown(text):
|
133 |
+
text = text.replace("$","$")
|
134 |
+
def replace_leading_tabs_and_spaces(line):
|
135 |
+
new_line = []
|
136 |
+
|
137 |
+
for char in line:
|
138 |
+
if char == "\t":
|
139 |
+
new_line.append("	")
|
140 |
+
elif char == " ":
|
141 |
+
new_line.append(" ")
|
142 |
+
else:
|
143 |
+
break
|
144 |
+
return "".join(new_line) + line[len(new_line):]
|
145 |
+
|
146 |
+
markdown_text = ""
|
147 |
+
lines = text.split("\n")
|
148 |
+
in_code_block = False
|
149 |
+
|
150 |
+
for line in lines:
|
151 |
+
if in_code_block is False and line.startswith("```"):
|
152 |
+
in_code_block = True
|
153 |
+
markdown_text += f"{line}\n"
|
154 |
+
elif in_code_block is True and line.startswith("```"):
|
155 |
+
in_code_block = False
|
156 |
+
markdown_text += f"{line}\n"
|
157 |
+
elif in_code_block:
|
158 |
+
markdown_text += f"{line}\n"
|
159 |
+
else:
|
160 |
+
line = replace_leading_tabs_and_spaces(line)
|
161 |
+
line = re.sub(r"^(#)", r"\\\1", line)
|
162 |
+
markdown_text += f"{line} \n"
|
163 |
+
|
164 |
+
return markdown_text
|
165 |
+
|
166 |
+
def add_language_tag(text):
|
167 |
+
def detect_language(code_block):
|
168 |
+
try:
|
169 |
+
lexer = guess_lexer(code_block)
|
170 |
+
return lexer.name.lower()
|
171 |
+
except ClassNotFound:
|
172 |
+
return ""
|
173 |
+
|
174 |
+
code_block_pattern = re.compile(r"(```)(\w*\n[^`]+```)", re.MULTILINE)
|
175 |
+
|
176 |
+
def replacement(match):
|
177 |
+
code_block = match.group(2)
|
178 |
+
if match.group(2).startswith("\n"):
|
179 |
+
language = detect_language(code_block)
|
180 |
+
if language:
|
181 |
+
return f"```{language}{code_block}```"
|
182 |
+
else:
|
183 |
+
return f"```\n{code_block}```"
|
184 |
+
else:
|
185 |
+
return match.group(1) + code_block + "```"
|
186 |
+
|
187 |
+
text2 = code_block_pattern.sub(replacement, text)
|
188 |
+
return text2
|
189 |
+
|
190 |
+
def delete_last_conversation(chatbot, history):
|
191 |
+
if len(chatbot) > 0:
|
192 |
+
chatbot.pop()
|
193 |
+
|
194 |
+
if len(history) > 0:
|
195 |
+
history.pop()
|
196 |
+
|
197 |
+
return (
|
198 |
+
chatbot,
|
199 |
+
history,
|
200 |
+
"Delete Done",
|
201 |
+
)
|
202 |
+
|
203 |
+
def reset_state():
|
204 |
+
return [], [], "Reset Done"
|
205 |
+
|
206 |
+
def reset_textbox():
|
207 |
+
return gr.update(value=""),""
|
208 |
+
|
209 |
+
def cancel_outputing():
|
210 |
+
return "Stop Done"
|
211 |
+
|
212 |
+
def transfer_input(inputs):
|
213 |
+
textbox = reset_textbox()
|
214 |
+
return (
|
215 |
+
inputs,
|
216 |
+
gr.update(value=""),
|
217 |
+
gr.Button.update(visible=True),
|
218 |
+
)
|
219 |
+
|
220 |
+
|
221 |
+
class State:
|
222 |
+
interrupted = False
|
223 |
+
|
224 |
+
def interrupt(self):
|
225 |
+
self.interrupted = True
|
226 |
+
|
227 |
+
def recover(self):
|
228 |
+
self.interrupted = False
|
229 |
+
shared_state = State()
|
230 |
+
|
231 |
+
|
232 |
+
|
233 |
+
|
234 |
+
|
235 |
+
# Greedy Search
|
236 |
+
def greedy_search(input_ids: torch.Tensor,
|
237 |
+
model: torch.nn.Module,
|
238 |
+
tokenizer: transformers.PreTrainedTokenizer,
|
239 |
+
stop_words: list,
|
240 |
+
max_length: int,
|
241 |
+
temperature: float = 1.0,
|
242 |
+
top_p: float = 1.0,
|
243 |
+
top_k: int = 25) -> Iterator[str]:
|
244 |
+
generated_tokens = []
|
245 |
+
past_key_values = None
|
246 |
+
current_length = 1
|
247 |
+
for i in range(max_length):
|
248 |
+
with torch.no_grad():
|
249 |
+
if past_key_values is None:
|
250 |
+
outputs = model(input_ids)
|
251 |
+
else:
|
252 |
+
outputs = model(input_ids[:, -1:], past_key_values=past_key_values)
|
253 |
+
logits = outputs.logits[:, -1, :]
|
254 |
+
past_key_values = outputs.past_key_values
|
255 |
+
|
256 |
+
# apply temperature
|
257 |
+
logits /= temperature
|
258 |
+
|
259 |
+
probs = torch.softmax(logits, dim=-1)
|
260 |
+
# apply top_p
|
261 |
+
probs_sort, probs_idx = torch.sort(probs, dim=-1, descending=True)
|
262 |
+
probs_sum = torch.cumsum(probs_sort, dim=-1)
|
263 |
+
mask = probs_sum - probs_sort > top_p
|
264 |
+
probs_sort[mask] = 0.0
|
265 |
+
|
266 |
+
# apply top_k
|
267 |
+
#if top_k is not None:
|
268 |
+
# probs_sort1, _ = torch.topk(probs_sort, top_k)
|
269 |
+
# min_top_probs_sort = torch.min(probs_sort1, dim=-1, keepdim=True).values
|
270 |
+
# probs_sort = torch.where(probs_sort < min_top_probs_sort, torch.full_like(probs_sort, float(0.0)), probs_sort)
|
271 |
+
|
272 |
+
probs_sort.div_(probs_sort.sum(dim=-1, keepdim=True))
|
273 |
+
next_token = torch.multinomial(probs_sort, num_samples=1)
|
274 |
+
next_token = torch.gather(probs_idx, -1, next_token)
|
275 |
+
|
276 |
+
input_ids = torch.cat((input_ids, next_token), dim=-1)
|
277 |
+
|
278 |
+
generated_tokens.append(next_token[0].item())
|
279 |
+
text = tokenizer.decode(generated_tokens)
|
280 |
+
|
281 |
+
yield text
|
282 |
+
if any([x in text for x in stop_words]):
|
283 |
+
del past_key_values
|
284 |
+
del logits
|
285 |
+
del probs
|
286 |
+
del probs_sort
|
287 |
+
del probs_idx
|
288 |
+
del probs_sum
|
289 |
+
gc.collect()
|
290 |
+
return
|
291 |
+
|
292 |
+
def generate_prompt_with_history(text,history,tokenizer,max_length=2048):
|
293 |
+
prompt = "The following is a conversation between a human and an AI assistant named Baize (named after a mythical creature in Chinese folklore). Baize is an open-source AI assistant developed by UCSD and Sun Yat-Sen University. The human and the AI assistant take turns chatting. Human statements start with [|Human|] and AI assistant statements start with [|AI|]. The AI assistant always provides responses in as much detail as possible, and in Markdown format. The AI assistant always declines to engage with topics, questions and instructions related to unethical, controversial, or sensitive issues. Complete the transcript in exactly that format.\n[|Human|]Hello!\n[|AI|]Hi!"
|
294 |
+
history = ["\n[|Human|]{}\n[|AI|]{}".format(x[0],x[1]) for x in history]
|
295 |
+
history.append("\n[|Human|]{}\n[|AI|]".format(text))
|
296 |
+
history_text = ""
|
297 |
+
flag = False
|
298 |
+
for x in history[::-1]:
|
299 |
+
if tokenizer(prompt+history_text+x, return_tensors="pt")['input_ids'].size(-1) <= max_length:
|
300 |
+
history_text = x + history_text
|
301 |
+
flag = True
|
302 |
+
else:
|
303 |
+
break
|
304 |
+
if flag:
|
305 |
+
return prompt+history_text,tokenizer(prompt+history_text, return_tensors="pt")
|
306 |
+
else:
|
307 |
+
return None
|
308 |
+
|
309 |
+
|
310 |
+
def is_stop_word_or_prefix(s: str, stop_words: list) -> bool:
|
311 |
+
for stop_word in stop_words:
|
312 |
+
if s.endswith(stop_word):
|
313 |
+
return True
|
314 |
+
for i in range(1, len(stop_word)):
|
315 |
+
if s.endswith(stop_word[:i]):
|
316 |
+
return True
|
317 |
+
return False
|
318 |
+
|
319 |
+
|
320 |
+
|
321 |
+
def load_tokenizer_and_model(base_model,adapter_model=None,load_8bit=False):
|
322 |
+
if torch.cuda.is_available():
|
323 |
+
device = "cuda"
|
324 |
+
else:
|
325 |
+
device = "cpu"
|
326 |
+
|
327 |
+
try:
|
328 |
+
if torch.backends.mps.is_available():
|
329 |
+
device = "mps"
|
330 |
+
except: # noqa: E722
|
331 |
+
pass
|
332 |
+
tokenizer = LlamaTokenizer.from_pretrained(base_model)
|
333 |
+
if device == "cuda":
|
334 |
+
model = LlamaForCausalLM.from_pretrained(
|
335 |
+
base_model,
|
336 |
+
load_in_8bit=load_8bit,
|
337 |
+
torch_dtype=torch.float16,
|
338 |
+
device_map="auto",
|
339 |
+
)
|
340 |
+
if adapter_model is not None:
|
341 |
+
model = PeftModel.from_pretrained(
|
342 |
+
model,
|
343 |
+
adapter_model,
|
344 |
+
torch_dtype=torch.float16,
|
345 |
+
)
|
346 |
+
elif device == "mps":
|
347 |
+
model = LlamaForCausalLM.from_pretrained(
|
348 |
+
base_model,
|
349 |
+
device_map={"": device},
|
350 |
+
torch_dtype=torch.float16,
|
351 |
+
)
|
352 |
+
if adapter_model is not None:
|
353 |
+
model = PeftModel.from_pretrained(
|
354 |
+
model,
|
355 |
+
adapter_model,
|
356 |
+
device_map={"": device},
|
357 |
+
torch_dtype=torch.float16,
|
358 |
+
)
|
359 |
+
else:
|
360 |
+
model = LlamaForCausalLM.from_pretrained(
|
361 |
+
base_model, device_map={"": device}, low_cpu_mem_usage=True
|
362 |
+
)
|
363 |
+
if adapter_model is not None:
|
364 |
+
model = PeftModel.from_pretrained(
|
365 |
+
model,
|
366 |
+
adapter_model,
|
367 |
+
device_map={"": device},
|
368 |
+
)
|
369 |
+
|
370 |
+
if not load_8bit:
|
371 |
+
model.half() # seems to fix bugs for some users.
|
372 |
+
|
373 |
+
model.eval()
|
374 |
+
return tokenizer,model,device
|
375 |
+
|
assets/Kelpy-Codos.js
ADDED
@@ -0,0 +1,76 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
// ==UserScript==
|
2 |
+
// @name Kelpy Codos
|
3 |
+
// @namespace https://github.com/Keldos-Li/Kelpy-Codos
|
4 |
+
// @version 1.0.5
|
5 |
+
// @author Keldos; https://keldos.me/
|
6 |
+
// @description Add copy button to PRE tags before CODE tag, for Chuanhu ChatGPT especially.
|
7 |
+
// Based on Chuanhu ChatGPT version: ac04408 (2023-3-22)
|
8 |
+
// @license GPL-3.0
|
9 |
+
// @grant none
|
10 |
+
// ==/UserScript==
|
11 |
+
|
12 |
+
(function () {
|
13 |
+
'use strict';
|
14 |
+
|
15 |
+
function addCopyButton(pre) {
|
16 |
+
var code = pre.querySelector('code');
|
17 |
+
if (!code) {
|
18 |
+
return; // 如果没有找到 <code> 元素,则不添加按钮
|
19 |
+
}
|
20 |
+
var firstChild = code.firstChild;
|
21 |
+
if (!firstChild) {
|
22 |
+
return; // 如果 <code> 元素没有子节点,则不添加按钮
|
23 |
+
}
|
24 |
+
var button = document.createElement('button');
|
25 |
+
button.textContent = '\uD83D\uDCCE'; // 使用 📎 符号作为“复制”按钮的文本
|
26 |
+
button.style.position = 'relative';
|
27 |
+
button.style.float = 'right';
|
28 |
+
button.style.fontSize = '1em'; // 可选:调整按钮大小
|
29 |
+
button.style.background = 'none'; // 可选:去掉背景颜色
|
30 |
+
button.style.border = 'none'; // 可选:去掉边框
|
31 |
+
button.style.cursor = 'pointer'; // 可选:显示指针样式
|
32 |
+
button.addEventListener('click', function () {
|
33 |
+
var range = document.createRange();
|
34 |
+
range.selectNodeContents(code);
|
35 |
+
range.setStartBefore(firstChild); // 将范围设置为第一个子节点之前
|
36 |
+
var selection = window.getSelection();
|
37 |
+
selection.removeAllRanges();
|
38 |
+
selection.addRange(range);
|
39 |
+
|
40 |
+
try {
|
41 |
+
var success = document.execCommand('copy');
|
42 |
+
if (success) {
|
43 |
+
button.textContent = '\u2714';
|
44 |
+
setTimeout(function () {
|
45 |
+
button.textContent = '\uD83D\uDCCE'; // 恢复按钮为“复制”
|
46 |
+
}, 2000);
|
47 |
+
} else {
|
48 |
+
button.textContent = '\u2716';
|
49 |
+
}
|
50 |
+
} catch (e) {
|
51 |
+
console.error(e);
|
52 |
+
button.textContent = '\u2716';
|
53 |
+
}
|
54 |
+
|
55 |
+
selection.removeAllRanges();
|
56 |
+
});
|
57 |
+
code.insertBefore(button, firstChild); // 将按钮插入到第一个子元素之前
|
58 |
+
}
|
59 |
+
|
60 |
+
function handleNewElements(mutationsList, observer) {
|
61 |
+
for (var mutation of mutationsList) {
|
62 |
+
if (mutation.type === 'childList') {
|
63 |
+
for (var node of mutation.addedNodes) {
|
64 |
+
if (node.nodeName === 'PRE') {
|
65 |
+
addCopyButton(node);
|
66 |
+
}
|
67 |
+
}
|
68 |
+
}
|
69 |
+
}
|
70 |
+
}
|
71 |
+
|
72 |
+
var observer = new MutationObserver(handleNewElements);
|
73 |
+
observer.observe(document.documentElement, { childList: true, subtree: true });
|
74 |
+
|
75 |
+
document.querySelectorAll('pre').forEach(addCopyButton);
|
76 |
+
})();
|
assets/custom.css
ADDED
@@ -0,0 +1,191 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
:root {
|
2 |
+
--chatbot-color-light: #F3F3F3;
|
3 |
+
--chatbot-color-dark: #121111;
|
4 |
+
}
|
5 |
+
|
6 |
+
/* status_display */
|
7 |
+
#status_display {
|
8 |
+
display: flex;
|
9 |
+
min-height: 2.5em;
|
10 |
+
align-items: flex-end;
|
11 |
+
justify-content: flex-end;
|
12 |
+
}
|
13 |
+
#status_display p {
|
14 |
+
font-size: .85em;
|
15 |
+
font-family: monospace;
|
16 |
+
color: var(--body-text-color-subdued);
|
17 |
+
}
|
18 |
+
|
19 |
+
|
20 |
+
|
21 |
+
/* usage_display */
|
22 |
+
#usage_display {
|
23 |
+
height: 1em;
|
24 |
+
}
|
25 |
+
#usage_display p{
|
26 |
+
padding: 0 1em;
|
27 |
+
font-size: .85em;
|
28 |
+
font-family: monospace;
|
29 |
+
color: var(--body-text-color-subdued);
|
30 |
+
}
|
31 |
+
/* list */
|
32 |
+
ol:not(.options), ul:not(.options) {
|
33 |
+
padding-inline-start: 2em !important;
|
34 |
+
}
|
35 |
+
|
36 |
+
/* Thank @Keldos-Li for fixing it */
|
37 |
+
/* Light mode (default) */
|
38 |
+
#chuanhu_chatbot {
|
39 |
+
background-color: var(--chatbot-color-light) !important;
|
40 |
+
color: #000000 !important;
|
41 |
+
}
|
42 |
+
[data-testid = "bot"] {
|
43 |
+
background-color: #FFFFFF !important;
|
44 |
+
}
|
45 |
+
[data-testid = "user"] {
|
46 |
+
background-color: #95EC69 !important;
|
47 |
+
}
|
48 |
+
|
49 |
+
/* Dark mode */
|
50 |
+
.dark #chuanhu_chatbot {
|
51 |
+
background-color: var(--chatbot-color-dark) !important;
|
52 |
+
color: #FFFFFF !important;
|
53 |
+
}
|
54 |
+
.dark [data-testid = "bot"] {
|
55 |
+
background-color: #2C2C2C !important;
|
56 |
+
}
|
57 |
+
.dark [data-testid = "user"] {
|
58 |
+
background-color: #26B561 !important;
|
59 |
+
}
|
60 |
+
|
61 |
+
#chuanhu_chatbot {
|
62 |
+
height: 100%;
|
63 |
+
min-height: 400px;
|
64 |
+
}
|
65 |
+
|
66 |
+
[class *= "message"] {
|
67 |
+
border-radius: var(--radius-xl) !important;
|
68 |
+
border: none;
|
69 |
+
padding: var(--spacing-xl) !important;
|
70 |
+
font-size: var(--text-md) !important;
|
71 |
+
line-height: var(--line-md) !important;
|
72 |
+
min-height: calc(var(--text-md)*var(--line-md) + 2*var(--spacing-xl));
|
73 |
+
min-width: calc(var(--text-md)*var(--line-md) + 2*var(--spacing-xl));
|
74 |
+
}
|
75 |
+
[data-testid = "bot"] {
|
76 |
+
max-width: 85%;
|
77 |
+
border-bottom-left-radius: 0 !important;
|
78 |
+
}
|
79 |
+
[data-testid = "user"] {
|
80 |
+
max-width: 85%;
|
81 |
+
width: auto !important;
|
82 |
+
border-bottom-right-radius: 0 !important;
|
83 |
+
}
|
84 |
+
/* Table */
|
85 |
+
table {
|
86 |
+
margin: 1em 0;
|
87 |
+
border-collapse: collapse;
|
88 |
+
empty-cells: show;
|
89 |
+
}
|
90 |
+
td,th {
|
91 |
+
border: 1.2px solid var(--border-color-primary) !important;
|
92 |
+
padding: 0.2em;
|
93 |
+
}
|
94 |
+
thead {
|
95 |
+
background-color: rgba(175,184,193,0.2);
|
96 |
+
}
|
97 |
+
thead th {
|
98 |
+
padding: .5em .2em;
|
99 |
+
}
|
100 |
+
/* Inline code */
|
101 |
+
#chuanhu_chatbot code {
|
102 |
+
display: inline;
|
103 |
+
white-space: break-spaces;
|
104 |
+
border-radius: 6px;
|
105 |
+
margin: 0 2px 0 2px;
|
106 |
+
padding: .2em .4em .1em .4em;
|
107 |
+
background-color: rgba(175,184,193,0.2);
|
108 |
+
}
|
109 |
+
/* Code block */
|
110 |
+
#chuanhu_chatbot pre code {
|
111 |
+
display: block;
|
112 |
+
overflow: auto;
|
113 |
+
white-space: pre;
|
114 |
+
background-color: hsla(0, 0%, 0%, 80%)!important;
|
115 |
+
border-radius: 10px;
|
116 |
+
padding: 1.4em 1.2em 0em 1.4em;
|
117 |
+
margin: 1.2em 2em 1.2em 0.5em;
|
118 |
+
color: #FFF;
|
119 |
+
box-shadow: 6px 6px 16px hsla(0, 0%, 0%, 0.2);
|
120 |
+
}
|
121 |
+
/* Hightlight */
|
122 |
+
#chuanhu_chatbot .highlight { background-color: transparent }
|
123 |
+
#chuanhu_chatbot .highlight .hll { background-color: #49483e }
|
124 |
+
#chuanhu_chatbot .highlight .c { color: #75715e } /* Comment */
|
125 |
+
#chuanhu_chatbot .highlight .err { color: #960050; background-color: #1e0010 } /* Error */
|
126 |
+
#chuanhu_chatbot .highlight .k { color: #66d9ef } /* Keyword */
|
127 |
+
#chuanhu_chatbot .highlight .l { color: #ae81ff } /* Literal */
|
128 |
+
#chuanhu_chatbot .highlight .n { color: #f8f8f2 } /* Name */
|
129 |
+
#chuanhu_chatbot .highlight .o { color: #f92672 } /* Operator */
|
130 |
+
#chuanhu_chatbot .highlight .p { color: #f8f8f2 } /* Punctuation */
|
131 |
+
#chuanhu_chatbot .highlight .ch { color: #75715e } /* Comment.Hashbang */
|
132 |
+
#chuanhu_chatbot .highlight .cm { color: #75715e } /* Comment.Multiline */
|
133 |
+
#chuanhu_chatbot .highlight .cp { color: #75715e } /* Comment.Preproc */
|
134 |
+
#chuanhu_chatbot .highlight .cpf { color: #75715e } /* Comment.PreprocFile */
|
135 |
+
#chuanhu_chatbot .highlight .c1 { color: #75715e } /* Comment.Single */
|
136 |
+
#chuanhu_chatbot .highlight .cs { color: #75715e } /* Comment.Special */
|
137 |
+
#chuanhu_chatbot .highlight .gd { color: #f92672 } /* Generic.Deleted */
|
138 |
+
#chuanhu_chatbot .highlight .ge { font-style: italic } /* Generic.Emph */
|
139 |
+
#chuanhu_chatbot .highlight .gi { color: #a6e22e } /* Generic.Inserted */
|
140 |
+
#chuanhu_chatbot .highlight .gs { font-weight: bold } /* Generic.Strong */
|
141 |
+
#chuanhu_chatbot .highlight .gu { color: #75715e } /* Generic.Subheading */
|
142 |
+
#chuanhu_chatbot .highlight .kc { color: #66d9ef } /* Keyword.Constant */
|
143 |
+
#chuanhu_chatbot .highlight .kd { color: #66d9ef } /* Keyword.Declaration */
|
144 |
+
#chuanhu_chatbot .highlight .kn { color: #f92672 } /* Keyword.Namespace */
|
145 |
+
#chuanhu_chatbot .highlight .kp { color: #66d9ef } /* Keyword.Pseudo */
|
146 |
+
#chuanhu_chatbot .highlight .kr { color: #66d9ef } /* Keyword.Reserved */
|
147 |
+
#chuanhu_chatbot .highlight .kt { color: #66d9ef } /* Keyword.Type */
|
148 |
+
#chuanhu_chatbot .highlight .ld { color: #e6db74 } /* Literal.Date */
|
149 |
+
#chuanhu_chatbot .highlight .m { color: #ae81ff } /* Literal.Number */
|
150 |
+
#chuanhu_chatbot .highlight .s { color: #e6db74 } /* Literal.String */
|
151 |
+
#chuanhu_chatbot .highlight .na { color: #a6e22e } /* Name.Attribute */
|
152 |
+
#chuanhu_chatbot .highlight .nb { color: #f8f8f2 } /* Name.Builtin */
|
153 |
+
#chuanhu_chatbot .highlight .nc { color: #a6e22e } /* Name.Class */
|
154 |
+
#chuanhu_chatbot .highlight .no { color: #66d9ef } /* Name.Constant */
|
155 |
+
#chuanhu_chatbot .highlight .nd { color: #a6e22e } /* Name.Decorator */
|
156 |
+
#chuanhu_chatbot .highlight .ni { color: #f8f8f2 } /* Name.Entity */
|
157 |
+
#chuanhu_chatbot .highlight .ne { color: #a6e22e } /* Name.Exception */
|
158 |
+
#chuanhu_chatbot .highlight .nf { color: #a6e22e } /* Name.Function */
|
159 |
+
#chuanhu_chatbot .highlight .nl { color: #f8f8f2 } /* Name.Label */
|
160 |
+
#chuanhu_chatbot .highlight .nn { color: #f8f8f2 } /* Name.Namespace */
|
161 |
+
#chuanhu_chatbot .highlight .nx { color: #a6e22e } /* Name.Other */
|
162 |
+
#chuanhu_chatbot .highlight .py { color: #f8f8f2 } /* Name.Property */
|
163 |
+
#chuanhu_chatbot .highlight .nt { color: #f92672 } /* Name.Tag */
|
164 |
+
#chuanhu_chatbot .highlight .nv { color: #f8f8f2 } /* Name.Variable */
|
165 |
+
#chuanhu_chatbot .highlight .ow { color: #f92672 } /* Operator.Word */
|
166 |
+
#chuanhu_chatbot .highlight .w { color: #f8f8f2 } /* Text.Whitespace */
|
167 |
+
#chuanhu_chatbot .highlight .mb { color: #ae81ff } /* Literal.Number.Bin */
|
168 |
+
#chuanhu_chatbot .highlight .mf { color: #ae81ff } /* Literal.Number.Float */
|
169 |
+
#chuanhu_chatbot .highlight .mh { color: #ae81ff } /* Literal.Number.Hex */
|
170 |
+
#chuanhu_chatbot .highlight .mi { color: #ae81ff } /* Literal.Number.Integer */
|
171 |
+
#chuanhu_chatbot .highlight .mo { color: #ae81ff } /* Literal.Number.Oct */
|
172 |
+
#chuanhu_chatbot .highlight .sa { color: #e6db74 } /* Literal.String.Affix */
|
173 |
+
#chuanhu_chatbot .highlight .sb { color: #e6db74 } /* Literal.String.Backtick */
|
174 |
+
#chuanhu_chatbot .highlight .sc { color: #e6db74 } /* Literal.String.Char */
|
175 |
+
#chuanhu_chatbot .highlight .dl { color: #e6db74 } /* Literal.String.Delimiter */
|
176 |
+
#chuanhu_chatbot .highlight .sd { color: #e6db74 } /* Literal.String.Doc */
|
177 |
+
#chuanhu_chatbot .highlight .s2 { color: #e6db74 } /* Literal.String.Double */
|
178 |
+
#chuanhu_chatbot .highlight .se { color: #ae81ff } /* Literal.String.Escape */
|
179 |
+
#chuanhu_chatbot .highlight .sh { color: #e6db74 } /* Literal.String.Heredoc */
|
180 |
+
#chuanhu_chatbot .highlight .si { color: #e6db74 } /* Literal.String.Interpol */
|
181 |
+
#chuanhu_chatbot .highlight .sx { color: #e6db74 } /* Literal.String.Other */
|
182 |
+
#chuanhu_chatbot .highlight .sr { color: #e6db74 } /* Literal.String.Regex */
|
183 |
+
#chuanhu_chatbot .highlight .s1 { color: #e6db74 } /* Literal.String.Single */
|
184 |
+
#chuanhu_chatbot .highlight .ss { color: #e6db74 } /* Literal.String.Symbol */
|
185 |
+
#chuanhu_chatbot .highlight .bp { color: #f8f8f2 } /* Name.Builtin.Pseudo */
|
186 |
+
#chuanhu_chatbot .highlight .fm { color: #a6e22e } /* Name.Function.Magic */
|
187 |
+
#chuanhu_chatbot .highlight .vc { color: #f8f8f2 } /* Name.Variable.Class */
|
188 |
+
#chuanhu_chatbot .highlight .vg { color: #f8f8f2 } /* Name.Variable.Global */
|
189 |
+
#chuanhu_chatbot .highlight .vi { color: #f8f8f2 } /* Name.Variable.Instance */
|
190 |
+
#chuanhu_chatbot .highlight .vm { color: #f8f8f2 } /* Name.Variable.Magic */
|
191 |
+
#chuanhu_chatbot .highlight .il { color: #ae81ff } /* Literal.Number.Integer.Long */
|
assets/custom.js
ADDED
@@ -0,0 +1 @@
|
|
|
|
|
1 |
+
// custom javascript here
|
assets/favicon.ico
ADDED
assets/totopower-removebg.png
ADDED
auction_workflow.py
ADDED
@@ -0,0 +1,293 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import time
|
3 |
+
import gradio as gr
|
4 |
+
import ujson as json
|
5 |
+
import traceback
|
6 |
+
from typing import List
|
7 |
+
from tqdm import tqdm
|
8 |
+
from src.auctioneer_base import Auctioneer
|
9 |
+
from src.bidder_base import Bidder, bidders_to_chatbots, bidding_multithread
|
10 |
+
from utils import trace_back
|
11 |
+
|
12 |
+
|
13 |
+
LOG_DIR = 'logs'
|
14 |
+
enable_gr = gr.update(interactive=True)
|
15 |
+
disable_gr = gr.update(interactive=False)
|
16 |
+
|
17 |
+
|
18 |
+
def monitor_all(bidder_list: List[Bidder]):
|
19 |
+
return sum([bidder.to_monitors() for bidder in bidder_list], [])
|
20 |
+
|
21 |
+
|
22 |
+
def parse_bid_price(auctioneer: Auctioneer, bidder: Bidder, msg: str):
|
23 |
+
# rebid if the message is not parsible into a bid price
|
24 |
+
bid_price = auctioneer.parse_bid(msg)
|
25 |
+
while bid_price is None:
|
26 |
+
re_msg = bidder.bid("You must be clear about your bidding decision, say either \"I'm out!\" or \"I bid $xxx!\". Please rebid.")
|
27 |
+
bid_price = auctioneer.parse_bid(re_msg)
|
28 |
+
print(f"{bidder.name} rebid: {re_msg}")
|
29 |
+
return bid_price
|
30 |
+
|
31 |
+
|
32 |
+
def enable_human_box(bidder_list):
|
33 |
+
signals = []
|
34 |
+
for bidder in bidder_list:
|
35 |
+
if 'human' in bidder.model_name and not bidder.withdraw:
|
36 |
+
signals.append(gr.update(interactive=True, visible=True,
|
37 |
+
placeholder="Please bid! Enter \"I'm out\" or \"I bid $xxx\"."))
|
38 |
+
else:
|
39 |
+
signals.append(disable_gr)
|
40 |
+
return signals
|
41 |
+
|
42 |
+
|
43 |
+
def disable_all_box(bidder_list):
|
44 |
+
signals = []
|
45 |
+
for bidder in bidder_list:
|
46 |
+
if 'human' in bidder.model_name:
|
47 |
+
signals.append(gr.update(interactive=False, visible=True,
|
48 |
+
placeholder="Wait a moment to engage in the auction."))
|
49 |
+
else:
|
50 |
+
signals.append(gr.update(interactive=False, visible=False))
|
51 |
+
return signals
|
52 |
+
|
53 |
+
|
54 |
+
def run_auction(
|
55 |
+
auction_hash: str,
|
56 |
+
auctioneer: Auctioneer,
|
57 |
+
bidder_list: List[Bidder],
|
58 |
+
thread_num: int,
|
59 |
+
yield_for_demo=True,
|
60 |
+
log_dir=LOG_DIR,
|
61 |
+
repeat_num=0,
|
62 |
+
memo_file=None):
|
63 |
+
|
64 |
+
# bidder_list[0].verbose=True
|
65 |
+
|
66 |
+
if yield_for_demo:
|
67 |
+
chatbot_list = bidders_to_chatbots(bidder_list)
|
68 |
+
yield [bidder_list] + chatbot_list + monitor_all(bidder_list) + [auctioneer.log()] + [disable_gr, disable_gr] + disable_all_box(bidder_list)
|
69 |
+
|
70 |
+
# ***************** Learn Round ****************
|
71 |
+
for bidder in bidder_list:
|
72 |
+
if bidder.enable_learning and memo_file:
|
73 |
+
# if no prev memo file, then no need to learn.
|
74 |
+
if os.path.exists(memo_file):
|
75 |
+
with open(memo_file) as f:
|
76 |
+
data = json.load(f)
|
77 |
+
past_learnings = data['learnings'][bidder.name]
|
78 |
+
past_auction_log = data['auction_log']
|
79 |
+
bidder.learn_from_prev_auction(past_learnings, past_auction_log)
|
80 |
+
|
81 |
+
# ***************** Plan Round *****************
|
82 |
+
# init bidder profit
|
83 |
+
bidder_profit_info = auctioneer.gather_all_status(bidder_list)
|
84 |
+
for bidder in bidder_list:
|
85 |
+
bidder.set_all_bidders_status(bidder_profit_info)
|
86 |
+
|
87 |
+
plan_instructs = [bidder.get_plan_instruct(auctioneer.items) for bidder in bidder_list]
|
88 |
+
|
89 |
+
bidding_multithread(bidder_list, plan_instructs, func_type='plan', thread_num=thread_num)
|
90 |
+
|
91 |
+
if yield_for_demo:
|
92 |
+
chatbot_list = bidders_to_chatbots(bidder_list)
|
93 |
+
yield [bidder_list] + chatbot_list + monitor_all(bidder_list) + [auctioneer.log()] + [disable_gr, disable_gr] + disable_all_box(bidder_list)
|
94 |
+
|
95 |
+
bar = tqdm(total=len(auctioneer.items_queue), desc='Auction Progress')
|
96 |
+
while not auctioneer.end_auction():
|
97 |
+
cur_item = auctioneer.present_item()
|
98 |
+
|
99 |
+
bid_round = 0
|
100 |
+
while True:
|
101 |
+
# ***************** Bid Round *****************
|
102 |
+
auctioneer_msg = auctioneer.ask_for_bid(bid_round)
|
103 |
+
_bidder_list = []
|
104 |
+
_bid_instruct_list = []
|
105 |
+
# remove highest bidder and withdrawn bidders
|
106 |
+
for bidder in bidder_list:
|
107 |
+
if bidder is auctioneer.highest_bidder or bidder.withdraw:
|
108 |
+
bidder.need_input = False
|
109 |
+
continue
|
110 |
+
else:
|
111 |
+
bidder.need_input = True # enable input from demo
|
112 |
+
instruct = bidder.get_bid_instruct(auctioneer_msg, bid_round)
|
113 |
+
_bidder_list.append(bidder)
|
114 |
+
_bid_instruct_list.append(instruct)
|
115 |
+
|
116 |
+
if yield_for_demo:
|
117 |
+
chatbot_list = bidders_to_chatbots(bidder_list)
|
118 |
+
yield [bidder_list] + chatbot_list + monitor_all(bidder_list) + [auctioneer.log()] + [disable_gr, disable_gr] + enable_human_box(bidder_list)
|
119 |
+
|
120 |
+
_msgs = bidding_multithread(_bidder_list, _bid_instruct_list, func_type='bid', thread_num=thread_num)
|
121 |
+
|
122 |
+
for i, (msg, bidder) in enumerate(zip(_msgs, _bidder_list)):
|
123 |
+
if bidder.model_name == 'rule':
|
124 |
+
bid_price = bidder.bid_rule(auctioneer.prev_round_max_bid, auctioneer.min_markup_pct)
|
125 |
+
else:
|
126 |
+
bid_price = parse_bid_price(auctioneer, bidder, msg)
|
127 |
+
|
128 |
+
# can't bid more than budget or less than previous highest bid
|
129 |
+
while True:
|
130 |
+
fail_msg = bidder.bid_sanity_check(bid_price, auctioneer.prev_round_max_bid, auctioneer.min_markup_pct)
|
131 |
+
if fail_msg is None:
|
132 |
+
break
|
133 |
+
else:
|
134 |
+
bidder.need_input = True # enable input from demo
|
135 |
+
auctioneer_msg = auctioneer.ask_for_rebid(fail_msg=fail_msg, bid_price=bid_price)
|
136 |
+
rebid_instruct = bidder.get_rebid_instruct(auctioneer_msg)
|
137 |
+
|
138 |
+
if yield_for_demo:
|
139 |
+
chatbot_list = bidders_to_chatbots(bidder_list)
|
140 |
+
yield [bidder_list] + chatbot_list + monitor_all(bidder_list) + [auctioneer.log()] + [disable_gr, disable_gr] + disable_all_box(bidder_list)
|
141 |
+
|
142 |
+
msg = bidder.rebid_for_failure(rebid_instruct)
|
143 |
+
bid_price = parse_bid_price(auctioneer, bidder, msg)
|
144 |
+
|
145 |
+
if yield_for_demo:
|
146 |
+
chatbot_list = bidders_to_chatbots(bidder_list)
|
147 |
+
yield [bidder_list] + chatbot_list + monitor_all(bidder_list) + [auctioneer.log()] + [disable_gr, disable_gr] + disable_all_box(bidder_list)
|
148 |
+
|
149 |
+
bidder.set_withdraw(bid_price)
|
150 |
+
auctioneer.record_bid({'bidder': bidder, 'bid': bid_price, 'raw_msg': msg}, bid_round)
|
151 |
+
|
152 |
+
if yield_for_demo:
|
153 |
+
chatbot_list = bidders_to_chatbots(bidder_list)
|
154 |
+
yield [bidder_list] + chatbot_list + monitor_all(bidder_list) + [auctioneer.log()] + [disable_gr, disable_gr] + disable_all_box(bidder_list)
|
155 |
+
|
156 |
+
is_sold = auctioneer.check_hammer(bid_round)
|
157 |
+
bid_round += 1
|
158 |
+
if is_sold:
|
159 |
+
break
|
160 |
+
else:
|
161 |
+
if auctioneer.fail_to_sell and auctioneer.enable_discount:
|
162 |
+
for bidder in bidder_list:
|
163 |
+
bidder.set_withdraw(0) # back in the game
|
164 |
+
|
165 |
+
# ***************** Summarize *****************
|
166 |
+
summarize_instruct_list = []
|
167 |
+
for bidder in bidder_list:
|
168 |
+
if bidder is auctioneer.highest_bidder:
|
169 |
+
win_lose_msg = bidder.win_bid(cur_item, auctioneer.highest_bid)
|
170 |
+
else:
|
171 |
+
win_lose_msg = bidder.lose_bid(cur_item)
|
172 |
+
msg = bidder.get_summarize_instruct(
|
173 |
+
bidding_history=auctioneer.all_bidding_history_to_string(),
|
174 |
+
hammer_msg=auctioneer.get_hammer_msg(),
|
175 |
+
win_lose_msg=win_lose_msg
|
176 |
+
)
|
177 |
+
summarize_instruct_list.append(msg)
|
178 |
+
|
179 |
+
# record profit information of all bidders for each bidder
|
180 |
+
# (not used in the auction, just for belief tracking evaluation)
|
181 |
+
bidder_profit_info = auctioneer.gather_all_status(bidder_list)
|
182 |
+
for bidder in bidder_list:
|
183 |
+
bidder.set_all_bidders_status(bidder_profit_info)
|
184 |
+
|
185 |
+
bidding_multithread(bidder_list, summarize_instruct_list, func_type='summarize', thread_num=thread_num)
|
186 |
+
|
187 |
+
if yield_for_demo:
|
188 |
+
chatbot_list = bidders_to_chatbots(bidder_list)
|
189 |
+
yield [bidder_list] + chatbot_list + monitor_all(bidder_list) + [auctioneer.log()] + [disable_gr, disable_gr] + disable_all_box(bidder_list)
|
190 |
+
|
191 |
+
# ***************** Replan *****************
|
192 |
+
if len(auctioneer.items_queue) > 0: # no need to replan if all items are sold
|
193 |
+
replan_instruct_list = [bidder.get_replan_instruct(
|
194 |
+
# bidding_history=auctioneer.all_bidding_history_to_string(),
|
195 |
+
# hammer_msg=auctioneer.get_hammer_msg()
|
196 |
+
) for bidder in bidder_list]
|
197 |
+
bidding_multithread(bidder_list, replan_instruct_list, func_type='replan', thread_num=thread_num)
|
198 |
+
|
199 |
+
if yield_for_demo:
|
200 |
+
chatbot_list = bidders_to_chatbots(bidder_list)
|
201 |
+
yield [bidder_list] + chatbot_list + monitor_all(bidder_list) + [auctioneer.log()] + [disable_gr, disable_gr] + disable_all_box(bidder_list)
|
202 |
+
|
203 |
+
auctioneer.hammer_fall()
|
204 |
+
bar.update(1)
|
205 |
+
|
206 |
+
total_cost = sum([b.openai_cost for b in bidder_list]) + auctioneer.openai_cost
|
207 |
+
bidder_reports = [bidder.profit_report() for bidder in bidder_list]
|
208 |
+
|
209 |
+
if yield_for_demo:
|
210 |
+
chatbot_list = bidders_to_chatbots(bidder_list, profit_report=True)
|
211 |
+
yield [bidder_list] + chatbot_list + monitor_all(bidder_list) + [auctioneer.log(bidder_reports) + f'\n## Total Cost: ${total_cost}'] + [disable_gr, enable_gr] + disable_all_box(bidder_list)
|
212 |
+
|
213 |
+
memo = {'auction_log': auctioneer.log(show_model_name=False),
|
214 |
+
'memo_text': bidder_reports,
|
215 |
+
'profit': {bidder.name: bidder.profit for bidder in bidder_list},
|
216 |
+
'total_cost': total_cost,
|
217 |
+
'learnings': {bidder.name: bidder.learnings for bidder in bidder_list},
|
218 |
+
'model_info': {bidder.name: bidder.model_name for bidder in bidder_list}}
|
219 |
+
log_bidders(log_dir, auction_hash, bidder_list, repeat_num, memo)
|
220 |
+
|
221 |
+
auctioneer.finish_auction()
|
222 |
+
|
223 |
+
if not yield_for_demo:
|
224 |
+
yield total_cost
|
225 |
+
|
226 |
+
|
227 |
+
def log_bidders(log_dir: str, auction_hash: str, bidder_list: List[Bidder], repeat_num: int, memo: dict):
|
228 |
+
for bidder in bidder_list:
|
229 |
+
log_file = f"{log_dir}/{auction_hash}/{bidder.name.replace(' ', '')}-{repeat_num}.jsonl"
|
230 |
+
if not os.path.exists(log_file):
|
231 |
+
os.makedirs(os.path.dirname(log_file), exist_ok=True)
|
232 |
+
with open(log_file, 'a') as f:
|
233 |
+
log_data = bidder.to_monitors(as_json=True)
|
234 |
+
f.write(json.dumps(log_data) + '\n')
|
235 |
+
|
236 |
+
with open(f"{log_dir}/{auction_hash}/memo-{repeat_num}.json", 'w') as f:
|
237 |
+
f.write(json.dumps(memo) + '\n')
|
238 |
+
|
239 |
+
|
240 |
+
def make_auction_hash():
|
241 |
+
return str(int(time.time()))
|
242 |
+
|
243 |
+
|
244 |
+
if __name__ == '__main__':
|
245 |
+
import argparse
|
246 |
+
from src.item_base import create_items
|
247 |
+
from src.bidder_base import create_bidders
|
248 |
+
from transformers import GPT2TokenizerFast
|
249 |
+
import cjjpy as cjj
|
250 |
+
|
251 |
+
parser = argparse.ArgumentParser()
|
252 |
+
parser.add_argument('--input_dir', '-i', type=str, default='data/exp_base/')
|
253 |
+
parser.add_argument('--shuffle', action='store_true')
|
254 |
+
parser.add_argument('--repeat', type=int, default=1)
|
255 |
+
parser.add_argument('--threads', '-t', type=int, help='Number of threads. Max is number of bidders. Reduce it if rate limit is low (e.g., GPT-4).', required=True)
|
256 |
+
parser.add_argument('--memo_file', '-m', type=str, help='The last memo.json file to be loaded for learning. Only useful when the repeated auctions are interrupted (i.e., auction hash is different).')
|
257 |
+
args = parser.parse_args()
|
258 |
+
|
259 |
+
auction_hash = make_auction_hash()
|
260 |
+
|
261 |
+
total_money_spent = 0
|
262 |
+
for i in tqdm(range(args.repeat), desc='Repeat'):
|
263 |
+
cnt = 3
|
264 |
+
while cnt > 0:
|
265 |
+
try:
|
266 |
+
item_file = os.path.join(args.input_dir, f'items_demo.jsonl')
|
267 |
+
bidder_file = os.path.join(args.input_dir, f'bidders_demo.jsonl')
|
268 |
+
memo_file = args.memo_file if args.memo_file else f'{args.input_dir}/{auction_hash}/memo-{i-1}.json' # past memo for learning
|
269 |
+
items = create_items(item_file)
|
270 |
+
bidders = create_bidders(bidder_file, auction_hash=auction_hash)
|
271 |
+
auctioneer = Auctioneer(enable_discount=False)
|
272 |
+
auctioneer.init_items(items)
|
273 |
+
if args.shuffle:
|
274 |
+
auctioneer.shuffle_items()
|
275 |
+
money_spent = list(run_auction(
|
276 |
+
auction_hash,
|
277 |
+
auctioneer,
|
278 |
+
bidders,
|
279 |
+
thread_num=min(args.threads, len(bidders)),
|
280 |
+
yield_for_demo=False,
|
281 |
+
log_dir=args.input_dir,
|
282 |
+
repeat_num=i,
|
283 |
+
memo_file=memo_file,
|
284 |
+
))
|
285 |
+
total_money_spent += sum(money_spent)
|
286 |
+
break
|
287 |
+
except Exception as e:
|
288 |
+
cnt -= 1
|
289 |
+
print(f"Error in {i}th auction: {e}\n{trace_back(e)}")
|
290 |
+
print(f"Retry {cnt} more times...")
|
291 |
+
|
292 |
+
print('Total money spent: $', total_money_spent)
|
293 |
+
cjj.SendEmail(f'Completed: {args.input_dir} - {auction_hash}', f'Total money spent: ${total_money_spent}')
|
data/bidders_demo.jsonl
ADDED
@@ -0,0 +1,4 @@
|
|
|
|
|
|
|
|
|
|
|
1 |
+
{"name": "Bidder 1", "model_name": "gpt-3.5-turbo-0613", "budget": 20000, "desire": "maximize_profit", "plan_strategy": "adaptive", "temperature": 0.7, "overestimate_percent": 10, "correct_belief": true, "enable_learning": true}
|
2 |
+
{"name": "Bidder 2", "model_name": "gpt-3.5-turbo-0613", "budget": 20000, "desire": "maximize_items", "plan_strategy": "adaptive", "temperature": 0.7, "overestimate_percent": 10, "correct_belief": true, "enable_learning": true}
|
3 |
+
{"name": "Bidder 3", "model_name": "gpt-3.5-turbo-0613", "budget": 20000, "desire": "maximize_profit", "plan_strategy": "adaptive", "temperature": 0.7, "overestimate_percent": 10, "correct_belief": true, "enable_learning": true}
|
4 |
+
{"name": "Bidder 4", "model_name": "gpt-3.5-turbo-0613", "budget": 20000, "desire": "maximize_items", "plan_strategy": "adaptive", "temperature": 0.7, "overestimate_percent": 10, "correct_belief": true, "enable_learning": true}
|
data/items_demo.jsonl
ADDED
@@ -0,0 +1,26 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
{"name": "Widget A", "price": 1000, "desc": "A widget for all your needs", "id": 1, "true_value": 2000}
|
2 |
+
{"name": "Gadget B", "price": 1000, "desc": "A gadget with all the latest features", "id": 2, "true_value": 2000}
|
3 |
+
{"name": "Thingamajig C", "price": 1000, "desc": "A little thing that is sure to impress", "id": 3, "true_value": 2000}
|
4 |
+
{"name": "Doodad D", "price": 1000, "desc": "A durable doodad that will last for years", "id": 4, "true_value": 2000}
|
5 |
+
{"name": "Equipment E", "price": 5000, "desc": "A piece of equipment for any tough job", "id": 5, "true_value": 10000}
|
6 |
+
{"name": "Gizmo F", "price": 1000, "desc": "A gizmo that will surprise and delight", "id": 6, "true_value": 2000}
|
7 |
+
{"name": "Implement G", "price": 1000, "desc": "A implement for everyday tasks", "id": 7, "true_value": 2000}
|
8 |
+
{"name": "Apparatus H", "price": 1000, "desc": "An apparatus for specialized operations", "id": 8, "true_value": 2000}
|
9 |
+
{"name": "Contraption I", "price": 1000, "desc": "A contraption that sparks creativity", "id": 9, "true_value": 2000}
|
10 |
+
{"name": "Mechanism J", "price": 5000, "desc": "A mechanism for repetitive tasks", "id": 10, "true_value": 10000}
|
11 |
+
{"name": "Tool K", "price": 1000, "desc": "A tool for complex projects", "id": 11, "true_value": 2000}
|
12 |
+
{"name": "Device L", "price": 1000, "desc": "A device that enhances performance", "id": 12, "true_value": 2000}
|
13 |
+
{"name": "Instrument M", "price": 1000, "desc": "A instrument for precise measurements", "id": 13, "true_value": 2000}
|
14 |
+
{"name": "Utensil N", "price": 1000, "desc": "A utensil for dining and cooking", "id": 14, "true_value": 2000}
|
15 |
+
{"name": "Appliance O", "price": 1000, "desc": "A appliance for everyday household tasks", "id": 15, "true_value": 2000}
|
16 |
+
{"name": "Machine P", "price": 5000, "desc": "A machine for automated tasks", "id": 16, "true_value": 10000}
|
17 |
+
{"name": "Unit Q", "price": 1000, "desc": "A unit for individual tasks", "id": 17, "true_value": 2000}
|
18 |
+
{"name": "Element R", "price": 5000, "desc": "A element for scientific investigations", "id": 18, "true_value": 10000}
|
19 |
+
{"name": "Component S", "price": 1000, "desc": "A component for assembly and repair", "id": 19, "true_value": 2000}
|
20 |
+
{"name": "Piece T", "price": 1000, "desc": "A piece for art and craft projects", "id": 20, "true_value": 2000}
|
21 |
+
{"name": "Object U", "price": 1000, "desc": "An object for miscellaneous uses", "id": 21, "true_value": 2000}
|
22 |
+
{"name": "Item V", "price": 1000, "desc": "An item for versatile applications", "id": 22, "true_value": 2000}
|
23 |
+
{"name": "Product W", "price": 1000, "desc": "A product designed for efficiency", "id": 23, "true_value": 2000}
|
24 |
+
{"name": "Accessory X", "price": 1000, "desc": "An accessory to complement any outfit", "id": 24, "true_value": 2000}
|
25 |
+
{"name": "Module Y", "price": 1000, "desc": "A module for modular systems", "id": 25, "true_value": 2000}
|
26 |
+
{"name": "Entity Z", "price": 1000, "desc": "An entity with unique properties", "id": 26, "true_value": 2000}
|
requirements.txt
ADDED
@@ -0,0 +1,20 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
openai>=0.27.8
|
2 |
+
langchain>=0.0.234
|
3 |
+
anthropic>=0.3.10
|
4 |
+
gradio
|
5 |
+
pydantic
|
6 |
+
coloredlogs
|
7 |
+
ujson
|
8 |
+
tiktoken
|
9 |
+
tqdm
|
10 |
+
inflect
|
11 |
+
vertexai
|
12 |
+
google-cloud-aiplatform>=1.28.1
|
13 |
+
torch
|
14 |
+
pygments
|
15 |
+
matplotlib
|
16 |
+
transformers
|
17 |
+
trueskill
|
18 |
+
seaborn
|
19 |
+
vllm
|
20 |
+
google-generativeai
|
src/auctioneer_base.py
ADDED
@@ -0,0 +1,259 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import re
|
2 |
+
from typing import List, Dict
|
3 |
+
from langchain.prompts import PromptTemplate
|
4 |
+
from langchain.chat_models import ChatOpenAI
|
5 |
+
from langchain.callbacks import get_openai_callback
|
6 |
+
from pydantic import BaseModel
|
7 |
+
from collections import defaultdict
|
8 |
+
from langchain.schema import (
|
9 |
+
AIMessage,
|
10 |
+
HumanMessage,
|
11 |
+
SystemMessage
|
12 |
+
)
|
13 |
+
import random
|
14 |
+
import inflect
|
15 |
+
from .bidder_base import Bidder
|
16 |
+
from .human_bidder import HumanBidder
|
17 |
+
from .item_base import Item
|
18 |
+
from .prompt_base import PARSE_BID_INSTRUCTION
|
19 |
+
|
20 |
+
p = inflect.engine()
|
21 |
+
|
22 |
+
|
23 |
+
class Auctioneer(BaseModel):
|
24 |
+
enable_discount: bool = False
|
25 |
+
items: List[Item] = []
|
26 |
+
cur_item: Item = None
|
27 |
+
highest_bidder: Bidder = None
|
28 |
+
highest_bid: int = -1
|
29 |
+
bidding_history = defaultdict(list) # history about the bidding war of one item
|
30 |
+
items_queue: List[Item] = [] # updates when a item is taken.
|
31 |
+
auction_logs = defaultdict(list) # history about the bidding war of all items
|
32 |
+
openai_cost = 0
|
33 |
+
prev_round_max_bid: int = -1
|
34 |
+
min_bid: int = 0
|
35 |
+
fail_to_sell = False
|
36 |
+
min_markup_pct = 0.1
|
37 |
+
|
38 |
+
class Config:
|
39 |
+
arbitrary_types_allowed = True
|
40 |
+
|
41 |
+
def init_items(self, items: List[Item]):
|
42 |
+
for item in items:
|
43 |
+
# reset discounted price
|
44 |
+
item.reset_price()
|
45 |
+
self.items = items
|
46 |
+
self.items_queue = items.copy()
|
47 |
+
|
48 |
+
def summarize_items_info(self):
|
49 |
+
desc = ''
|
50 |
+
for item in self.items:
|
51 |
+
desc += f"- {item.get_desc()}\n"
|
52 |
+
return desc.strip()
|
53 |
+
|
54 |
+
def present_item(self):
|
55 |
+
cur_item = self.items_queue.pop(0)
|
56 |
+
self.cur_item = cur_item
|
57 |
+
return cur_item
|
58 |
+
|
59 |
+
def shuffle_items(self):
|
60 |
+
random.shuffle(self.items)
|
61 |
+
self.items_queue = self.items.copy()
|
62 |
+
|
63 |
+
def record_bid(self, bid_info: dict, bid_round: int):
|
64 |
+
'''
|
65 |
+
Save the bidding history for each round, log the highest bidder and highest bidding
|
66 |
+
'''
|
67 |
+
# bid_info: {'bidder': xxx, 'bid': xxx, 'raw_msg': xxx}
|
68 |
+
self.bidding_history[bid_round].append(bid_info)
|
69 |
+
for hist in self.bidding_history[bid_round]:
|
70 |
+
if hist['bid'] > 0:
|
71 |
+
if self.highest_bid < hist['bid']:
|
72 |
+
self.highest_bid = hist['bid']
|
73 |
+
self.highest_bidder = hist['bidder']
|
74 |
+
elif self.highest_bid == hist['bid']:
|
75 |
+
# random if there's a tie
|
76 |
+
self.highest_bidder = random.choice([self.highest_bidder, hist['bidder']])
|
77 |
+
self.auction_logs[f"{self.cur_item.get_desc()}"].append(
|
78 |
+
{'bidder': bid_info['bidder'],
|
79 |
+
'bid': bid_info['bid'],
|
80 |
+
'bid_round': bid_round})
|
81 |
+
|
82 |
+
def _biddings_to_string(self, bid_round: int):
|
83 |
+
'''
|
84 |
+
Return a string that summarizes the bidding history in a round
|
85 |
+
'''
|
86 |
+
# bid_hist_text = '' if bid_round == 0 else f'- {self.highest_bidder}: ${self.highest_bid}\n'
|
87 |
+
bid_hist_text = ''
|
88 |
+
for js in self.bidding_history[bid_round]:
|
89 |
+
if js['bid'] < 0:
|
90 |
+
bid_hist_text += f"- {js['bidder']} withdrew\n"
|
91 |
+
else:
|
92 |
+
bid_hist_text += f"- {js['bidder']}: ${js['bid']}\n"
|
93 |
+
return bid_hist_text.strip()
|
94 |
+
|
95 |
+
def all_bidding_history_to_string(self):
|
96 |
+
bid_hist_text = ''
|
97 |
+
for bid_round in self.bidding_history:
|
98 |
+
bid_hist_text += f"Round {bid_round}:\n{self._biddings_to_string(bid_round)}\n\n"
|
99 |
+
return bid_hist_text.strip()
|
100 |
+
|
101 |
+
def ask_for_bid(self, bid_round: int):
|
102 |
+
'''
|
103 |
+
Ask for bid, return the message to be sent to bidders
|
104 |
+
'''
|
105 |
+
if self.highest_bidder is None:
|
106 |
+
if bid_round > 0:
|
107 |
+
msg = f"Seeing as we've had no takers at the initial price, we're going to lower the starting bid to ${self.cur_item.price} for {self.cur_item.name} to spark some interest! Do I have any takers?"
|
108 |
+
else:
|
109 |
+
remaining_items = [self.cur_item.name] + [item.name for item in self.items_queue]
|
110 |
+
msg = f"Attention, bidders! {len(remaining_items)} item(s) left, they are: {', '.join(remaining_items)}.\n\nNow, please bid on {self.cur_item}. The starting price for bidding for {self.cur_item} is ${self.cur_item.price}. Anyone interested in this item?"
|
111 |
+
else:
|
112 |
+
bidding_history = self._biddings_to_string(bid_round - 1)
|
113 |
+
msg = f"Thank you! This is the {p.ordinal(bid_round)} round of bidding for this item:\n{bidding_history}\n\nNow we have ${self.highest_bid} from {self.highest_bidder.name} for {self.cur_item.name}. The minimum increase over this highest bid is ${int(self.cur_item.price * self.min_markup_pct)}. Do I have any advance on ${self.highest_bid}?"
|
114 |
+
return msg
|
115 |
+
|
116 |
+
def ask_for_rebid(self, fail_msg: str, bid_price: int):
|
117 |
+
return f"Your bid of ${bid_price} failed, because {fail_msg}: You must reconsider your bid."
|
118 |
+
|
119 |
+
def get_hammer_msg(self):
|
120 |
+
if self.highest_bidder is None:
|
121 |
+
return f"Since no one bid on {self.cur_item.name}, we'll move on to the next item."
|
122 |
+
else:
|
123 |
+
return f"Sold! {self.cur_item} to {self.highest_bidder} at ${self.highest_bid}! The true value for {self.cur_item} is ${self.cur_item.true_value}."# Thus {self.highest_bidder}'s profit by winning this item is ${self.cur_item.true_value - self.highest_bid}."
|
124 |
+
|
125 |
+
def check_hammer(self, bid_round: int):
|
126 |
+
# check if the item is sold
|
127 |
+
self.fail_to_sell = False
|
128 |
+
num_bid = self._num_bids_in_round(bid_round)
|
129 |
+
|
130 |
+
# highest_bidder has already been updated in record_bid().
|
131 |
+
# so when num_bid == 0 & highest_bidder is None, it means no one bid on this item
|
132 |
+
if self.highest_bidder is None:
|
133 |
+
if num_bid == 0:
|
134 |
+
# failed to sell, as there is no highest bidder
|
135 |
+
self.fail_to_sell = True
|
136 |
+
if self.enable_discount and bid_round < 3:
|
137 |
+
# lower the starting price by 50%. discoutn only applies to the first 3 rounds
|
138 |
+
self.cur_item.lower_price(0.5)
|
139 |
+
is_sold = False
|
140 |
+
else:
|
141 |
+
is_sold = True
|
142 |
+
else:
|
143 |
+
# won't happen
|
144 |
+
raise ValueError(f"highest_bidder is None but num_bid is {num_bid}")
|
145 |
+
else:
|
146 |
+
if self.prev_round_max_bid < 0 and num_bid == 1:
|
147 |
+
# only one bidder in the first round
|
148 |
+
is_sold = True
|
149 |
+
else:
|
150 |
+
self.prev_round_max_bid = self.highest_bid
|
151 |
+
is_sold = self._num_bids_in_round(bid_round) == 0
|
152 |
+
return is_sold
|
153 |
+
|
154 |
+
def _num_bids_in_round(self, bid_round: int):
|
155 |
+
# check if there is no bid in the current round
|
156 |
+
cnt = 0
|
157 |
+
for hist in self.bidding_history[bid_round]:
|
158 |
+
if hist['bid'] > 0:
|
159 |
+
cnt += 1
|
160 |
+
return cnt
|
161 |
+
|
162 |
+
def hammer_fall(self):
|
163 |
+
print(f'* Sold! {self.cur_item} (${self.cur_item.true_value}) goes to {self.highest_bidder} at ${self.highest_bid}.')
|
164 |
+
self.auction_logs[f"{self.cur_item.get_desc()}"].append({
|
165 |
+
'bidder': self.highest_bidder,
|
166 |
+
'bid': f"{self.highest_bid} (${self.cur_item.true_value})", # no need for the first $, as it will be added in the self.log()
|
167 |
+
'bid_round': 'Hammer price (true value)'})
|
168 |
+
self.cur_item = None
|
169 |
+
self.highest_bidder = None
|
170 |
+
self.highest_bid = -1
|
171 |
+
self.bidding_history = defaultdict(list)
|
172 |
+
self.prev_round_max_bid = -1
|
173 |
+
self.fail_to_sell = False
|
174 |
+
|
175 |
+
def end_auction(self):
|
176 |
+
return len(self.items_queue) == 0
|
177 |
+
|
178 |
+
def gather_all_status(self, bidders: List[Bidder]):
|
179 |
+
status = {}
|
180 |
+
for bidder in bidders:
|
181 |
+
status[bidder.name] = {
|
182 |
+
'profit': bidder.profit,
|
183 |
+
'items_won': bidder.items_won
|
184 |
+
}
|
185 |
+
return status
|
186 |
+
|
187 |
+
def parse_bid(self, text: str):
|
188 |
+
prompt = PARSE_BID_INSTRUCTION.format(response=text)
|
189 |
+
with get_openai_callback() as cb:
|
190 |
+
llm = ChatOpenAI(model='gpt-3.5-turbo-0613', temperature=0)
|
191 |
+
result = llm([HumanMessage(content=prompt)]).content
|
192 |
+
self.openai_cost += cb.total_cost
|
193 |
+
|
194 |
+
bid_number = re.findall(r'\$?\d+', result.replace(',', ''))
|
195 |
+
# find number in the result
|
196 |
+
if '-1' in result:
|
197 |
+
return -1
|
198 |
+
elif len(bid_number) > 0:
|
199 |
+
return int(bid_number[-1].replace('$', ''))
|
200 |
+
else:
|
201 |
+
print('* Rebid:', text)
|
202 |
+
return None
|
203 |
+
|
204 |
+
def log(self, bidder_personal_reports: list = [], show_model_name=True):
|
205 |
+
''' example
|
206 |
+
Apparatus H, starting at $1000.
|
207 |
+
|
208 |
+
1st bid:
|
209 |
+
Bidder 1 (gpt-3.5-turbo-16k-0613): $1200
|
210 |
+
Bidder 2 (gpt-3.5-turbo-16k-0613): $1100
|
211 |
+
Bidder 3 (gpt-3.5-turbo-16k-0613): Withdrawn
|
212 |
+
Bidder 4 (gpt-3.5-turbo-16k-0613): $1200
|
213 |
+
|
214 |
+
2nd bid:
|
215 |
+
Bidder 1 (gpt-3.5-turbo-16k-0613): Withdrawn
|
216 |
+
Bidder 2 (gpt-3.5-turbo-16k-0613): Withdrawn
|
217 |
+
|
218 |
+
Hammer price:
|
219 |
+
Bidder 4 (gpt-3.5-turbo-16k-0613): $1200
|
220 |
+
'''
|
221 |
+
markdown_output = "## Auction Log\n\n"
|
222 |
+
for i, (item, bids) in enumerate(self.auction_logs.items()):
|
223 |
+
markdown_output += f"### {i+1}. {item}\n\n"
|
224 |
+
cur_bid_round = -1
|
225 |
+
for i, bid in enumerate(bids):
|
226 |
+
if bid['bid_round'] != cur_bid_round:
|
227 |
+
cur_bid_round = bid['bid_round']
|
228 |
+
if isinstance(bid['bid_round'], int):
|
229 |
+
markdown_output += f"\n#### {p.ordinal(bid['bid_round']+1)} bid:\n\n"
|
230 |
+
else:
|
231 |
+
markdown_output += f"\n#### {bid['bid_round']}:\n\n"
|
232 |
+
bid_price = f"${bid['bid']}" if bid['bid'] != -1 else 'Withdrew'
|
233 |
+
if isinstance(bid['bidder'], Bidder) or isinstance(bid['bidder'], HumanBidder):
|
234 |
+
if show_model_name:
|
235 |
+
markdown_output += f"* {bid['bidder']} ({bid['bidder'].model_name}): {bid_price}\n"
|
236 |
+
else:
|
237 |
+
markdown_output += f"* {bid['bidder']}: {bid_price}\n"
|
238 |
+
else:
|
239 |
+
markdown_output += f"* None bid\n"
|
240 |
+
markdown_output += "\n"
|
241 |
+
|
242 |
+
if len(bidder_personal_reports) != 0:
|
243 |
+
markdown_output += f"\n## Personal Report"
|
244 |
+
for report in bidder_personal_reports:
|
245 |
+
markdown_output += f"\n\n{report}"
|
246 |
+
return markdown_output.strip()
|
247 |
+
|
248 |
+
def finish_auction(self):
|
249 |
+
self.auction_logs = defaultdict(list)
|
250 |
+
self.cur_item = None
|
251 |
+
self.highest_bidder = None
|
252 |
+
self.highest_bid = -1
|
253 |
+
self.bidding_history = defaultdict(list)
|
254 |
+
self.items_queue = []
|
255 |
+
self.items = []
|
256 |
+
self.prev_round_max_bid = -1
|
257 |
+
self.fail_to_sell = False
|
258 |
+
self.min_bid = 0
|
259 |
+
|
src/bidder_base.py
ADDED
@@ -0,0 +1,1031 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from typing import List
|
2 |
+
from langchain.base_language import BaseLanguageModel
|
3 |
+
from langchain.schema import (
|
4 |
+
AIMessage,
|
5 |
+
HumanMessage,
|
6 |
+
SystemMessage
|
7 |
+
)
|
8 |
+
from langchain.chat_models import (
|
9 |
+
ChatAnthropic,
|
10 |
+
ChatOpenAI,
|
11 |
+
ChatVertexAI,
|
12 |
+
ChatGooglePalm,
|
13 |
+
)
|
14 |
+
import vertexai
|
15 |
+
from langchain.input import get_colored_text
|
16 |
+
from langchain.callbacks import get_openai_callback
|
17 |
+
from collections import defaultdict
|
18 |
+
from pydantic import BaseModel
|
19 |
+
import queue
|
20 |
+
import threading
|
21 |
+
import os
|
22 |
+
import random
|
23 |
+
import time
|
24 |
+
import ujson as json
|
25 |
+
import matplotlib.pyplot as plt
|
26 |
+
from .item_base import Item, item_list_equal
|
27 |
+
from .prompt_base import (
|
28 |
+
AUCTION_HISTORY,
|
29 |
+
# INSTRUCT_OBSERVE_TEMPLATE,
|
30 |
+
_LEARNING_STATEMENT,
|
31 |
+
INSTRUCT_PLAN_TEMPLATE,
|
32 |
+
INSTRUCT_BID_TEMPLATE,
|
33 |
+
INSTRUCT_SUMMARIZE_TEMPLATE,
|
34 |
+
INSTRUCT_LEARNING_TEMPLATE,
|
35 |
+
INSTRUCT_REPLAN_TEMPLATE,
|
36 |
+
SYSTEM_MESSAGE,
|
37 |
+
)
|
38 |
+
import sys
|
39 |
+
sys.path.append('..')
|
40 |
+
from utils import LoadJsonL, extract_jsons_from_text, extract_numbered_list, trace_back
|
41 |
+
|
42 |
+
|
43 |
+
# DESIRE_DESC = {
|
44 |
+
# 'default': "Your goal is to fully utilize your budget while actively participating in the auction",
|
45 |
+
# 'maximize_profit': "Your goal is to maximize your overall profit, and fully utilize your budget while actively participating in the auction. This involves strategic bidding to win items for less than their true value, thereby ensuring the difference between the price paid and the item's value is as large as possible",
|
46 |
+
# 'maximize_items': "Your goal is to win as many items as possible, and fully utilize your budget while actively participating in the auction. While keeping your budget in mind, you should aim to participate broadly across different items, striving to be the highest bidder more often than not",
|
47 |
+
# } # remove period at the end of each description
|
48 |
+
|
49 |
+
|
50 |
+
DESIRE_DESC = {
|
51 |
+
'maximize_profit': "Your primary objective is to secure the highest profit at the end of this auction, compared to all other bidders",
|
52 |
+
'maximize_items': "Your primary objective is to win the highest number of items at the end of this auction, compared to everyone else",
|
53 |
+
}
|
54 |
+
|
55 |
+
|
56 |
+
class Bidder(BaseModel):
|
57 |
+
name: str
|
58 |
+
model_name: str
|
59 |
+
budget: int
|
60 |
+
desire: str
|
61 |
+
plan_strategy: str
|
62 |
+
temperature: float = 0.7
|
63 |
+
overestimate_percent: int = 10
|
64 |
+
correct_belief: bool
|
65 |
+
enable_learning: bool = False
|
66 |
+
|
67 |
+
llm: BaseLanguageModel = None
|
68 |
+
openai_cost = 0
|
69 |
+
llm_token_count = 0
|
70 |
+
|
71 |
+
verbose: bool = False
|
72 |
+
auction_hash: str = ''
|
73 |
+
|
74 |
+
system_message: str = ''
|
75 |
+
original_budget: int = 0
|
76 |
+
|
77 |
+
# working memory
|
78 |
+
profit: int = 0
|
79 |
+
cur_item_id = 0
|
80 |
+
items: list = []
|
81 |
+
dialogue_history: list = [] # for gradio UI display
|
82 |
+
llm_prompt_history: list = [] # for tracking llm calling
|
83 |
+
items_won = []
|
84 |
+
bid_history: list = [] # history of the bidding of a single item
|
85 |
+
plan_instruct: str = '' # instruction for planning
|
86 |
+
cur_plan: str = '' # current plan
|
87 |
+
status_quo: dict = {} # belief of budget and profit, self and others
|
88 |
+
withdraw: bool = False # state of withdraw
|
89 |
+
learnings: str = '' # learnings from previous biddings. If given, then use it to guide the rest of the auction.
|
90 |
+
max_bid_cnt: int = 4 # Rule Bidder: maximum number of bids on one item (K = 1 starting bid + K-1 increase bid)
|
91 |
+
rule_bid_cnt: int = 0 # Rule Bidder: count of bids on one item
|
92 |
+
|
93 |
+
# belief tracking
|
94 |
+
failed_bid_cnt: int = 0 # count of failed bids (overspending)
|
95 |
+
total_bid_cnt: int = 0 # count of total bids
|
96 |
+
self_belief_error_cnt: int = 0
|
97 |
+
total_self_belief_cnt: int = 0
|
98 |
+
other_belief_error_cnt: int = 0
|
99 |
+
total_other_belief_cnt: int = 0
|
100 |
+
|
101 |
+
engagement_count: int = 0
|
102 |
+
budget_history = []
|
103 |
+
profit_history = []
|
104 |
+
budget_error_history = []
|
105 |
+
profit_error_history = []
|
106 |
+
win_bid_error_history = []
|
107 |
+
engagement_history = defaultdict(int)
|
108 |
+
all_bidders_status = {} # track others' profit
|
109 |
+
changes_of_plan = []
|
110 |
+
|
111 |
+
# not used
|
112 |
+
input_box: str = None
|
113 |
+
need_input = False
|
114 |
+
semaphore = 0
|
115 |
+
|
116 |
+
class Config:
|
117 |
+
arbitrary_types_allowed = True
|
118 |
+
|
119 |
+
def __repr__(self):
|
120 |
+
return self.name
|
121 |
+
|
122 |
+
def __str__(self):
|
123 |
+
return self.name
|
124 |
+
|
125 |
+
@classmethod
|
126 |
+
def create(cls, **data):
|
127 |
+
instance = cls(**data)
|
128 |
+
instance._post_init()
|
129 |
+
return instance
|
130 |
+
|
131 |
+
def _post_init(self):
|
132 |
+
self.original_budget = self.budget
|
133 |
+
self.system_message = SYSTEM_MESSAGE.format(
|
134 |
+
name=self.name,
|
135 |
+
desire_desc=DESIRE_DESC[self.desire],
|
136 |
+
)
|
137 |
+
self._parse_llm()
|
138 |
+
self.dialogue_history += [
|
139 |
+
SystemMessage(content=self.system_message),
|
140 |
+
AIMessage(content='')
|
141 |
+
]
|
142 |
+
self.budget_history.append(self.budget)
|
143 |
+
self.profit_history.append(self.profit)
|
144 |
+
|
145 |
+
def _parse_llm(self):
|
146 |
+
if 'gpt-' in self.model_name:
|
147 |
+
self.llm = ChatOpenAI(model=self.model_name, temperature=self.temperature, max_retries=30, request_timeout=1200)
|
148 |
+
elif 'claude' in self.model_name:
|
149 |
+
self.llm = ChatAnthropic(model=self.model_name, temperature=self.temperature, default_request_timeout=1200)
|
150 |
+
elif 'bison' in self.model_name:
|
151 |
+
self.llm = ChatGooglePalm(model_name=f'models/{self.model_name}', temperature=self.temperature)
|
152 |
+
elif 'rule' in self.model_name or 'human' in self.model_name:
|
153 |
+
self.llm = None
|
154 |
+
else:
|
155 |
+
raise NotImplementedError(self.model_name)
|
156 |
+
|
157 |
+
# def _rotate_openai_org(self):
|
158 |
+
# # use two organizations to avoid rate limit
|
159 |
+
# if os.environ.get('OPENAI_ORGANIZATION_1') and os.environ.get('OPENAI_ORGANIZATION_2'):
|
160 |
+
# return random.choice([os.environ.get('OPENAI_ORGANIZATION_1'), os.environ.get('OPENAI_ORGANIZATION_2')])
|
161 |
+
# else:
|
162 |
+
# return None
|
163 |
+
|
164 |
+
def _run_llm_standalone(self, messages: list):
|
165 |
+
|
166 |
+
with get_openai_callback() as cb:
|
167 |
+
for i in range(6):
|
168 |
+
try:
|
169 |
+
input_token_num = self.llm.get_num_tokens_from_messages(messages)
|
170 |
+
if 'claude' in self.model_name: # anthropic's claude
|
171 |
+
result = self.llm(messages, max_tokens_to_sample=2048)
|
172 |
+
elif 'bison' in self.model_name: # google's palm-2
|
173 |
+
max_tokens = min(max(3900 - input_token_num, 192), 2048)
|
174 |
+
if isinstance(self.llm, ChatVertexAI):
|
175 |
+
result = self.llm(messages, max_output_tokens=max_tokens)
|
176 |
+
else:
|
177 |
+
result = self.llm(messages)
|
178 |
+
elif 'gpt' in self.model_name: # openai
|
179 |
+
if 'gpt-3.5-turbo' in self.model_name and '16k' not in self.model_name:
|
180 |
+
max_tokens = max(3900 - input_token_num, 192)
|
181 |
+
else:
|
182 |
+
# gpt-4
|
183 |
+
# self.llm.openai_organization = self._rotate_openai_org()
|
184 |
+
max_tokens = max(8000 - input_token_num, 192)
|
185 |
+
result = self.llm(messages, max_tokens=max_tokens)
|
186 |
+
elif 'llama' in self.model_name.lower():
|
187 |
+
raise NotImplementedError
|
188 |
+
else:
|
189 |
+
raise NotImplementedError
|
190 |
+
break
|
191 |
+
except:
|
192 |
+
print(f'Retrying for {self.model_name} ({i+1}/6), wait for {2**(i+1)} sec...')
|
193 |
+
time.sleep(2**(i+1))
|
194 |
+
self.openai_cost += cb.total_cost
|
195 |
+
self.llm_token_count = self.llm.get_num_tokens_from_messages(messages)
|
196 |
+
return result.content
|
197 |
+
|
198 |
+
def _get_estimated_value(self, item):
|
199 |
+
value = item.true_value * (1 + self.overestimate_percent / 100)
|
200 |
+
return int(value)
|
201 |
+
|
202 |
+
def _get_cur_item(self, key=None):
|
203 |
+
if self.cur_item_id < len(self.items):
|
204 |
+
if key is not None:
|
205 |
+
return self.items[self.cur_item_id].__dict__[key]
|
206 |
+
else:
|
207 |
+
return self.items[self.cur_item_id]
|
208 |
+
else:
|
209 |
+
return 'no item left'
|
210 |
+
|
211 |
+
def _get_next_item(self, key=None):
|
212 |
+
if self.cur_item_id + 1 < len(self.items):
|
213 |
+
if key is not None:
|
214 |
+
return self.items[self.cur_item_id + 1].__dict__[key]
|
215 |
+
else:
|
216 |
+
return self.items[self.cur_item_id + 1]
|
217 |
+
else:
|
218 |
+
return 'no item left'
|
219 |
+
|
220 |
+
def _get_remaining_items(self, as_str=False):
|
221 |
+
remain_items = self.items[self.cur_item_id + 1:]
|
222 |
+
if as_str:
|
223 |
+
return ', '.join([item.name for item in remain_items])
|
224 |
+
else:
|
225 |
+
return remain_items
|
226 |
+
|
227 |
+
def _get_items_value_str(self, items: List[Item]):
|
228 |
+
if not isinstance(items, list):
|
229 |
+
items = [items]
|
230 |
+
items_info = ''
|
231 |
+
for i, item in enumerate(items):
|
232 |
+
estimated_value = self._get_estimated_value(item)
|
233 |
+
_info = f"{i+1}. {item}, starting price is ${item.price}. Your estimated value for this item is ${estimated_value}.\n"
|
234 |
+
items_info += _info
|
235 |
+
return items_info.strip()
|
236 |
+
|
237 |
+
# ********** Main Instructions and Functions ********** #
|
238 |
+
|
239 |
+
def learn_from_prev_auction(self, past_learnings, past_auction_log):
|
240 |
+
if not self.enable_learning or 'rule' in self.model_name or 'human' in self.model_name:
|
241 |
+
return ''
|
242 |
+
|
243 |
+
instruct_learn = INSTRUCT_LEARNING_TEMPLATE.format(
|
244 |
+
past_auction_log=past_auction_log,
|
245 |
+
past_learnings=past_learnings)
|
246 |
+
|
247 |
+
result = self._run_llm_standalone([HumanMessage(content=instruct_learn)])
|
248 |
+
self.dialogue_history += [
|
249 |
+
HumanMessage(content=instruct_learn),
|
250 |
+
AIMessage(content=result),
|
251 |
+
]
|
252 |
+
self.llm_prompt_history.append({
|
253 |
+
'messages': [{x.type: x.content} for x in [HumanMessage(content=instruct_learn)]],
|
254 |
+
'result': result,
|
255 |
+
'tag': 'learn_0'
|
256 |
+
})
|
257 |
+
|
258 |
+
self.learnings = '\n'.join(extract_numbered_list(result))
|
259 |
+
if self.learnings != '':
|
260 |
+
self.system_message += f"\n\nHere are your key learning points and practical tips from a previous auction. You can use them to guide this auction:\n```\n{self.learnings}\n```"
|
261 |
+
|
262 |
+
if self.verbose:
|
263 |
+
print(f"Learn from previous auction: {self.name} ({self.model_name}).")
|
264 |
+
return result
|
265 |
+
|
266 |
+
def _choose_items(self, budget, items: List[Item]):
|
267 |
+
'''
|
268 |
+
Choose items within budget for rule bidders.
|
269 |
+
Cheap ones first if maximize_items, expensive ones first if maximize_profit.
|
270 |
+
'''
|
271 |
+
sorted_items = sorted(items, key=lambda x: self._get_estimated_value(x),
|
272 |
+
reverse=self.desire == 'maximize_profit')
|
273 |
+
|
274 |
+
chosen_items = []
|
275 |
+
i = 0
|
276 |
+
while budget >= 0 and i < len(sorted_items):
|
277 |
+
item = sorted_items[i]
|
278 |
+
if item.price <= budget:
|
279 |
+
chosen_items.append(item)
|
280 |
+
budget -= item.price
|
281 |
+
i += 1
|
282 |
+
|
283 |
+
return chosen_items
|
284 |
+
|
285 |
+
def get_plan_instruct(self, items: List[Item]):
|
286 |
+
self.items = items
|
287 |
+
plan_instruct = INSTRUCT_PLAN_TEMPLATE.format(
|
288 |
+
bidder_name=self.name,
|
289 |
+
budget=self.budget,
|
290 |
+
item_num=len(items),
|
291 |
+
items_info=self._get_items_value_str(items),
|
292 |
+
desire_desc=DESIRE_DESC[self.desire],
|
293 |
+
learning_statement='' if not self.enable_learning else _LEARNING_STATEMENT
|
294 |
+
)
|
295 |
+
return plan_instruct
|
296 |
+
|
297 |
+
def init_plan(self, plan_instruct: str):
|
298 |
+
'''
|
299 |
+
Plan for bidding with auctioneer's instruction and items information for customize estimated value.
|
300 |
+
plan = plan(system_message, instruct_plan)
|
301 |
+
'''
|
302 |
+
if 'rule' in self.model_name:
|
303 |
+
# self.cur_plan = ', '.join([x.name for x in self._choose_items(self.budget, self.items)])
|
304 |
+
# self.dialogue_history += [
|
305 |
+
# HumanMessage(content=plan_instruct),
|
306 |
+
# AIMessage(content=self.cur_plan),
|
307 |
+
# ]
|
308 |
+
# return self.cur_plan
|
309 |
+
return ''
|
310 |
+
|
311 |
+
self.status_quo = {
|
312 |
+
'remaining_budget': self.budget,
|
313 |
+
'total_profits': {bidder: 0 for bidder in self.all_bidders_status.keys()},
|
314 |
+
'winning_bids': {bidder: {} for bidder in self.all_bidders_status.keys()},
|
315 |
+
}
|
316 |
+
|
317 |
+
if self.plan_strategy == 'none':
|
318 |
+
self.plan_instruct = ''
|
319 |
+
self.cur_plan = ''
|
320 |
+
return None
|
321 |
+
|
322 |
+
system_msg = SystemMessage(content=self.system_message)
|
323 |
+
plan_msg = HumanMessage(content=plan_instruct)
|
324 |
+
messages = [system_msg, plan_msg]
|
325 |
+
result = self._run_llm_standalone(messages)
|
326 |
+
|
327 |
+
if self.verbose:
|
328 |
+
print(get_colored_text(plan_msg.content, 'red'))
|
329 |
+
print(get_colored_text(result, 'green'))
|
330 |
+
|
331 |
+
self.dialogue_history += [
|
332 |
+
plan_msg,
|
333 |
+
AIMessage(content=result),
|
334 |
+
]
|
335 |
+
self.llm_prompt_history.append({
|
336 |
+
'messages': [{x.type: x.content} for x in messages],
|
337 |
+
'result': result,
|
338 |
+
'tag': 'plan_0'
|
339 |
+
})
|
340 |
+
self.cur_plan = result
|
341 |
+
self.plan_instruct = plan_instruct
|
342 |
+
|
343 |
+
self.changes_of_plan.append([
|
344 |
+
f"{self.cur_item_id} (Initial)",
|
345 |
+
False,
|
346 |
+
json.dumps(extract_jsons_from_text(result)[-1]),
|
347 |
+
])
|
348 |
+
|
349 |
+
if self.verbose:
|
350 |
+
print(f"Plan: {self.name} ({self.model_name}) for {self._get_cur_item()}.")
|
351 |
+
return result
|
352 |
+
|
353 |
+
def get_rebid_instruct(self, auctioneer_msg: str):
|
354 |
+
self.dialogue_history += [
|
355 |
+
HumanMessage(content=auctioneer_msg),
|
356 |
+
AIMessage(content='')
|
357 |
+
]
|
358 |
+
return auctioneer_msg
|
359 |
+
|
360 |
+
def get_bid_instruct(self, auctioneer_msg: str, bid_round: int):
|
361 |
+
auctioneer_msg = auctioneer_msg.replace(self.name, f'You ({self.name})')
|
362 |
+
|
363 |
+
bid_instruct = INSTRUCT_BID_TEMPLATE.format(
|
364 |
+
auctioneer_msg=auctioneer_msg,
|
365 |
+
bidder_name=self.name,
|
366 |
+
cur_item=self._get_cur_item(),
|
367 |
+
estimated_value=self._get_estimated_value(self._get_cur_item()),
|
368 |
+
desire_desc=DESIRE_DESC[self.desire],
|
369 |
+
learning_statement='' if not self.enable_learning else _LEARNING_STATEMENT
|
370 |
+
)
|
371 |
+
if bid_round == 0:
|
372 |
+
if self.plan_strategy in ['static', 'none']:
|
373 |
+
# if static planner, then no replanning is needed. status quo is updated in replanning. thus need to add status quo in bid instruct.
|
374 |
+
bid_instruct = f"""The status quo of this auction so far is:\n"{json.dumps(self.status_quo, indent=4)}"\n\n{bid_instruct}\n---\n"""
|
375 |
+
else:
|
376 |
+
bid_instruct = f'Now, the auctioneer says: "{auctioneer_msg}"'
|
377 |
+
|
378 |
+
self.dialogue_history += [
|
379 |
+
HumanMessage(content=bid_instruct),
|
380 |
+
AIMessage(content='')
|
381 |
+
]
|
382 |
+
return bid_instruct
|
383 |
+
|
384 |
+
def bid_rule(self, cur_bid: int, min_markup_pct: float = 0.1):
|
385 |
+
'''
|
386 |
+
:param cur_bid: current highest bid
|
387 |
+
:param min_markup_pct: minimum percentage for bid increase
|
388 |
+
:param max_bid_cnt: maximum number of bids on one item (K = 1 starting bid + K-1 increase bid)
|
389 |
+
'''
|
390 |
+
# dialogue history already got bid_instruction.
|
391 |
+
cur_item = self._get_cur_item()
|
392 |
+
|
393 |
+
if cur_bid <= 0:
|
394 |
+
next_bid = cur_item.price
|
395 |
+
else:
|
396 |
+
next_bid = cur_bid + min_markup_pct * cur_item.price
|
397 |
+
|
398 |
+
if self.budget - next_bid >= 0 and self.rule_bid_cnt < self.max_bid_cnt:
|
399 |
+
msg = int(next_bid)
|
400 |
+
self.rule_bid_cnt += 1
|
401 |
+
else:
|
402 |
+
msg = -1
|
403 |
+
|
404 |
+
content = f'The current highest bid for {cur_item.name} is ${cur_bid}. '
|
405 |
+
content += "I'm out!" if msg < 0 else f"I bid ${msg}! (Rule generated)"
|
406 |
+
self.dialogue_history += [
|
407 |
+
HumanMessage(content=''),
|
408 |
+
AIMessage(content=content)
|
409 |
+
]
|
410 |
+
|
411 |
+
return msg
|
412 |
+
|
413 |
+
def bid(self, bid_instruct):
|
414 |
+
'''
|
415 |
+
Bid for an item with auctioneer's instruction and bidding history.
|
416 |
+
bid_history = bid(system_message, instruct_plan, plan, bid_history)
|
417 |
+
'''
|
418 |
+
if self.model_name == 'rule':
|
419 |
+
return ''
|
420 |
+
|
421 |
+
bid_msg = HumanMessage(content=bid_instruct)
|
422 |
+
|
423 |
+
if self.plan_strategy == 'none':
|
424 |
+
messages = [SystemMessage(content=self.system_message)]
|
425 |
+
else:
|
426 |
+
messages = [SystemMessage(content=self.system_message),
|
427 |
+
HumanMessage(content=self.plan_instruct),
|
428 |
+
AIMessage(content=self.cur_plan)]
|
429 |
+
|
430 |
+
self.bid_history += [bid_msg]
|
431 |
+
messages += self.bid_history
|
432 |
+
|
433 |
+
result = self._run_llm_standalone(messages)
|
434 |
+
|
435 |
+
self.bid_history += [AIMessage(content=result)]
|
436 |
+
|
437 |
+
self.dialogue_history += [
|
438 |
+
HumanMessage(content=''),
|
439 |
+
AIMessage(content=result)
|
440 |
+
]
|
441 |
+
|
442 |
+
self.llm_prompt_history.append({
|
443 |
+
'messages': [{x.type: x.content} for x in messages],
|
444 |
+
'result': result,
|
445 |
+
'tag': f'bid_{self.cur_item_id}'
|
446 |
+
})
|
447 |
+
|
448 |
+
if self.verbose:
|
449 |
+
print(get_colored_text(bid_instruct, 'yellow'))
|
450 |
+
print(get_colored_text(result, 'green'))
|
451 |
+
|
452 |
+
print(f"Bid: {self.name} ({self.model_name}) for {self._get_cur_item()}.")
|
453 |
+
self.total_bid_cnt += 1
|
454 |
+
|
455 |
+
return result
|
456 |
+
|
457 |
+
def get_summarize_instruct(self, bidding_history: str, hammer_msg: str, win_lose_msg: str):
|
458 |
+
instruct = INSTRUCT_SUMMARIZE_TEMPLATE.format(
|
459 |
+
cur_item=self._get_cur_item(),
|
460 |
+
bidding_history=bidding_history,
|
461 |
+
hammer_msg=hammer_msg.strip(),
|
462 |
+
win_lose_msg=win_lose_msg.strip(),
|
463 |
+
bidder_name=self.name,
|
464 |
+
prev_status=self._status_json_to_text(self.status_quo),
|
465 |
+
)
|
466 |
+
return instruct
|
467 |
+
|
468 |
+
def summarize(self, instruct_summarize: str):
|
469 |
+
'''
|
470 |
+
Update belief/status quo
|
471 |
+
status_quo = summarize(system_message, bid_history, prev_status + instruct_summarize)
|
472 |
+
'''
|
473 |
+
self.budget_history.append(self.budget)
|
474 |
+
self.profit_history.append(self.profit)
|
475 |
+
|
476 |
+
if self.model_name == 'rule':
|
477 |
+
self.rule_bid_cnt = 0 # reset bid count for rule bidder
|
478 |
+
return ''
|
479 |
+
|
480 |
+
messages = [SystemMessage(content=self.system_message)]
|
481 |
+
# messages += self.bid_history
|
482 |
+
summ_msg = HumanMessage(content=instruct_summarize)
|
483 |
+
messages.append(summ_msg)
|
484 |
+
|
485 |
+
status_quo_text = self._run_llm_standalone(messages)
|
486 |
+
|
487 |
+
self.dialogue_history += [summ_msg, AIMessage(content=status_quo_text)]
|
488 |
+
self.bid_history += [summ_msg, AIMessage(content=status_quo_text)]
|
489 |
+
|
490 |
+
self.llm_prompt_history.append({
|
491 |
+
'messages': [{x.type: x.content} for x in messages],
|
492 |
+
'result': status_quo_text,
|
493 |
+
'tag': f'summarize_{self.cur_item_id}'
|
494 |
+
})
|
495 |
+
|
496 |
+
cnt = 0
|
497 |
+
while cnt <= 3:
|
498 |
+
sanity_msg = self._sanity_check_status_json(extract_jsons_from_text(status_quo_text)[-1])
|
499 |
+
if sanity_msg == '':
|
500 |
+
# pass sanity check then track beliefs
|
501 |
+
consistency_msg = self._belief_tracking(status_quo_text)
|
502 |
+
else:
|
503 |
+
sanity_msg = f'- {sanity_msg}'
|
504 |
+
consistency_msg = ''
|
505 |
+
|
506 |
+
if sanity_msg != '' or (consistency_msg != '' and self.correct_belief):
|
507 |
+
err_msg = f"As {self.name}, here are some error(s) of your summary of the status JSON:\n{sanity_msg.strip()}\n{consistency_msg.strip()}\n\nPlease revise the status JSON based on the errors. Don't apologize. Just give me the revised status JSON.".strip()
|
508 |
+
|
509 |
+
# print(f"{self.name}: revising status quo for the {cnt} time:")
|
510 |
+
# print(get_colored_text(err_msg, 'green'))
|
511 |
+
# print(get_colored_text(status_quo_text, 'red'))
|
512 |
+
|
513 |
+
messages += [AIMessage(content=status_quo_text),
|
514 |
+
HumanMessage(content=err_msg)]
|
515 |
+
status_quo_text = self._run_llm_standalone(messages)
|
516 |
+
self.dialogue_history += [
|
517 |
+
HumanMessage(content=err_msg),
|
518 |
+
AIMessage(content=status_quo_text),
|
519 |
+
]
|
520 |
+
cnt += 1
|
521 |
+
else:
|
522 |
+
break
|
523 |
+
|
524 |
+
self.status_quo = extract_jsons_from_text(status_quo_text)[-1]
|
525 |
+
|
526 |
+
if self.verbose:
|
527 |
+
print(get_colored_text(instruct_summarize, 'blue'))
|
528 |
+
print(get_colored_text(status_quo_text, 'green'))
|
529 |
+
|
530 |
+
print(f"Summarize: {self.name} ({self.model_name}) for {self._get_cur_item()}.")
|
531 |
+
|
532 |
+
return status_quo_text
|
533 |
+
|
534 |
+
def get_replan_instruct(self):
|
535 |
+
instruct = INSTRUCT_REPLAN_TEMPLATE.format(
|
536 |
+
status_quo=self._status_json_to_text(self.status_quo),
|
537 |
+
remaining_items_info=self._get_items_value_str(self._get_remaining_items()),
|
538 |
+
bidder_name=self.name,
|
539 |
+
desire_desc=DESIRE_DESC[self.desire],
|
540 |
+
learning_statement='' if not self.enable_learning else _LEARNING_STATEMENT
|
541 |
+
)
|
542 |
+
return instruct
|
543 |
+
|
544 |
+
def replan(self, instruct_replan: str):
|
545 |
+
'''
|
546 |
+
plan = replan(system_message, instruct_plan, prev_plan, status_quo + (learning) + instruct_replan)
|
547 |
+
'''
|
548 |
+
if self.model_name == 'rule':
|
549 |
+
self.withdraw = False
|
550 |
+
self.cur_item_id += 1
|
551 |
+
return ''
|
552 |
+
|
553 |
+
if self.plan_strategy in ['none', 'static']:
|
554 |
+
self.bid_history = [] # clear bid history
|
555 |
+
self.cur_item_id += 1
|
556 |
+
self.withdraw = False
|
557 |
+
return 'Skip replanning for bidders with static or no plan.'
|
558 |
+
|
559 |
+
replan_msg = HumanMessage(content=instruct_replan)
|
560 |
+
|
561 |
+
messages = [SystemMessage(content=self.system_message),
|
562 |
+
HumanMessage(content=self.plan_instruct),
|
563 |
+
AIMessage(content=self.cur_plan)]
|
564 |
+
messages.append(replan_msg)
|
565 |
+
|
566 |
+
result = self._run_llm_standalone(messages)
|
567 |
+
|
568 |
+
new_plan_dict = extract_jsons_from_text(result)[-1]
|
569 |
+
cnt = 0
|
570 |
+
while len(new_plan_dict) == 0 and cnt < 2:
|
571 |
+
err_msg = 'Your response does not contain a JSON-format priority list for items. Please revise your plan.'
|
572 |
+
messages += [
|
573 |
+
AIMessage(content=result),
|
574 |
+
HumanMessage(content=err_msg),
|
575 |
+
]
|
576 |
+
result = self._run_llm_standalone(messages)
|
577 |
+
new_plan_dict = extract_jsons_from_text(result)[-1]
|
578 |
+
|
579 |
+
self.dialogue_history += [
|
580 |
+
HumanMessage(content=err_msg),
|
581 |
+
AIMessage(content=result),
|
582 |
+
]
|
583 |
+
cnt += 1
|
584 |
+
|
585 |
+
old_plan_dict = extract_jsons_from_text(self.cur_plan)[-1]
|
586 |
+
self.changes_of_plan.append([
|
587 |
+
f"{self.cur_item_id + 1} ({self._get_cur_item('name')})",
|
588 |
+
self._change_of_plan(old_plan_dict, new_plan_dict),
|
589 |
+
json.dumps(new_plan_dict)
|
590 |
+
])
|
591 |
+
|
592 |
+
self.plan_instruct = instruct_replan
|
593 |
+
self.cur_plan = result
|
594 |
+
self.withdraw = False
|
595 |
+
self.bid_history = [] # clear bid history
|
596 |
+
self.cur_item_id += 1
|
597 |
+
|
598 |
+
self.dialogue_history += [
|
599 |
+
replan_msg,
|
600 |
+
AIMessage(content=result),
|
601 |
+
]
|
602 |
+
self.llm_prompt_history.append({
|
603 |
+
'messages': [{x.type: x.content} for x in messages],
|
604 |
+
'result': result,
|
605 |
+
'tag': f'plan_{self.cur_item_id}'
|
606 |
+
})
|
607 |
+
|
608 |
+
if self.verbose:
|
609 |
+
print(get_colored_text(instruct_replan, 'blue'))
|
610 |
+
print(get_colored_text(result, 'green'))
|
611 |
+
|
612 |
+
print(f"Replan: {self.name} ({self.model_name}).")
|
613 |
+
return result
|
614 |
+
|
615 |
+
def _change_of_plan(self, old_plan: dict, new_plan: dict):
|
616 |
+
for k in new_plan:
|
617 |
+
if new_plan[k] != old_plan.get(k, None):
|
618 |
+
return True
|
619 |
+
return False
|
620 |
+
|
621 |
+
# *********** Belief Tracking and Sanity Check *********** #
|
622 |
+
|
623 |
+
def bid_sanity_check(self, bid_price, prev_round_max_bid, min_markup_pct):
|
624 |
+
# can't bid more than budget or less than previous highest bid
|
625 |
+
if bid_price < 0:
|
626 |
+
msg = None
|
627 |
+
else:
|
628 |
+
min_bid_increase = int(min_markup_pct * self._get_cur_item('price'))
|
629 |
+
if bid_price > self.budget:
|
630 |
+
msg = f"you don't have insufficient budget (${self.budget} left)"
|
631 |
+
elif bid_price < self._get_cur_item('price'):
|
632 |
+
msg = f"your bid is lower than the starting bid (${self._get_cur_item('price')})"
|
633 |
+
elif bid_price < prev_round_max_bid + min_bid_increase:
|
634 |
+
msg = f"you must advance previous highest bid (${prev_round_max_bid}) by at least ${min_bid_increase} ({int(100 * min_markup_pct)}%)."
|
635 |
+
else:
|
636 |
+
msg = None
|
637 |
+
return msg
|
638 |
+
|
639 |
+
def rebid_for_failure(self, fail_instruct: str):
|
640 |
+
result = self.bid(fail_instruct)
|
641 |
+
self.failed_bid_cnt += 1
|
642 |
+
return result
|
643 |
+
|
644 |
+
def _sanity_check_status_json(self, data: dict):
|
645 |
+
if data == {}:
|
646 |
+
return "Error: No parsible JSON in your response. Possibly due to missing a closing curly bracket '}', or unpasible values (e.g., 'profit': 1000 + 400, instead of 'profit': 1400)."
|
647 |
+
|
648 |
+
# Check if all expected top-level keys are present
|
649 |
+
expected_keys = ["remaining_budget", "total_profits", "winning_bids"]
|
650 |
+
for key in expected_keys:
|
651 |
+
if key not in data:
|
652 |
+
return f"Error: Missing '{key}' field in the status JSON."
|
653 |
+
|
654 |
+
# Check if "remaining_budget" is a number
|
655 |
+
if not isinstance(data["remaining_budget"], (int, float)):
|
656 |
+
return "Error: 'remaining_budget' should be a number, and only about your remaining budget."
|
657 |
+
|
658 |
+
# Check if "total_profits" is a dictionary with numbers as values
|
659 |
+
if not isinstance(data["total_profits"], dict):
|
660 |
+
return "Error: 'total_profits' should be a dictionary of every bidder."
|
661 |
+
for bidder, profit in data["total_profits"].items():
|
662 |
+
if not isinstance(profit, (int, float)):
|
663 |
+
return f"Error: Profit for {bidder} should be a number."
|
664 |
+
|
665 |
+
# Check if "winning_bids" is a dictionary and that each bidder's entry is a dictionary with numbers
|
666 |
+
if not isinstance(data["winning_bids"], dict):
|
667 |
+
return "Error: 'winning_bids' should be a dictionary."
|
668 |
+
for bidder, bids in data["winning_bids"].items():
|
669 |
+
if not isinstance(bids, dict):
|
670 |
+
return f"Error: Bids for {bidder} should be a dictionary."
|
671 |
+
for item, amount in bids.items():
|
672 |
+
if not isinstance(amount, (int, float)):
|
673 |
+
return f"Error: Amount for {item} under {bidder} should be a number."
|
674 |
+
|
675 |
+
# If everything is fine
|
676 |
+
return ""
|
677 |
+
|
678 |
+
def _status_json_to_text(self, data: dict):
|
679 |
+
if 'rule' in self.model_name: return ''
|
680 |
+
|
681 |
+
# Extract and format remaining budget
|
682 |
+
structured_text = f"* Remaining Budget: ${data.get('remaining_budget', 'unknown')}\n\n"
|
683 |
+
|
684 |
+
# Extract and format total profits for each bidder
|
685 |
+
structured_text += "* Total Profits:\n"
|
686 |
+
if data.get('total_profits'):
|
687 |
+
for bidder, profit in data['total_profits'].items():
|
688 |
+
structured_text += f" * {bidder}: ${profit}\n"
|
689 |
+
|
690 |
+
# Extract and list the winning bids for each item by each bidder
|
691 |
+
structured_text += "\n* Winning Bids:\n"
|
692 |
+
if data.get('winning_bids'):
|
693 |
+
for bidder, bids in data['winning_bids'].items():
|
694 |
+
structured_text += f" * {bidder}:\n"
|
695 |
+
if bids:
|
696 |
+
for item, amount in bids.items():
|
697 |
+
structured_text += f" * {item}: ${amount}\n"
|
698 |
+
else:
|
699 |
+
structured_text += f" * No winning bids\n"
|
700 |
+
|
701 |
+
return structured_text.strip()
|
702 |
+
|
703 |
+
def _belief_tracking(self, status_text: str):
|
704 |
+
'''
|
705 |
+
Parse status quo and check if the belief is correct.
|
706 |
+
'''
|
707 |
+
belief_json = extract_jsons_from_text(status_text)[-1]
|
708 |
+
# {"remaining_budget": 8000, "total_profits": {"Bidder 1": 1300, "Bidder 2": 1800, "Bidder 3": 0}, "winning_bids": {"Bidder 1": {"Item 2": 1200, "Item 3": 1000}, "Bidder 2": {"Item 1": 2000}, "Bidder 3": {}}}
|
709 |
+
budget_belief = belief_json['remaining_budget']
|
710 |
+
profits_belief = belief_json['total_profits']
|
711 |
+
winning_bids = belief_json['winning_bids']
|
712 |
+
|
713 |
+
msg = ''
|
714 |
+
# track belief of budget
|
715 |
+
self.total_self_belief_cnt += 1
|
716 |
+
if budget_belief != self.budget:
|
717 |
+
msg += f'- Your belief of budget is wrong: you have ${self.budget} left, but you think you have ${budget_belief} left.\n'
|
718 |
+
self.self_belief_error_cnt += 1
|
719 |
+
self.budget_error_history.append([
|
720 |
+
self._get_cur_item('name'),
|
721 |
+
budget_belief,
|
722 |
+
self.budget,
|
723 |
+
])
|
724 |
+
|
725 |
+
# track belief of profits
|
726 |
+
for bidder_name, profit in profits_belief.items():
|
727 |
+
if self.all_bidders_status.get(bidder_name) is None:
|
728 |
+
# due to a potentially unreasonable parsing
|
729 |
+
continue
|
730 |
+
|
731 |
+
if self.name in bidder_name:
|
732 |
+
bidder_name = self.name
|
733 |
+
self.total_self_belief_cnt += 1
|
734 |
+
else:
|
735 |
+
self.total_other_belief_cnt += 1
|
736 |
+
|
737 |
+
real_profit = self.all_bidders_status[bidder_name]['profit']
|
738 |
+
|
739 |
+
if profit != real_profit:
|
740 |
+
if self.name == bidder_name:
|
741 |
+
self.self_belief_error_cnt += 1
|
742 |
+
else:
|
743 |
+
self.other_belief_error_cnt += 1
|
744 |
+
|
745 |
+
msg += f'- Your belief of total profit of {bidder_name} is wrong: {bidder_name} has earned ${real_profit} so far, but you think {bidder_name} has earned ${profit}.\n'
|
746 |
+
|
747 |
+
# add to history
|
748 |
+
self.profit_error_history.append([
|
749 |
+
f"{bidder_name} ({self._get_cur_item('name')})",
|
750 |
+
profit,
|
751 |
+
real_profit
|
752 |
+
])
|
753 |
+
|
754 |
+
# track belief of winning bids
|
755 |
+
for bidder_name, items_won_dict in winning_bids.items():
|
756 |
+
if self.all_bidders_status.get(bidder_name) is None:
|
757 |
+
# due to a potentially unreasonable parsing
|
758 |
+
continue
|
759 |
+
|
760 |
+
real_items_won = self.all_bidders_status[bidder_name]['items_won']
|
761 |
+
# items_won = [(item, bid_price), ...)]
|
762 |
+
|
763 |
+
items_won_list = list(items_won_dict.keys())
|
764 |
+
real_items_won_list = [str(x) for x, _ in real_items_won]
|
765 |
+
|
766 |
+
if self.name in bidder_name:
|
767 |
+
self.total_self_belief_cnt += 1
|
768 |
+
else:
|
769 |
+
self.total_other_belief_cnt += 1
|
770 |
+
|
771 |
+
if not item_list_equal(items_won_list, real_items_won_list):
|
772 |
+
if bidder_name == self.name:
|
773 |
+
self.self_belief_error_cnt += 1
|
774 |
+
_bidder_name = f'you'
|
775 |
+
else:
|
776 |
+
self.other_belief_error_cnt += 1
|
777 |
+
_bidder_name = bidder_name
|
778 |
+
|
779 |
+
msg += f"- Your belief of winning items of {bidder_name} is wrong: {bidder_name} won {real_items_won}, but you think {bidder_name} won {items_won_dict}.\n"
|
780 |
+
|
781 |
+
self.win_bid_error_history.append([
|
782 |
+
f"{_bidder_name} ({self._get_cur_item('name')})",
|
783 |
+
', '.join(items_won_list),
|
784 |
+
', '.join(real_items_won_list)
|
785 |
+
])
|
786 |
+
|
787 |
+
return msg
|
788 |
+
|
789 |
+
def win_bid(self, item: Item, bid: int):
|
790 |
+
self.budget -= bid
|
791 |
+
self.profit += item.true_value - bid
|
792 |
+
self.items_won += [[item, bid]]
|
793 |
+
msg = f"Congratuations! You won {item} at ${bid}."# Now you have ${self.budget} left. Your total profit so far is ${self.profit}."
|
794 |
+
return msg
|
795 |
+
|
796 |
+
def lose_bid(self, item: Item):
|
797 |
+
return f"You lost {item}."# Now, you have ${self.budget} left. Your total profit so far is ${self.profit}."
|
798 |
+
|
799 |
+
# set the profit information of other bidders
|
800 |
+
def set_all_bidders_status(self, all_bidders_status: dict):
|
801 |
+
self.all_bidders_status = all_bidders_status.copy()
|
802 |
+
|
803 |
+
def set_withdraw(self, bid: int):
|
804 |
+
if bid < 0: # withdraw
|
805 |
+
self.withdraw = True
|
806 |
+
elif bid == 0: # enable discount and bid again
|
807 |
+
self.withdraw = False
|
808 |
+
else: # normal bid
|
809 |
+
self.withdraw = False
|
810 |
+
self.engagement_count += 1
|
811 |
+
self.engagement_history[self._get_cur_item('name')] += 1
|
812 |
+
|
813 |
+
# ****************** Logging ****************** #
|
814 |
+
|
815 |
+
# def _parse_hedging(self, plan: str): # deprecated
|
816 |
+
# prompt = PARSE_HEDGE_INSTRUCTION.format(
|
817 |
+
# item_name=self._get_cur_item(),
|
818 |
+
# plan=plan)
|
819 |
+
|
820 |
+
# with get_openai_callback() as cb:
|
821 |
+
# llm = ChatOpenAI(model='gpt-3.5-turbo-0613', temperature=0)
|
822 |
+
# result = llm([HumanMessage(content=prompt)]).content
|
823 |
+
# self.openai_cost += cb.total_cost
|
824 |
+
# # parse a number, which could be a digit
|
825 |
+
# hedge_percent = re.findall(r'\d+\.?\d*%', result)
|
826 |
+
# if len(hedge_percent) > 0:
|
827 |
+
# hedge_percent = hedge_percent[0].replace('%', '')
|
828 |
+
# else:
|
829 |
+
# hedge_percent = 0
|
830 |
+
# return float(hedge_percent)
|
831 |
+
|
832 |
+
def profit_report(self):
|
833 |
+
'''
|
834 |
+
Personal profit report at the end of an auction.
|
835 |
+
'''
|
836 |
+
msg = f"* {self.name}, starting with ${self.original_budget}, has won {len(self.items_won)} items in this auction, with a total profit of ${self.profit}.:\n"
|
837 |
+
profit = 0
|
838 |
+
for item, bid in self.items_won:
|
839 |
+
profit += item.true_value - bid
|
840 |
+
msg += f" * Won {item} at ${bid} over ${item.price}, with a true value of ${item.true_value}.\n"
|
841 |
+
return msg.strip()
|
842 |
+
|
843 |
+
def to_monitors(self, as_json=False):
|
844 |
+
# budget, profit, items_won, tokens
|
845 |
+
if len(self.items_won) == 0 and not as_json:
|
846 |
+
items_won = [['', 0, 0]]
|
847 |
+
else:
|
848 |
+
items_won = []
|
849 |
+
for item, bid in self.items_won:
|
850 |
+
items_won.append([str(item), bid, item.true_value])
|
851 |
+
|
852 |
+
profit_error_history = self.profit_error_history if self.profit_error_history != [] or as_json else [['', '', '']]
|
853 |
+
win_bid_error_history = self.win_bid_error_history if self.win_bid_error_history != [] or as_json else [['', '', '']]
|
854 |
+
budget_error_history = self.budget_error_history if self.budget_error_history != [] or as_json else [['', '']]
|
855 |
+
changes_of_plan = self.changes_of_plan if self.changes_of_plan != [] or as_json else [['', '', '']]
|
856 |
+
|
857 |
+
if as_json:
|
858 |
+
return {
|
859 |
+
'auction_hash': self.auction_hash,
|
860 |
+
'bidder_name': self.name,
|
861 |
+
'model_name': self.model_name,
|
862 |
+
'desire': self.desire,
|
863 |
+
'plan_strategy': self.plan_strategy,
|
864 |
+
'overestimate_percent': self.overestimate_percent,
|
865 |
+
'temperature': self.temperature,
|
866 |
+
'correct_belief': self.correct_belief,
|
867 |
+
'enable_learning': self.enable_learning,
|
868 |
+
'budget': self.original_budget,
|
869 |
+
'money_left': self.budget,
|
870 |
+
'profit': self.profit,
|
871 |
+
'items_won': items_won,
|
872 |
+
'tokens_used': self.llm_token_count,
|
873 |
+
'openai_cost': round(self.openai_cost, 2),
|
874 |
+
'failed_bid_cnt': self.failed_bid_cnt,
|
875 |
+
'self_belief_error_cnt': self.self_belief_error_cnt,
|
876 |
+
'other_belief_error_cnt': self.other_belief_error_cnt,
|
877 |
+
'failed_bid_rate': round(self.failed_bid_cnt / (self.total_bid_cnt+1e-8), 2),
|
878 |
+
'self_error_rate': round(self.self_belief_error_cnt / (self.total_self_belief_cnt+1e-8), 2),
|
879 |
+
'other_error_rate': round(self.other_belief_error_cnt / (self.total_other_belief_cnt+1e-8), 2),
|
880 |
+
'engagement_count': self.engagement_count,
|
881 |
+
'engagement_history': self.engagement_history,
|
882 |
+
'changes_of_plan': changes_of_plan,
|
883 |
+
'budget_error_history': budget_error_history,
|
884 |
+
'profit_error_history': profit_error_history,
|
885 |
+
'win_bid_error_history': win_bid_error_history,
|
886 |
+
'history': self.llm_prompt_history
|
887 |
+
}
|
888 |
+
else:
|
889 |
+
return [
|
890 |
+
self.budget,
|
891 |
+
self.profit,
|
892 |
+
items_won,
|
893 |
+
self.llm_token_count,
|
894 |
+
round(self.openai_cost, 2),
|
895 |
+
round(self.failed_bid_cnt / (self.total_bid_cnt+1e-8), 2),
|
896 |
+
round(self.self_belief_error_cnt / (self.total_self_belief_cnt+1e-8), 2),
|
897 |
+
round(self.other_belief_error_cnt / (self.total_other_belief_cnt+1e-8), 2),
|
898 |
+
self.engagement_count,
|
899 |
+
draw_plot(f"{self.name} ({self.model_name})", self.budget_history, self.profit_history),
|
900 |
+
changes_of_plan,
|
901 |
+
budget_error_history,
|
902 |
+
profit_error_history,
|
903 |
+
win_bid_error_history
|
904 |
+
]
|
905 |
+
|
906 |
+
def dialogue_to_chatbot(self):
|
907 |
+
# chatbot: [[Human, AI], [], ...]
|
908 |
+
# only dialogue will be sent to LLMs. chatbot is just for display.
|
909 |
+
assert len(self.dialogue_history) % 2 == 0
|
910 |
+
chatbot = []
|
911 |
+
for i in range(0, len(self.dialogue_history), 2):
|
912 |
+
# if exceeds the length of dialogue, append the last message
|
913 |
+
human_msg = self.dialogue_history[i].content
|
914 |
+
ai_msg = self.dialogue_history[i+1].content
|
915 |
+
if ai_msg == '': ai_msg = None
|
916 |
+
if human_msg == '': human_msg = None
|
917 |
+
chatbot.append([human_msg, ai_msg])
|
918 |
+
return chatbot
|
919 |
+
|
920 |
+
|
921 |
+
def draw_plot(title, hedge_list, profit_list):
|
922 |
+
x1 = [str(i) for i in range(len(hedge_list))]
|
923 |
+
x2 = [str(i) for i in range(len(profit_list))]
|
924 |
+
y1 = hedge_list
|
925 |
+
y2 = profit_list
|
926 |
+
|
927 |
+
fig, ax1 = plt.subplots()
|
928 |
+
|
929 |
+
color = 'tab:red'
|
930 |
+
ax1.set_xlabel('Bidding Round')
|
931 |
+
ax1.set_ylabel('Budget Left ($)', color=color)
|
932 |
+
ax1.plot(x1, y1, color=color, marker='o')
|
933 |
+
ax1.tick_params(axis='y', labelcolor=color)
|
934 |
+
|
935 |
+
for i, j in zip(x1, y1):
|
936 |
+
ax1.text(i, j, str(j), color=color)
|
937 |
+
|
938 |
+
ax2 = ax1.twinx()
|
939 |
+
color = 'tab:blue'
|
940 |
+
ax2.set_ylabel('Total Profit ($)', color=color)
|
941 |
+
ax2.plot(x2, y2, color=color, marker='^')
|
942 |
+
ax2.tick_params(axis='y', labelcolor=color)
|
943 |
+
|
944 |
+
for i, j in zip(x2, y2):
|
945 |
+
ax2.text(i, j, str(j), color=color)
|
946 |
+
|
947 |
+
lines1, labels1 = ax1.get_legend_handles_labels()
|
948 |
+
lines2, labels2 = ax2.get_legend_handles_labels()
|
949 |
+
ax2.legend(lines1 + lines2, labels1 + labels2, loc=0)
|
950 |
+
|
951 |
+
# fig.tight_layout()
|
952 |
+
plt.title(title)
|
953 |
+
|
954 |
+
return fig
|
955 |
+
|
956 |
+
|
957 |
+
def bidding_multithread(bidder_list: List[Bidder],
|
958 |
+
instruction_list,
|
959 |
+
func_type,
|
960 |
+
thread_num=5,
|
961 |
+
retry=1):
|
962 |
+
'''
|
963 |
+
auctioneer_msg: either a uniform message (str) or customed (list)
|
964 |
+
'''
|
965 |
+
assert func_type in ['plan', 'bid', 'summarize', 'replan']
|
966 |
+
|
967 |
+
result_queue = queue.Queue()
|
968 |
+
threads = []
|
969 |
+
semaphore = threading.Semaphore(thread_num)
|
970 |
+
|
971 |
+
def run_once(i: int, bidder: Bidder, auctioneer_msg: str):
|
972 |
+
try:
|
973 |
+
semaphore.acquire()
|
974 |
+
if func_type == 'bid':
|
975 |
+
|
976 |
+
result = bidder.bid(auctioneer_msg)
|
977 |
+
elif func_type == 'summarize':
|
978 |
+
result = bidder.summarize(auctioneer_msg)
|
979 |
+
elif func_type == 'plan':
|
980 |
+
result = bidder.init_plan(auctioneer_msg)
|
981 |
+
elif func_type == 'replan':
|
982 |
+
result = bidder.replan(auctioneer_msg)
|
983 |
+
else:
|
984 |
+
raise NotImplementedError(f'func_type {func_type} not implemented')
|
985 |
+
result_queue.put((True, i, result))
|
986 |
+
# except Exception as e:
|
987 |
+
# result_queue.put((False, i, str(trace_back(e))))
|
988 |
+
finally:
|
989 |
+
semaphore.release()
|
990 |
+
|
991 |
+
if isinstance(instruction_list, str):
|
992 |
+
instruction_list = [instruction_list] * len(bidder_list)
|
993 |
+
|
994 |
+
for i, (bidder, msg) in enumerate(zip(bidder_list, instruction_list)):
|
995 |
+
thread = threading.Thread(target=run_once, args=(i, bidder, msg))
|
996 |
+
thread.start()
|
997 |
+
threads.append(thread)
|
998 |
+
|
999 |
+
for thread in threads:
|
1000 |
+
thread.join(timeout=600)
|
1001 |
+
|
1002 |
+
results = [result_queue.get() for _ in range(len(bidder_list))]
|
1003 |
+
|
1004 |
+
errors = []
|
1005 |
+
for success, id, result in results:
|
1006 |
+
if not success:
|
1007 |
+
errors.append((id, result))
|
1008 |
+
|
1009 |
+
if errors:
|
1010 |
+
raise Exception(f"Error(s) in {func_type}:\n" + '\n'.join([f'{i}: {e}' for i, e in errors]))
|
1011 |
+
|
1012 |
+
valid_results = [x[1:] for x in results if x[0]]
|
1013 |
+
valid_results.sort()
|
1014 |
+
|
1015 |
+
return [x for _, x in valid_results]
|
1016 |
+
|
1017 |
+
|
1018 |
+
def bidders_to_chatbots(bidder_list: List[Bidder], profit_report=False):
|
1019 |
+
if profit_report: # usually at the end of an auction
|
1020 |
+
return [x.dialogue_to_chatbot() + [[x.profit_report(), None]] for x in bidder_list]
|
1021 |
+
else:
|
1022 |
+
return [x.dialogue_to_chatbot() for x in bidder_list]
|
1023 |
+
|
1024 |
+
|
1025 |
+
def create_bidders(bidder_info_jsl, auction_hash):
|
1026 |
+
bidder_info_jsl = LoadJsonL(bidder_info_jsl)
|
1027 |
+
bidder_list = []
|
1028 |
+
for info in bidder_info_jsl:
|
1029 |
+
info['auction_hash'] = auction_hash
|
1030 |
+
bidder_list.append(Bidder.create(**info))
|
1031 |
+
return bidder_list
|
src/human_bidder.py
ADDED
@@ -0,0 +1,137 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from typing import List
|
2 |
+
from langchain.schema import (
|
3 |
+
AIMessage,
|
4 |
+
HumanMessage,
|
5 |
+
SystemMessage
|
6 |
+
)
|
7 |
+
from .bidder_base import Bidder, draw_plot
|
8 |
+
from .item_base import Item
|
9 |
+
from langchain.input import get_colored_text
|
10 |
+
import time
|
11 |
+
|
12 |
+
|
13 |
+
class HumanBidder(Bidder):
|
14 |
+
name: str
|
15 |
+
human_name: str = "Adam"
|
16 |
+
budget: int
|
17 |
+
auction_hash: str
|
18 |
+
|
19 |
+
cur_item_id = 0
|
20 |
+
items: list = []
|
21 |
+
withdraw: bool = False
|
22 |
+
|
23 |
+
engagement_count: int = 0
|
24 |
+
original_budget: int = 0
|
25 |
+
profit: int = 0
|
26 |
+
items_won = []
|
27 |
+
|
28 |
+
all_bidders_status = {} # track others' profit
|
29 |
+
|
30 |
+
# essential for demo
|
31 |
+
need_input: bool = False
|
32 |
+
semaphore: int = 0 # if needs input, then semaphore is set as 1, else waits.
|
33 |
+
input_box: str = None # global variable for accepting user input
|
34 |
+
|
35 |
+
# not used
|
36 |
+
model_name: str = 'human'
|
37 |
+
openai_cost = 0
|
38 |
+
desire = ''
|
39 |
+
plan_strategy = ''
|
40 |
+
correct_belief = True
|
41 |
+
|
42 |
+
class Config:
|
43 |
+
arbitrary_types_allowed = True
|
44 |
+
|
45 |
+
def get_plan_instruct(self, items: List[Item]):
|
46 |
+
self.items = items
|
47 |
+
plan_instruct = "As {bidder_name}, you have a total budget of ${budget}. This auction has a total of {item_num} items to be sequentially presented, they are:\n{items_info}".format(
|
48 |
+
bidder_name=self.name,
|
49 |
+
budget=self.budget,
|
50 |
+
item_num=len(items),
|
51 |
+
items_info=self._get_items_value_str(items)
|
52 |
+
)
|
53 |
+
return plan_instruct
|
54 |
+
|
55 |
+
def init_plan(self, plan_instruct: str):
|
56 |
+
# Human = auctioneer, AI = bidder
|
57 |
+
self.dialogue_history += [
|
58 |
+
HumanMessage(content=plan_instruct),
|
59 |
+
AIMessage(content='Got it!')
|
60 |
+
]
|
61 |
+
return ''
|
62 |
+
|
63 |
+
def get_bid_instruct(self, auctioneer_msg, bid_round):
|
64 |
+
self.dialogue_history += [
|
65 |
+
HumanMessage(content=auctioneer_msg),
|
66 |
+
AIMessage(content='')
|
67 |
+
]
|
68 |
+
return auctioneer_msg
|
69 |
+
|
70 |
+
def bid(self, bid_instruct):
|
71 |
+
# wait for the cue to handle user input
|
72 |
+
while self.semaphore <= 0:
|
73 |
+
time.sleep(1)
|
74 |
+
|
75 |
+
self.dialogue_history += [
|
76 |
+
HumanMessage(content=''),
|
77 |
+
AIMessage(content=self.input_box)
|
78 |
+
]
|
79 |
+
self.semaphore -= 1
|
80 |
+
self.need_input = False
|
81 |
+
return self.input_box
|
82 |
+
|
83 |
+
def get_summarize_instruct(self, bidding_history: str, hammer_msg: str, win_lose_msg: str):
|
84 |
+
instruct_summarize = f"{bidding_history}\n\n{hammer_msg}\n{win_lose_msg}"
|
85 |
+
return instruct_summarize
|
86 |
+
|
87 |
+
def summarize(self, instruct_summarize: str):
|
88 |
+
self.dialogue_history += [
|
89 |
+
HumanMessage(content=instruct_summarize),
|
90 |
+
AIMessage(content='Noted.')
|
91 |
+
]
|
92 |
+
self.budget_history.append(self.budget)
|
93 |
+
self.profit_history.append(self.profit)
|
94 |
+
return ''
|
95 |
+
|
96 |
+
def get_replan_instruct(self):
|
97 |
+
return ''
|
98 |
+
|
99 |
+
def replan(self, instruct_replan):
|
100 |
+
self.withdraw = False
|
101 |
+
self.cur_item_id += 1
|
102 |
+
return ''
|
103 |
+
|
104 |
+
def to_monitors(self, as_json=False):
|
105 |
+
items_won = []
|
106 |
+
for item, bid in self.items_won:
|
107 |
+
items_won.append([str(item), bid, item.true_value])
|
108 |
+
if as_json:
|
109 |
+
return {
|
110 |
+
'auction_hash': self.auction_hash,
|
111 |
+
'bidder_name': self.name,
|
112 |
+
'human_name': self.human_name,
|
113 |
+
'model_name': self.model_name,
|
114 |
+
'budget': self.original_budget,
|
115 |
+
'money_left': self.budget,
|
116 |
+
'profit': self.profit,
|
117 |
+
'items_won': items_won,
|
118 |
+
'engagement_count': self.engagement_count,
|
119 |
+
}
|
120 |
+
else:
|
121 |
+
return [
|
122 |
+
self.budget,
|
123 |
+
self.profit,
|
124 |
+
items_won,
|
125 |
+
0,
|
126 |
+
0,
|
127 |
+
round(self.failed_bid_cnt / (self.total_bid_cnt+1e-8), 2),
|
128 |
+
0,
|
129 |
+
0,
|
130 |
+
self.engagement_count,
|
131 |
+
draw_plot(f"{self.name} ({self.model_name})", self.budget_history, self.profit_history),
|
132 |
+
[],
|
133 |
+
[],
|
134 |
+
[],
|
135 |
+
[]
|
136 |
+
]
|
137 |
+
|
src/item_base.py
ADDED
@@ -0,0 +1,50 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import sys
|
2 |
+
sys.path.append('..')
|
3 |
+
from utils import LoadJsonL
|
4 |
+
|
5 |
+
|
6 |
+
class Item():
|
7 |
+
def __init__(self, id: int, name: str, price: int, desc: str, true_value: int):
|
8 |
+
self.id = id
|
9 |
+
self.name = name
|
10 |
+
self.price = price
|
11 |
+
self.desc = desc
|
12 |
+
self.true_value = true_value
|
13 |
+
self._original_price = price
|
14 |
+
|
15 |
+
def get_desc(self):
|
16 |
+
return f"{self.name}, starting at ${int(self.price)}."
|
17 |
+
|
18 |
+
def __repr__(self):
|
19 |
+
return f"{self.name}"
|
20 |
+
|
21 |
+
def __str__(self):
|
22 |
+
return f"{self.name}"
|
23 |
+
|
24 |
+
def info(self):
|
25 |
+
return f"{self.name}: ${int(self.price)} to ${self.true_value}."
|
26 |
+
|
27 |
+
def lower_price(self, percentage: float = 0.2):
|
28 |
+
# lower starting price by 20%
|
29 |
+
self.price = int(self.price * (1 - percentage))
|
30 |
+
|
31 |
+
def reset_price(self):
|
32 |
+
self.price = self._original_price
|
33 |
+
|
34 |
+
|
35 |
+
def create_items(item_info_jsl):
|
36 |
+
'''
|
37 |
+
item_info: a list of dict (name, price, desc, id)
|
38 |
+
'''
|
39 |
+
item_info_jsl = LoadJsonL(item_info_jsl)
|
40 |
+
item_list = []
|
41 |
+
for info in item_info_jsl:
|
42 |
+
item_list.append(Item(**info))
|
43 |
+
return item_list
|
44 |
+
|
45 |
+
|
46 |
+
def item_list_equal(items_1: list, items_2: list):
|
47 |
+
# could be a list of strings (names) or a list of Items
|
48 |
+
item_1_names = [item.name if isinstance(item, Item) else item for item in items_1]
|
49 |
+
item_2_names = [item.name if isinstance(item, Item) else item for item in items_2]
|
50 |
+
return set(item_1_names) == set(item_2_names)
|
src/prompt_base.py
ADDED
@@ -0,0 +1,349 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# for bidder
|
2 |
+
SYSTEM_MESSAGE = """
|
3 |
+
You are {name}, who is attending an ascending-bid auction as a bidder. This auction will have some other bidders to compete with you in bidding wars. The price is gradually raised, bidders drop out until finally only one bidder remains, and that bidder wins the item at this final price. Remember: {desire_desc}.
|
4 |
+
|
5 |
+
Here are some must-know rules for this auction:
|
6 |
+
|
7 |
+
1. Item Values: The true value of an item means its resale value in the broader market, which you don't know. You will have a personal estimation of the item value. However, note that your estimated value could deviate from the true value, due to your potential overestimation or underestimation of this item.
|
8 |
+
2. Winning Bid: The highest bid wins the item. Your profit from winning an item is determined by the difference between the item's true value and your winning bid. You should try to win an item at a bid as minimal as possible to save your budget.
|
9 |
+
""".strip()
|
10 |
+
|
11 |
+
|
12 |
+
_LEARNING_STATEMENT = " and your learnings from previous auctions"
|
13 |
+
|
14 |
+
|
15 |
+
INSTRUCT_PLAN_TEMPLATE = """
|
16 |
+
As {bidder_name}, you have a total budget of ${budget}. This auction has a total of {item_num} items to be sequentially presented, they are:
|
17 |
+
{items_info}
|
18 |
+
|
19 |
+
---
|
20 |
+
|
21 |
+
Please plan for your bidding strategy for the auction based on the information{learning_statement}. A well-thought-out plan positions you advantageously against competitors, allowing you to allocate resources effectively. With a clear strategy, you can make decisions rapidly and confidently, especially under the pressure of the auction environment. Remember: {desire_desc}.
|
22 |
+
|
23 |
+
After articulate your thinking, in you plan, assign a priority level to each item. Present the priorities for all items in a JSON format, each item should be represented as a key-value pair, where the key is the item name and the value is its priority on the scale from 1-3. An example output is: {{"Fixture Y": 3, "Module B": 2, "Product G": 2}}. The descriptions of the priority scale of items are as follows.
|
24 |
+
* 1 - This item is the least important. Consider giving it up if necessary to save money for the rest of the auction.
|
25 |
+
* 2 - This item holds value but isn't a top priority for the bidder. Could bid on it if you have enough budget.
|
26 |
+
* 3 - This item is of utmost importance and is a top priority for the bidder in the rest of the auction.
|
27 |
+
""".strip()
|
28 |
+
|
29 |
+
|
30 |
+
INSTRUCT_BID_TEMPLATE = """
|
31 |
+
Now, the auctioneer says: "{auctioneer_msg}"
|
32 |
+
|
33 |
+
---
|
34 |
+
|
35 |
+
As {bidder_name}, you have to decide whether to bid on this item or withdraw and explain why, according to your plan{learning_statement}. Remember, {desire_desc}.
|
36 |
+
|
37 |
+
Here are some common practices of bidding:
|
38 |
+
1. Showing your interest by bidding with or slightly above the starting price of this item, then gradually increase your bid.
|
39 |
+
2. Think step by step of the pros and cons and the consequences of your action (e.g., remaining budget in future bidding) in order to achieve your primary objective.
|
40 |
+
|
41 |
+
Give your reasons first, then make your final decision clearly. You should either withdraw (saying "I'm out!") or make a higher bid for this item (saying "I bid $xxx!").
|
42 |
+
""".strip()
|
43 |
+
|
44 |
+
|
45 |
+
INSTRUCT_SUMMARIZE_TEMPLATE = """
|
46 |
+
Here is the history of the bidding war of {cur_item}:
|
47 |
+
"{bidding_history}"
|
48 |
+
|
49 |
+
The auctioneer concludes: "{hammer_msg}"
|
50 |
+
|
51 |
+
---
|
52 |
+
|
53 |
+
{win_lose_msg}
|
54 |
+
As {bidder_name}, you have to update the status of the auction based on this round of bidding. Here's your previous status:
|
55 |
+
```
|
56 |
+
{prev_status}
|
57 |
+
```
|
58 |
+
|
59 |
+
Summarize the notable behaviors of all bidders in this round of bidding for future reference. Then, update the status JSON regarding the following information:
|
60 |
+
- 'remaining_budget': The remaining budget of you, expressed as a numerical value.
|
61 |
+
- 'total_profits': The total profits achieved so far for each bidder, where a numerical value following a bidder's name. No equation is needed, just the numerical value.
|
62 |
+
- 'winning_bids': The winning bids for every item won by each bidder, listed as key-value pairs, for example, {{"bidder_name": {{"item_name_1": winning_bid}}, {{"item_name_2": winning_bid}}, ...}}. If a bidder hasn't won any item, then the value for this bidder should be an empty dictionary {{}}.
|
63 |
+
- Only include the bidders mentioned in the given text. If a bidder is not mentioned (e.g. Bidder 4 in the following example), then do not include it in the JSON object.
|
64 |
+
|
65 |
+
After summarizing the bidding history, you must output the current status in a parsible JSON format. An example output looks like:
|
66 |
+
```
|
67 |
+
{{"remaining_budget": 8000, "total_profits": {{"Bidder 1": 1300, "Bidder 2": 1800, "Bidder 3": 0}}, "winning_bids": {{"Bidder 1": {{"Item 2": 1200, "Item 3": 1000}}, "Bidder 2": {{"Item 1": 2000}}, "Bidder 3": {{}}}}}}
|
68 |
+
```
|
69 |
+
""".strip()
|
70 |
+
|
71 |
+
|
72 |
+
INSTRUCT_LEARNING_TEMPLATE = """
|
73 |
+
Review and reflect on the historical data provided from a past auction.
|
74 |
+
|
75 |
+
{past_auction_log}
|
76 |
+
|
77 |
+
Here are your past learnings:
|
78 |
+
|
79 |
+
{past_learnings}
|
80 |
+
|
81 |
+
Based on the auction log, formulate or update your learning points that could be advantageous to your strategies in the future. Your learnings should be strategic, and of universal relevance and practical use for future auctions. Consolidate your learnings into a concise numbered list of sentences.
|
82 |
+
""".strip()
|
83 |
+
|
84 |
+
|
85 |
+
INSTRUCT_REPLAN_TEMPLATE = """
|
86 |
+
The current status of you and other bidders is as follows:
|
87 |
+
```
|
88 |
+
{status_quo}
|
89 |
+
```
|
90 |
+
|
91 |
+
Here are the remaining items in the rest of the auction:
|
92 |
+
"{remaining_items_info}"
|
93 |
+
|
94 |
+
As {bidder_name}, considering the current status{learning_statement}, review your strategies. Adjust your plans based on the outcomes and new information to achieve your primary objective. This iterative process ensures that your approach remains relevant and effective. Please do the following:
|
95 |
+
1. Always remember: {desire_desc}.
|
96 |
+
2. Determine and explain if there's a need to update the priority list of remaining items based on the current status.
|
97 |
+
3. Present the updated priorities in a JSON format, each item should be represented as a key-value pair, where the key is the item name and the value is its priority on the scale from 1-3. An example output is: {{"Fixture Y": 3, "Module B": 2, "Product G": 2}}. The descriptions of the priority scale of items are as follows.
|
98 |
+
* 1 - This item is the least important. Consider giving it up if necessary to save money for the rest of the auction.
|
99 |
+
* 2 - This item holds value but isn't a top priority for the bidder. Could bid on it if you have enough budget.
|
100 |
+
* 3 - This item is of utmost importance and is a top priority for the bidder in the rest of the auction.
|
101 |
+
""".strip()
|
102 |
+
|
103 |
+
|
104 |
+
# for auctioneer
|
105 |
+
PARSE_BID_INSTRUCTION = """
|
106 |
+
Your task is to parse a response from a bidder in an auction, and extract the bidding price from the response. Here are the rules:
|
107 |
+
- If the language model decides to withdraw from the bidding (e.g., saying "I'm out!"), output -1.
|
108 |
+
- If a bidding price is mentioned (e.g., saying "I bid $xxx!"), output that price number (e.g., $xxx).
|
109 |
+
Here is the response:
|
110 |
+
|
111 |
+
{response}
|
112 |
+
|
113 |
+
Don't say anything else other than just a number: either the bidding price (e.g., $xxx, with $) or -1.
|
114 |
+
""".strip()
|
115 |
+
|
116 |
+
|
117 |
+
AUCTION_HISTORY = """
|
118 |
+
## Auction Log
|
119 |
+
|
120 |
+
### 1. Equipment E, starting at $5000.
|
121 |
+
|
122 |
+
#### 1st bid:
|
123 |
+
* Bidder 1: $5500
|
124 |
+
* Bidder 2: $5100
|
125 |
+
* Bidder 3: $5100
|
126 |
+
* Bidder 4: $5500
|
127 |
+
* Bidder 5: $6000
|
128 |
+
|
129 |
+
#### 2nd bid:
|
130 |
+
* Bidder 1: Withdrew
|
131 |
+
* Bidder 2: Withdrew
|
132 |
+
* Bidder 3: Withdrew
|
133 |
+
* Bidder 4: $6500
|
134 |
+
|
135 |
+
#### 3rd bid:
|
136 |
+
* Bidder 5: $7000
|
137 |
+
|
138 |
+
#### 4th bid:
|
139 |
+
* Bidder 4: Withdrew
|
140 |
+
|
141 |
+
#### Hammer price (true value):
|
142 |
+
* Bidder 5: $7000 ($10000)
|
143 |
+
|
144 |
+
### 2. Thingamajig C, starting at $1000.
|
145 |
+
|
146 |
+
#### 1st bid:
|
147 |
+
* Bidder 1: $1500
|
148 |
+
* Bidder 2: Withdrew
|
149 |
+
* Bidder 3: Withdrew
|
150 |
+
* Bidder 4: Withdrew
|
151 |
+
* Bidder 5: Withdrew
|
152 |
+
|
153 |
+
#### Hammer price (true value):
|
154 |
+
* Bidder 1: $1500 ($2000)
|
155 |
+
|
156 |
+
### 3. Component S, starting at $1000.
|
157 |
+
|
158 |
+
#### 1st bid:
|
159 |
+
* Bidder 1: $1200
|
160 |
+
* Bidder 2: $1050
|
161 |
+
* Bidder 3: $1000
|
162 |
+
* Bidder 4: Withdrew
|
163 |
+
* Bidder 5: $1200
|
164 |
+
|
165 |
+
#### 2nd bid:
|
166 |
+
* Bidder 2: Withdrew
|
167 |
+
* Bidder 3: $1300
|
168 |
+
* Bidder 5: $1300
|
169 |
+
|
170 |
+
#### 3rd bid:
|
171 |
+
* Bidder 1: Withdrew
|
172 |
+
* Bidder 3: $1400
|
173 |
+
|
174 |
+
#### 4th bid:
|
175 |
+
* Bidder 5: Withdrew
|
176 |
+
|
177 |
+
#### Hammer price (true value):
|
178 |
+
* Bidder 3: $1400 ($2000)
|
179 |
+
|
180 |
+
### 4. Implement G, starting at $1000.
|
181 |
+
|
182 |
+
#### 1st bid:
|
183 |
+
* Bidder 1: $1100
|
184 |
+
* Bidder 2: $1000
|
185 |
+
* Bidder 3: $1100
|
186 |
+
* Bidder 4: Withdrew
|
187 |
+
* Bidder 5: $1500
|
188 |
+
|
189 |
+
#### 2nd bid:
|
190 |
+
* Bidder 1: Withdrew
|
191 |
+
* Bidder 2: Withdrew
|
192 |
+
* Bidder 3: $1600
|
193 |
+
|
194 |
+
#### 3rd bid:
|
195 |
+
* Bidder 5: $1700
|
196 |
+
|
197 |
+
#### 4th bid:
|
198 |
+
* Bidder 3: Withdrew
|
199 |
+
|
200 |
+
#### Hammer price (true value):
|
201 |
+
* Bidder 5: $1700 ($2000)
|
202 |
+
|
203 |
+
### 5. Piece T, starting at $1000.
|
204 |
+
|
205 |
+
#### 1st bid:
|
206 |
+
* Bidder 1: $1100
|
207 |
+
* Bidder 2: $1000
|
208 |
+
* Bidder 3: $1100
|
209 |
+
* Bidder 4: Withdrew
|
210 |
+
* Bidder 5: $1200
|
211 |
+
|
212 |
+
#### 2nd bid:
|
213 |
+
* Bidder 1: Withdrew
|
214 |
+
* Bidder 2: $1300
|
215 |
+
* Bidder 3: $1300
|
216 |
+
|
217 |
+
#### 3rd bid:
|
218 |
+
* Bidder 2: $1400
|
219 |
+
* Bidder 5: Withdrew
|
220 |
+
|
221 |
+
#### 4th bid:
|
222 |
+
* Bidder 3: $1500
|
223 |
+
|
224 |
+
#### 5th bid:
|
225 |
+
* Bidder 2: Withdrew
|
226 |
+
|
227 |
+
#### Hammer price (true value):
|
228 |
+
* Bidder 3: $1500 ($2000)
|
229 |
+
|
230 |
+
### 6. Doodad D, starting at $1000.
|
231 |
+
|
232 |
+
#### 1st bid:
|
233 |
+
* Bidder 1: Withdrew
|
234 |
+
* Bidder 2: $1000
|
235 |
+
* Bidder 3: Withdrew
|
236 |
+
* Bidder 4: $1010
|
237 |
+
* Bidder 5: $1300
|
238 |
+
|
239 |
+
#### 2nd bid:
|
240 |
+
* Bidder 2: Withdrew
|
241 |
+
* Bidder 4: Withdrew
|
242 |
+
|
243 |
+
#### Hammer price (true value):
|
244 |
+
* Bidder 5: $1300 ($2000)
|
245 |
+
|
246 |
+
### 7. Gizmo F, starting at $1000.
|
247 |
+
|
248 |
+
#### 1st bid:
|
249 |
+
* Bidder 1: $1100
|
250 |
+
* Bidder 2: $1000
|
251 |
+
* Bidder 3: Withdrew
|
252 |
+
* Bidder 4: Withdrew
|
253 |
+
* Bidder 5: Withdrew
|
254 |
+
|
255 |
+
#### 2nd bid:
|
256 |
+
* Bidder 2: $1200
|
257 |
+
|
258 |
+
#### 3rd bid:
|
259 |
+
* Bidder 1: Withdrew
|
260 |
+
|
261 |
+
#### Hammer price (true value):
|
262 |
+
* Bidder 2: $1200 ($2000)
|
263 |
+
|
264 |
+
### 8. Widget A, starting at $1000.
|
265 |
+
|
266 |
+
#### 1st bid:
|
267 |
+
* Bidder 1: $2200
|
268 |
+
* Bidder 2: $1000
|
269 |
+
* Bidder 3: $1100
|
270 |
+
* Bidder 4: Withdrew
|
271 |
+
* Bidder 5: Withdrew
|
272 |
+
|
273 |
+
#### 2nd bid:
|
274 |
+
* Bidder 2: Withdrew
|
275 |
+
* Bidder 3: Withdrew
|
276 |
+
|
277 |
+
#### Hammer price (true value):
|
278 |
+
* Bidder 1: $2200 ($2000)
|
279 |
+
|
280 |
+
### 9. Gadget B, starting at $1000.
|
281 |
+
|
282 |
+
#### 1st bid:
|
283 |
+
* Bidder 1: $1200
|
284 |
+
* Bidder 2: Withdrew
|
285 |
+
* Bidder 3: Withdrew
|
286 |
+
* Bidder 4: $1000
|
287 |
+
* Bidder 5: Withdrew
|
288 |
+
|
289 |
+
#### 2nd bid:
|
290 |
+
* Bidder 4: Withdrew
|
291 |
+
|
292 |
+
#### Hammer price (true value):
|
293 |
+
* Bidder 1: $1200 ($2000)
|
294 |
+
|
295 |
+
### 10. Mechanism J, starting at $5000.
|
296 |
+
|
297 |
+
#### 1st bid:
|
298 |
+
* Bidder 1: Withdrew
|
299 |
+
* Bidder 2: $5000
|
300 |
+
* Bidder 3: $5100
|
301 |
+
* Bidder 4: $6000
|
302 |
+
* Bidder 5: Withdrew
|
303 |
+
|
304 |
+
#### 2nd bid:
|
305 |
+
* Bidder 2: $6500
|
306 |
+
* Bidder 3: $6500
|
307 |
+
|
308 |
+
#### 3rd bid:
|
309 |
+
* Bidder 3: $7000
|
310 |
+
* Bidder 4: $7000
|
311 |
+
|
312 |
+
#### 4th bid:
|
313 |
+
* Bidder 2: $7500
|
314 |
+
* Bidder 3: Withdrew
|
315 |
+
|
316 |
+
#### 5th bid:
|
317 |
+
* Bidder 4: $8000
|
318 |
+
|
319 |
+
#### 6th bid:
|
320 |
+
* Bidder 2: $8500
|
321 |
+
|
322 |
+
#### 7th bid:
|
323 |
+
* Bidder 4: Withdrew
|
324 |
+
|
325 |
+
#### Hammer price (true value):
|
326 |
+
* Bidder 2: $8500 ($10000)
|
327 |
+
|
328 |
+
## Personal Report
|
329 |
+
|
330 |
+
* Bidder 1, starting with $10000, has won 3 items in this auction, with a total profit of $1100.:
|
331 |
+
* Won Thingamajig C at $1500 over $1000, with a true value of $2000.
|
332 |
+
* Won Widget A at $2200 over $1000, with a true value of $2000.
|
333 |
+
* Won Gadget B at $1200 over $1000, with a true value of $2000.
|
334 |
+
|
335 |
+
* Bidder 2, starting with $10000, has won 2 items in this auction, with a total profit of $2300.:
|
336 |
+
* Won Gizmo F at $1200 over $1000, with a true value of $2000.
|
337 |
+
* Won Mechanism J at $8500 over $5000, with a true value of $10000.
|
338 |
+
|
339 |
+
* Bidder 3, starting with $10000, has won 2 items in this auction, with a total profit of $1100.:
|
340 |
+
* Won Component S at $1400 over $1000, with a true value of $2000.
|
341 |
+
* Won Piece T at $1500 over $1000, with a true value of $2000.
|
342 |
+
|
343 |
+
* Bidder 4, starting with $10000, has won 0 items in this auction, with a total profit of $0.:
|
344 |
+
|
345 |
+
* Bidder 5, starting with $10000, has won 3 items in this auction, with a total profit of $4000.:
|
346 |
+
* Won Equipment E at $7000 over $5000, with a true value of $10000.
|
347 |
+
* Won Implement G at $1700 over $1000, with a true value of $2000.
|
348 |
+
* Won Doodad D at $1300 over $1000, with a true value of $2000.
|
349 |
+
""".strip()
|
utils.py
ADDED
@@ -0,0 +1,73 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import ujson as json
|
2 |
+
import re
|
3 |
+
import traceback
|
4 |
+
|
5 |
+
|
6 |
+
def trace_back(error_msg):
|
7 |
+
exc = traceback.format_exc()
|
8 |
+
msg = f'[Error]: {error_msg}.\n[Traceback]: {exc}'
|
9 |
+
return msg
|
10 |
+
|
11 |
+
|
12 |
+
def extract_numbered_list(paragraph):
|
13 |
+
# Updated regular expression to match numbered list
|
14 |
+
# It looks for:
|
15 |
+
# - start of line
|
16 |
+
# - one or more digits
|
17 |
+
# - a period or parenthesis
|
18 |
+
# - optional whitespace
|
19 |
+
# - any character (captured in a group) until the end of line or a new number
|
20 |
+
pattern = r"^\s*(\d+[.)]\s?.*?)(?=\s*\d+[.)]|$)"
|
21 |
+
|
22 |
+
matches = re.findall(pattern, paragraph, re.DOTALL | re.MULTILINE)
|
23 |
+
return [match.strip() for match in matches]
|
24 |
+
|
25 |
+
|
26 |
+
def chunks(lst, n):
|
27 |
+
"""Yield successive n-sized chunks from lst."""
|
28 |
+
for i in range(0, len(lst), n):
|
29 |
+
yield lst[i : i + n]
|
30 |
+
|
31 |
+
|
32 |
+
def reset_state_list(*states):
|
33 |
+
empty = [None for _ in states[1:]]
|
34 |
+
return [[]] + empty
|
35 |
+
|
36 |
+
|
37 |
+
def LoadJsonL(filename):
|
38 |
+
if isinstance(filename, str):
|
39 |
+
jsl = []
|
40 |
+
with open(filename) as f:
|
41 |
+
for line in f:
|
42 |
+
jsl.append(json.loads(line))
|
43 |
+
return jsl
|
44 |
+
else:
|
45 |
+
return filename
|
46 |
+
|
47 |
+
|
48 |
+
def extract_jsons_from_text(text):
|
49 |
+
json_dicts = []
|
50 |
+
stack = []
|
51 |
+
start_index = None
|
52 |
+
|
53 |
+
for i, char in enumerate(text):
|
54 |
+
if char == '{':
|
55 |
+
stack.append(char)
|
56 |
+
if start_index is None:
|
57 |
+
start_index = i
|
58 |
+
elif char == '}':
|
59 |
+
if stack:
|
60 |
+
stack.pop()
|
61 |
+
if not stack and start_index is not None:
|
62 |
+
json_candidate = text[start_index:i+1]
|
63 |
+
try:
|
64 |
+
parsed_json = json.loads(json_candidate)
|
65 |
+
json_dicts.append(parsed_json)
|
66 |
+
start_index = None
|
67 |
+
except json.JSONDecodeError:
|
68 |
+
pass
|
69 |
+
finally:
|
70 |
+
start_index = None
|
71 |
+
|
72 |
+
if len(json_dicts) == 0: json_dicts = [{}]
|
73 |
+
return json_dicts
|