From 8c6d1b1c8bfef4e14735f97cfa718970af6bff77 Mon Sep 17 00:00:00 2001 From: Anandaroop Roy Date: Fri, 12 Apr 2024 20:46:08 -0400 Subject: [PATCH 1/4] feat: add openai client --- .env.example | 1 + package.json | 3 +- yarn.lock | 163 +++++++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 166 insertions(+), 1 deletion(-) create mode 100644 .env.example diff --git a/.env.example b/.env.example new file mode 100644 index 0000000..ddda7fc --- /dev/null +++ b/.env.example @@ -0,0 +1 @@ +OPENAI_API_KEY=REPLACE diff --git a/package.json b/package.json index ce85229..84b6b43 100644 --- a/package.json +++ b/package.json @@ -25,6 +25,7 @@ "prepare": "husky" }, "dependencies": { - "dotenv": "^16.4.5" + "dotenv": "^16.4.5", + "openai": "^4.33.0" } } diff --git a/yarn.lock b/yarn.lock index e166b27..2d5c687 100644 --- a/yarn.lock +++ b/yarn.lock @@ -199,6 +199,28 @@ resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.15.tgz#596a1747233694d50f6ad8a7869fcb6f56cf5841" integrity sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA== +"@types/node-fetch@^2.6.4": + version "2.6.11" + resolved "https://registry.yarnpkg.com/@types/node-fetch/-/node-fetch-2.6.11.tgz#9b39b78665dae0e82a08f02f4967d62c66f95d24" + integrity sha512-24xFj9R5+rfQJLRyM56qh+wnVSYhyXC2tkoBndtY0U+vubqNsYXGjufB2nn8Q6gt0LrARwL6UBtMCSVCwl4B1g== + dependencies: + "@types/node" "*" + form-data "^4.0.0" + +"@types/node@*": + version "20.12.7" + resolved "https://registry.yarnpkg.com/@types/node/-/node-20.12.7.tgz#04080362fa3dd6c5822061aa3124f5c152cff384" + integrity sha512-wq0cICSkRLVaf3UGLMGItu/PtdY7oaXaI/RVU+xliKVOtRna3PRY57ZDfztpDL0n11vfymMUnXv8QwYCO7L1wg== + dependencies: + undici-types "~5.26.4" + +"@types/node@^18.11.18": + version "18.19.31" + resolved "https://registry.yarnpkg.com/@types/node/-/node-18.19.31.tgz#b7d4a00f7cb826b60a543cebdbda5d189aaecdcd" + integrity sha512-ArgCD39YpyyrtFKIqMDvjz79jto5fcI/SVUs2HwB+f0dAzq68yqOdyaSivLiLugSziTpNXLQrVb7RZFmdZzbhA== + dependencies: + undici-types "~5.26.4" + "@types/semver@^7.5.8": version "7.5.8" resolved "https://registry.yarnpkg.com/@types/semver/-/semver-7.5.8.tgz#8268a8c57a3e4abd25c165ecd36237db7948a55e" @@ -290,6 +312,13 @@ "@typescript-eslint/types" "7.6.0" eslint-visitor-keys "^3.4.3" +abort-controller@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/abort-controller/-/abort-controller-3.0.0.tgz#eaf54d53b62bae4138e809ca225c8439a6efb392" + integrity sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg== + dependencies: + event-target-shim "^5.0.0" + acorn-jsx@^5.3.2: version "5.3.2" resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz#7ed5bb55908b3b2f1bc55c6af1653bada7f07937" @@ -300,6 +329,13 @@ acorn@^8.11.3: resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.11.3.tgz#71e0b14e13a4ec160724b38fb7b0f233b1b81d7a" integrity sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg== +agentkeepalive@^4.2.1: + version "4.5.0" + resolved "https://registry.yarnpkg.com/agentkeepalive/-/agentkeepalive-4.5.0.tgz#2673ad1389b3c418c5a20c5d7364f93ca04be923" + integrity sha512-5GG/5IbQQpC9FpkRGsSvZI5QYeSCzlJHdpBQntCsuTOxhKD8lqKhrleg2Yi7yvMIf82Ycmmqln9U8V9qwEiJew== + dependencies: + humanize-ms "^1.2.1" + ajv@^6.12.4: version "6.12.6" resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" @@ -347,6 +383,11 @@ array-union@^2.1.0: resolved "https://registry.yarnpkg.com/array-union/-/array-union-2.1.0.tgz#b798420adbeb1de828d84acd8a2e23d3efe85e8d" integrity sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw== +asynckit@^0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" + integrity sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q== + balanced-match@^1.0.0: version "1.0.2" resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" @@ -424,6 +465,13 @@ colorette@^2.0.20: resolved "https://registry.yarnpkg.com/colorette/-/colorette-2.0.20.tgz#9eb793e6833067f7235902fcd3b09917a000a95a" integrity sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w== +combined-stream@^1.0.8: + version "1.0.8" + resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f" + integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg== + dependencies: + delayed-stream "~1.0.0" + commander@11.1.0: version "11.1.0" resolved "https://registry.yarnpkg.com/commander/-/commander-11.1.0.tgz#62fdce76006a68e5c1ab3314dc92e800eb83d906" @@ -455,6 +503,11 @@ deep-is@^0.1.3: resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.4.tgz#a6f2dce612fadd2ef1f519b73551f17e85199831" integrity sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ== +delayed-stream@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" + integrity sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ== + dir-glob@^3.0.1: version "3.0.1" resolved "https://registry.yarnpkg.com/dir-glob/-/dir-glob-3.0.1.tgz#56dbf73d992a4a93ba1584f4534063fd2e41717f" @@ -602,6 +655,11 @@ esutils@^2.0.2: resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64" integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g== +event-target-shim@^5.0.0: + version "5.0.1" + resolved "https://registry.yarnpkg.com/event-target-shim/-/event-target-shim-5.0.1.tgz#5d4d3ebdf9583d63a5333ce2deb7480ab2b05789" + integrity sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ== + eventemitter3@^5.0.1: version "5.0.1" resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-5.0.1.tgz#53f5ffd0a492ac800721bb42c66b841de96423c4" @@ -690,6 +748,28 @@ flatted@^3.2.9: resolved "https://registry.yarnpkg.com/flatted/-/flatted-3.3.1.tgz#21db470729a6734d4997002f439cb308987f567a" integrity sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw== +form-data-encoder@1.7.2: + version "1.7.2" + resolved "https://registry.yarnpkg.com/form-data-encoder/-/form-data-encoder-1.7.2.tgz#1f1ae3dccf58ed4690b86d87e4f57c654fbab040" + integrity sha512-qfqtYan3rxrnCk1VYaA4H+Ms9xdpPqvLZa6xmMgFvhO32x7/3J/ExcTd6qpxM0vH2GdMI+poehyBZvqfMTto8A== + +form-data@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/form-data/-/form-data-4.0.0.tgz#93919daeaf361ee529584b9b31664dc12c9fa452" + integrity sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww== + dependencies: + asynckit "^0.4.0" + combined-stream "^1.0.8" + mime-types "^2.1.12" + +formdata-node@^4.3.2: + version "4.4.1" + resolved "https://registry.yarnpkg.com/formdata-node/-/formdata-node-4.4.1.tgz#23f6a5cb9cb55315912cbec4ff7b0f59bbd191e2" + integrity sha512-0iirZp3uVDjVGt9p49aTaqjk84TrglENEDuqfdlZQ1roC9CWlPk6Avf8EEnZNcAqPonwkG35x4n3ww/1THYAeQ== + dependencies: + node-domexception "1.0.0" + web-streams-polyfill "4.0.0-beta.3" + fsevents@~2.3.3: version "2.3.3" resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.3.tgz#cac6407785d03675a2a5e1a5305c697b347d90d6" @@ -763,6 +843,13 @@ human-signals@^5.0.0: resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-5.0.0.tgz#42665a284f9ae0dade3ba41ebc37eb4b852f3a28" integrity sha512-AXcZb6vzzrFAUE61HnN4mpLqd/cSIwNQjtNWR0euPm6y0iqx3G4gOXaIDdtdDwZmhwe82LA6+zinmW4UBWVePQ== +humanize-ms@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/humanize-ms/-/humanize-ms-1.2.1.tgz#c46e3159a293f6b896da29316d8b6fe8bb79bbed" + integrity sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ== + dependencies: + ms "^2.0.0" + husky@^9.0.11: version "9.0.11" resolved "https://registry.yarnpkg.com/husky/-/husky-9.0.11.tgz#fc91df4c756050de41b3e478b2158b87c1e79af9" @@ -948,6 +1035,18 @@ micromatch@4.0.5, micromatch@^4.0.4: braces "^3.0.2" picomatch "^2.3.1" +mime-db@1.52.0: + version "1.52.0" + resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70" + integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg== + +mime-types@^2.1.12: + version "2.1.35" + resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a" + integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw== + dependencies: + mime-db "1.52.0" + mimic-fn@^2.1.0: version "2.1.0" resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b" @@ -977,11 +1076,28 @@ ms@2.1.2: resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009" integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== +ms@^2.0.0: + version "2.1.3" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" + integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== + natural-compare@^1.4.0: version "1.4.0" resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== +node-domexception@1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/node-domexception/-/node-domexception-1.0.0.tgz#6888db46a1f71c0b76b3f7555016b63fe64766e5" + integrity sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ== + +node-fetch@^2.6.7: + version "2.7.0" + resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-2.7.0.tgz#d0f0fa6e3e2dc1d27efcd8ad99d550bda94d187d" + integrity sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A== + dependencies: + whatwg-url "^5.0.0" + npm-run-path@^5.1.0: version "5.3.0" resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-5.3.0.tgz#e23353d0ebb9317f174e93417e4a4d82d0249e9f" @@ -1003,6 +1119,20 @@ onetime@^6.0.0: dependencies: mimic-fn "^4.0.0" +openai@^4.33.0: + version "4.33.0" + resolved "https://registry.yarnpkg.com/openai/-/openai-4.33.0.tgz#8c33da687d4a7f3dd7576179318341615394c79d" + integrity sha512-Sh4KvplkvkAREuhb8yZpohqsOo08cBBu6LNWLD8YyMxe8yCxbE+ouJYUs1X2oDPrzQGANj0rFNQYiwW9gWLBOg== + dependencies: + "@types/node" "^18.11.18" + "@types/node-fetch" "^2.6.4" + abort-controller "^3.0.0" + agentkeepalive "^4.2.1" + form-data-encoder "1.7.2" + formdata-node "^4.3.2" + node-fetch "^2.6.7" + web-streams-polyfill "^3.2.1" + optionator@^0.9.3: version "0.9.3" resolved "https://registry.yarnpkg.com/optionator/-/optionator-0.9.3.tgz#007397d44ed1872fdc6ed31360190f81814e2c64" @@ -1228,6 +1358,11 @@ to-regex-range@^5.0.1: dependencies: is-number "^7.0.0" +tr46@~0.0.3: + version "0.0.3" + resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a" + integrity sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw== + ts-api-utils@^1.3.0: version "1.3.0" resolved "https://registry.yarnpkg.com/ts-api-utils/-/ts-api-utils-1.3.0.tgz#4b490e27129f1e8e686b45cc4ab63714dc60eea1" @@ -1264,6 +1399,11 @@ typescript@^5.4.5: resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.4.5.tgz#42ccef2c571fdbd0f6718b1d1f5e6e5ef006f611" integrity sha512-vcI4UpRgg81oIRUFwR0WSIHKt11nJ7SAVlYNIu+QpqeyXP+gpQJy/Z4+F0aGxSE4MqwjyXvW/TzgkLAx2AGHwQ== +undici-types@~5.26.4: + version "5.26.5" + resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617" + integrity sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA== + uri-js@^4.2.2: version "4.4.1" resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" @@ -1271,6 +1411,29 @@ uri-js@^4.2.2: dependencies: punycode "^2.1.0" +web-streams-polyfill@4.0.0-beta.3: + version "4.0.0-beta.3" + resolved "https://registry.yarnpkg.com/web-streams-polyfill/-/web-streams-polyfill-4.0.0-beta.3.tgz#2898486b74f5156095e473efe989dcf185047a38" + integrity sha512-QW95TCTaHmsYfHDybGMwO5IJIM93I/6vTRk+daHTWFPhwh+C8Cg7j7XyKrwrj8Ib6vYXe0ocYNrmzY4xAAN6ug== + +web-streams-polyfill@^3.2.1: + version "3.3.3" + resolved "https://registry.yarnpkg.com/web-streams-polyfill/-/web-streams-polyfill-3.3.3.tgz#2073b91a2fdb1fbfbd401e7de0ac9f8214cecb4b" + integrity sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw== + +webidl-conversions@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871" + integrity sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ== + +whatwg-url@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-5.0.0.tgz#966454e8765462e37644d3626f6742ce8b70965d" + integrity sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw== + dependencies: + tr46 "~0.0.3" + webidl-conversions "^3.0.0" + which@^2.0.1: version "2.0.2" resolved "https://registry.yarnpkg.com/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1" From c2ce3d5bb15642a2f1f354bc3dce33fdc5a8cefd Mon Sep 17 00:00:00 2001 From: Anandaroop Roy Date: Fri, 12 Apr 2024 20:52:33 -0400 Subject: [PATCH 2/4] feat: add function-calling example code --- src/01-function-calling/example.ts | 49 ++++++++++++++++++++++++++++++ 1 file changed, 49 insertions(+) create mode 100644 src/01-function-calling/example.ts diff --git a/src/01-function-calling/example.ts b/src/01-function-calling/example.ts new file mode 100644 index 0000000..87a50b4 --- /dev/null +++ b/src/01-function-calling/example.ts @@ -0,0 +1,49 @@ +/* + * Sample code from https://platform.openai.com/docs/api-reference/chat/create, lightly TS-ified + */ + +// eslint-disable-next-line @typescript-eslint/no-var-requires + +import dotenv from "dotenv" +import OpenAI from "openai" + +dotenv.config() +const openai = new OpenAI() + +async function main() { + const messages: OpenAI.Chat.Completions.ChatCompletionMessageParam[] = [ + { role: "user", content: "What's the weather like in Boston today?" }, + ] + + const tools: OpenAI.Chat.Completions.ChatCompletionTool[] = [ + { + type: "function", + function: { + name: "get_current_weather", + description: "Get the current weather in a given location", + parameters: { + type: "object", + properties: { + location: { + type: "string", + description: "The city and state, e.g. San Francisco, CA", + }, + unit: { type: "string", enum: ["celsius", "fahrenheit"] }, + }, + required: ["location"], + }, + }, + }, + ] + + const response = await openai.chat.completions.create({ + model: "gpt-3.5-turbo", + messages, + tools, + tool_choice: "auto", + }) + + console.log(response.choices[0]) +} + +main() From ab916422233c4852aaba95ffe95aa8875ec277b9 Mon Sep 17 00:00:00 2001 From: Anandaroop Roy Date: Fri, 12 Apr 2024 20:56:59 -0400 Subject: [PATCH 3/4] feat: add function-calling experiment --- src/01-function-calling/index.ts | 248 +++++++++++++++++++++++++++++++ 1 file changed, 248 insertions(+) create mode 100644 src/01-function-calling/index.ts diff --git a/src/01-function-calling/index.ts b/src/01-function-calling/index.ts new file mode 100644 index 0000000..4cbbc0c --- /dev/null +++ b/src/01-function-calling/index.ts @@ -0,0 +1,248 @@ +/* + * This example demonstrates how to use the OpenAI API to create a chat completion that can call functions. + * + * The toy use case here is to ask for a list of artists. Depending on how the user phrases the question, the LLM + * will determine the correct function to call, then shape up the results into a nicely formatted response. + * + * Usage examples: + * + * bun run src/01-function-calling/index.ts "who is popular on artsy right now?" | jq + * bun run src/01-function-calling/index.ts "who is artsy into these days?" | jq + * bun run src/01-function-calling/index.ts "who is new on artsy?" | jq + * bun run src/01-function-calling/index.ts "give me the last 17 artists on artsy, alphabetically" | jq + */ + +import dotenv from "dotenv" +import OpenAI from "openai" + +dotenv.config() +const openai = new OpenAI() // uses `OPENAI_API_KEY` from .env + +/* + * Get user input + */ + +const input = + process.argv.slice(2).join(" ") || "Who's trending on Artsy right now?" + +/* + * Create the message payload for the chat completion + */ + +const messages: OpenAI.Chat.Completions.ChatCompletionMessageParam[] = [ + { + role: "system", + content: + `You are a helpful assistant that can provide information about the art world via Artsy's platform. ` + + ` ` + + `If you receive a question that cannot be answered with one of your known tools, do not force a tool response. ` + + `Instead ask for clarification or say that you don't know how to answer the question. ` + + ` `, + }, + { + role: "user", + content: input, + }, +] + +/* + * Define the tools that the chat completion can use: + * + * 1. get_artists: Get a list of artists on Artsy + * 2. get_curated_artists: Get a list of curated artists on Artsy + */ + +const tools: OpenAI.Chat.Completions.ChatCompletionTool[] = [ + { + type: "function", + function: { + name: "get_artists", + description: `Get a list of artists on Artsy. Artists may be sorted chronologically by creation date, alphabetically by name, or in descending order of a popularity/trending score.`, + parameters: { + type: "object", + properties: { + size: { + type: "integer", + description: "The number of artists to return", + default: 5, + minimum: 1, + maximum: 20, + }, + sort: { + type: "string", + description: "The sort order in which to return artists", + default: "SORTABLE_ID_ASC", + enum: [ + "CREATED_AT_ASC", + "CREATED_AT_DESC", + "SORTABLE_ID_ASC", + "SORTABLE_ID_DESC", + "TRENDING_DESC", + ], + }, + }, + }, + }, + }, + { + type: "function", + function: { + name: "get_curated_artists", + description: `Get a list of curated artists on Artsy. These are artists whose works have been highlighted by Artsy curators, and may change from week to week.`, + parameters: { + type: "object", + properties: { + size: { + type: "integer", + description: "The number of artists to return", + default: 5, + minimum: 1, + maximum: 20, + }, + }, + }, + }, + }, +] + +async function main() { + /* + * Call the chat completion and get the initial response, which may be a simple response or a function call + */ + + const response = await openai.chat.completions.create({ + model: "gpt-3.5-turbo", + temperature: 0, + messages, + tools, + tool_choice: "auto", + }) + + console.log(JSON.stringify(messages)) + console.log(JSON.stringify(response.choices[0].message)) + + /* + * If the response WAS a function call, then actually call the function (locally defined, further down) to get the result from Artsy's API + */ + + if (response.choices[0].finish_reason === "tool_calls") { + const name = response.choices[0].message.tool_calls?.[0].function.name + const args = JSON.parse( + response.choices[0].message.tool_calls?.[0].function.arguments || "null" + ) + + let artists + + if (name === "get_artists") { + artists = await get_artists(args) + console.log(JSON.stringify(artists)) + } + + if (name === "get_curated_artists") { + artists = await get_curated_artists(args) + console.log(JSON.stringify(artists)) + } + + if (artists) { + const response = await openai.chat.completions.create({ + model: "gpt-3.5-turbo", + temperature: 0, + messages: [ + { + role: "user", + content: ` + Based on the following JSON context, return a nicely formatted bullet list of artists. + Each artist entry should contain a name, nationality, birth/death dates, as well as + a link to their Artsy profile in the form https://www.artsy.net/artist/. + + An example entry would look like this: + - [Vincent van Gogh](https://www.artsy.net/artist/vincent-van-gogh) (Dutch, 1853–1890) + + Make sure you use all of the items in the context. + + Context + ''' + ${JSON.stringify(artists, null, 2)} + ''' + `, + }, + ], + }) + + console.error(response.choices[0].message.content) + } + } +} + +/* + * Define the get_artists() and get_curated_artists() functions that can be called by the chat completion + */ + +async function get_artists(args: { size: number; sort: string }) { + const query = `query GetArtists($size: Int!, $sort: ArtistSorts) { + artists(size: $size, sort: $sort) { + slug + name + formattedNationalityAndBirthday + counts { + forSaleArtworks + } + } + }` + + const variables = { + size: args.size, + sort: args.sort, + } + + const response = await metaphysics({ query, variables }) + return response +} + +async function get_curated_artists(args: { size: number }) { + const query = `query GetCuratedArtists($size: Int!) { + curatedTrendingArtists(first: $size) { + edges { + node { + slug + name + formattedNationalityAndBirthday + counts { + forSaleArtworks + } + } + } + } + }` + + const variables = { + size: args.size, + } + + const response = await metaphysics({ query, variables }) + return response +} + +/* + * Define the API helpers the the function calls will make use of + */ + +async function metaphysics(args: { + query: string + variables: Record +}) { + const { query, variables } = args + + const url = "https://metaphysics-production.artsy.net/v2" + const headers = { + "Content-Type": "application/json", + } + const body = JSON.stringify({ query, variables }) + const options = { method: "POST", headers, body } + + const response = await fetch(url, options) + const json = await response.json() + return json +} + +main() From 153bdf8094c3fe6b544c0ecddb3840e90f20c396 Mon Sep 17 00:00:00 2001 From: Anandaroop Roy Date: Mon, 15 Apr 2024 11:28:16 -0400 Subject: [PATCH 4/4] chore: dump the sample code which doesn't even work --- src/01-function-calling/example.ts | 49 ------------------------------ 1 file changed, 49 deletions(-) delete mode 100644 src/01-function-calling/example.ts diff --git a/src/01-function-calling/example.ts b/src/01-function-calling/example.ts deleted file mode 100644 index 87a50b4..0000000 --- a/src/01-function-calling/example.ts +++ /dev/null @@ -1,49 +0,0 @@ -/* - * Sample code from https://platform.openai.com/docs/api-reference/chat/create, lightly TS-ified - */ - -// eslint-disable-next-line @typescript-eslint/no-var-requires - -import dotenv from "dotenv" -import OpenAI from "openai" - -dotenv.config() -const openai = new OpenAI() - -async function main() { - const messages: OpenAI.Chat.Completions.ChatCompletionMessageParam[] = [ - { role: "user", content: "What's the weather like in Boston today?" }, - ] - - const tools: OpenAI.Chat.Completions.ChatCompletionTool[] = [ - { - type: "function", - function: { - name: "get_current_weather", - description: "Get the current weather in a given location", - parameters: { - type: "object", - properties: { - location: { - type: "string", - description: "The city and state, e.g. San Francisco, CA", - }, - unit: { type: "string", enum: ["celsius", "fahrenheit"] }, - }, - required: ["location"], - }, - }, - }, - ] - - const response = await openai.chat.completions.create({ - model: "gpt-3.5-turbo", - messages, - tools, - tool_choice: "auto", - }) - - console.log(response.choices[0]) -} - -main()