Compare commits
	
		
			786 Commits
		
	
	
		
			v0.0.4
			...
			mattw/howt
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 4522109b11 | ||
|   | b2974a7095 | ||
|   | 3c975f898f | ||
|   | 9245c8a1df | ||
|   | 7a537cdca9 | ||
|   | 257ffeb997 | ||
|   | 9b513bb6b1 | ||
|   | 042100f797 | ||
|   | 7804b8fab9 | ||
|   | 56497663c8 | ||
|   | e1afcb8af2 | ||
|   | 385eeea357 | ||
|   | 8a41b244e8 | ||
|   | 92578798bb | ||
|   | 788637918a | ||
|   | c413a55093 | ||
|   | 630bb75d2a | ||
|   | a2055a1e93 | ||
|   | b599946b74 | ||
|   | aca2d65b82 | ||
|   | b5e08e3373 | ||
|   | 274d5a5fdf | ||
|   | fc6b49be32 | ||
|   | 77295f716e | ||
|   | 615f7d1dea | ||
|   | cdf5e106ae | ||
|   | a85329f59a | ||
|   | f2ba1311aa | ||
|   | 65dcd0ce35 | ||
|   | 0040f543a2 | ||
|   | 767f9bdbbb | ||
|   | f7f5169c94 | ||
|   | 2cfffea02e | ||
|   | f6e98334e4 | ||
|   | ab0668293c | ||
|   | af4cf55884 | ||
|   | d6786f2945 | ||
|   | 38dc2f79bc | ||
|   | cb961c87ca | ||
|   | 0560b28a8d | ||
|   | 10199c5987 | ||
|   | 288814d3e4 | ||
|   | 04733438da | ||
|   | 711e891f0f | ||
|   | 090d08422b | ||
|   | 5b84404c64 | ||
|   | 8544edca21 | ||
|   | 5d22319a2c | ||
|   | 2130c0708b | ||
|   | 61ff1946e6 | ||
|   | d06bc0cb6e | ||
|   | d104b7e997 | ||
|   | 9e2de1bd2c | ||
|   | dc87e9c9ae | ||
|   | 367cb68dc1 | ||
|   | c02c0cd483 | ||
|   | 1852755154 | ||
|   | 6f2ce74231 | ||
|   | 6edcc5c79f | ||
|   | b1f7123301 | ||
|   | 1fbf3585d6 | ||
|   | 99d5161e8a | ||
|   | ea8380be45 | ||
|   | 4f25092dc1 | ||
|   | 4fc10acce9 | ||
|   | 0a4f21c0a7 | ||
|   | 9abb66254a | ||
|   | 1d0ebe67e8 | ||
|   | a1b2d95f96 | ||
|   | c0b1bf7537 | ||
|   | cdfeb165ca | ||
|   | 92d454ec5f | ||
|   | 9333b0cc82 | ||
|   | 9771b1ec51 | ||
|   | 76db4a49cf | ||
|   | 4aa0976a2e | ||
|   | 92c20fdae6 | ||
|   | c951da7096 | ||
|   | 24d82a23a2 | ||
|   | f40b3de758 | ||
|   | 5f4008c296 | ||
|   | 6ae33d8141 | ||
|   | c5664c1fef | ||
|   | 958a5a8184 | ||
|   | 8608eb4760 | ||
|   | a2b210130f | ||
|   | ed20837f9a | ||
|   | 1db2a61dd0 | ||
|   | 2ded8ab206 | ||
|   | e6b3648bbf | ||
|   | 0625e805f0 | ||
|   | c38ec5befb | ||
|   | c577721a43 | ||
|   | 29c056ea39 | ||
|   | 9fc3bba9cf | ||
|   | 7774ed4ae6 | ||
|   | 11f920f209 | ||
|   | 6e6b655956 | ||
|   | 110ae89a6c | ||
|   | 5e388f931e | ||
|   | d5ad41dd7b | ||
|   | d294a11bc9 | ||
|   | 93d887e4bc | ||
|   | 5306b0269d | ||
|   | 7de0c8345d | ||
|   | 1b9dcab3ab | ||
|   | 86279f4ae3 | ||
|   | b934bf23e6 | ||
|   | 2b8ef455ad | ||
|   | 0c5f47177c | ||
|   | 1210db2924 | ||
|   | d0854bf1e6 | ||
|   | 8396463255 | ||
|   | a027bbf4d7 | ||
|   | ed94a3dd02 | ||
|   | f14f62ab3b | ||
|   | 0fb5268496 | ||
|   | c65edb1506 | ||
|   | 1605af32ec | ||
|   | ee3032ad89 | ||
|   | 5b7a27281d | ||
|   | d2a784e33e | ||
|   | 413a2e4f91 | ||
|   | a92fdff620 | ||
|   | b5614f3ebc | ||
|   | 8b2ba9cab8 | ||
|   | e29662ab5c | ||
|   | cbc40aa996 | ||
|   | 5cb82540c9 | ||
|   | d7849a1dc9 | ||
|   | 01c44d687e | ||
|   | 9b12a511ca | ||
|   | e20362e0d5 | ||
|   | c928ceb927 | ||
|   | e1a0846483 | ||
|   | f997e29e45 | ||
|   | 87d9efb364 | ||
|   | 93d3a2568d | ||
|   | 5a81390b24 | ||
|   | a89ef99aed | ||
|   | dc0c725ceb | ||
|   | 5d71bda478 | ||
|   | 88897a90e4 | ||
|   | 9df31c3518 | ||
|   | 2044f9d4da | ||
|   | 0d186f3b33 | ||
|   | 82f5b66c01 | ||
|   | c986694367 | ||
|   | 058d0cd04b | ||
|   | ee1c994d15 | ||
|   | 4cba75efc5 | ||
|   | 8c83701e9f | ||
|   | 6137b12799 | ||
|   | 1fabba474b | ||
|   | 765770efdb | ||
|   | 9297ff8330 | ||
|   | ee4fd16f2c | ||
|   | a9ed7cc6aa | ||
|   | 6c6a31a1e8 | ||
|   | fc6ec356fc | ||
|   | 1255bc9b45 | ||
|   | 084e4c782a | ||
|   | 58ffa03d8b | ||
|   | 637f8bc6a5 | ||
|   | 499e9007a5 | ||
|   | b9bb5ca288 | ||
|   | 4e8be787c7 | ||
|   | aa45d7c1df | ||
|   | e35565c567 | ||
|   | a5520bfb42 | ||
|   | 2627c464ba | ||
|   | b58d5d16b0 | ||
|   | 24580df958 | ||
|   | 80dd44e80a | ||
|   | 94e1d96b29 | ||
|   | 66003e1d05 | ||
|   | c345053a8b | ||
|   | 08d7c2a944 | ||
|   | bc9573dcb1 | ||
|   | e53bc57d4d | ||
|   | f0b398d17f | ||
|   | 8efbc5df55 | ||
|   | ccc3e9ac6d | ||
|   | daa4f096f9 | ||
|   | 3ee85f1c6c | ||
|   | 2540c9181c | ||
|   | 83ffb154bc | ||
|   | 9aa192c812 | ||
|   | fc8707686f | ||
|   | f89c23764b | ||
|   | e6881cabd0 | ||
|   | d028853879 | ||
|   | 949553db23 | ||
|   | 0c5a454361 | ||
|   | f59c4d03f7 | ||
|   | 7dee25a07f | ||
|   | f221637053 | ||
|   | 45ac07cd02 | ||
|   | 7d749cc787 | ||
|   | e7e91cd71c | ||
|   | 3920e15386 | ||
|   | 41e976edde | ||
|   | de227b620f | ||
|   | 63def6ca49 | ||
|   | 738fe9c4aa | ||
|   | a8da0bacbe | ||
|   | bf146fb072 | ||
|   | f0f4943577 | ||
|   | 09dd2aeff9 | ||
|   | 07b4074e7b | ||
|   | 61dda6a5e0 | ||
|   | e1f9ced568 | ||
|   | 9795b43d93 | ||
|   | 0980d5c7e3 | ||
|   | 0dae34b6a7 | ||
|   | 83c6be1666 | ||
|   | 1adfa67589 | ||
|   | 790d24eb7b | ||
|   | 7de300856b | ||
|   | 213ffdb548 | ||
|   | d42d88386a | ||
|   | 154f24af91 | ||
|   | a1ecdd36d5 | ||
|   | d18282bfda | ||
|   | 9ae76ba8c9 | ||
|   | 2bc06565c7 | ||
|   | d1c2558f7e | ||
|   | 7b5aefb427 | ||
|   | 06ef90c051 | ||
|   | 7efbc84320 | ||
|   | e9f6df7dca | ||
|   | 7fa6e51686 | ||
|   | 8dc68417e7 | ||
|   | 681f3c4c42 | ||
|   | 59a705525c | ||
|   | 5d3f314b0b | ||
|   | adaa13088b | ||
|   | 62d29b2157 | ||
|   | ed19d10aa5 | ||
|   | 36c2f45c40 | ||
|   | 742226625f | ||
|   | 6bb8a16ccb | ||
|   | a5dbcf2e73 | ||
|   | 9304f0e7a8 | ||
|   | 6578b2f8a1 | ||
|   | 1c8fd627ad | ||
|   | ae950b00f1 | ||
|   | eeb40a672c | ||
|   | 0f541a0367 | ||
|   | 1363f537ce | ||
|   | bc3e21fdc6 | ||
|   | a82eb275ff | ||
|   | f964aea9a2 | ||
|   | 42998d797d | ||
|   | f4432e1dba | ||
|   | 982c535428 | ||
|   | 7df342a6ea | ||
|   | 8bbff2df98 | ||
|   | 16b06699fd | ||
|   | 246dc65417 | ||
|   | 865fceb73c | ||
|   | 72266c7684 | ||
|   | d3b838ce60 | ||
|   | e639a12fa1 | ||
|   | e82fcf30c6 | ||
|   | 495e8b0a6a | ||
|   | 59734ca24d | ||
|   | 22ab7f5f88 | ||
|   | b25dd1795d | ||
|   | 304f2b6c96 | ||
|   | 2ecc3a33c3 | ||
|   | ee6e1df118 | ||
|   | 177b69a211 | ||
|   | dad63f0821 | ||
|   | 041f9ad1a1 | ||
|   | 7a378f8b66 | ||
|   | de0bdd7f29 | ||
|   | b1cececb8e | ||
|   | e0d39fa3bf | ||
|   | 968ced2e71 | ||
|   | 32d1a00017 | ||
|   | 04e2128273 | ||
|   | 2cc634689b | ||
|   | 8f827641b0 | ||
|   | 95187d7e1e | ||
|   | 9ec7e37534 | ||
|   | 2c7f956b38 | ||
|   | a9f6c56652 | ||
|   | 0a892419ad | ||
|   | e3054fc74e | ||
|   | 23c2485044 | ||
|   | 386c66f285 | ||
|   | 3b49315f97 | ||
|   | 5ca05c2e88 | ||
|   | 7eda70f23b | ||
|   | 3d79b414d3 | ||
|   | c84bbf1dd6 | ||
|   | f723bf0879 | ||
|   | cbf725a9ba | ||
|   | 086449b6c7 | ||
|   | 3cbc6a5c01 | ||
|   | 54bb49a502 | ||
|   | cabaada956 | ||
|   | a894cc792d | ||
|   | 519f4d98ef | ||
|   | b963a83559 | ||
|   | bf6688abe6 | ||
|   | 6005b157c2 | ||
|   | 14220d9833 | ||
|   | 8ca50f24f3 | ||
|   | c149fc3143 | ||
|   | afbc763dac | ||
|   | 5dfe91be8b | ||
|   | 9f944c00f1 | ||
|   | 56e87cecb1 | ||
|   | 5ee6116420 | ||
|   | 5d9a4cd251 | ||
|   | 0ebec07569 | ||
|   | 08265515b3 | ||
|   | 67e593e355 | ||
|   | d15c7622b9 | ||
|   | 1deb35ca64 | ||
|   | e2de886831 | ||
|   | f0d7c2f5ea | ||
|   | 12052a7624 | ||
|   | 23e1da778d | ||
|   | 326de48930 | ||
|   | 18f2cb0472 | ||
|   | 53bc36d207 | ||
|   | 4dcf5c3e0b | ||
|   | d1b2f532b9 | ||
|   | e26085b921 | ||
|   | f7b613332c | ||
|   | f594c8eb91 | ||
|   | 76b85bc0e9 | ||
|   | af98a1773f | ||
|   | 9ae9a89883 | ||
|   | 648f0974c6 | ||
|   | fc5230dffa | ||
|   | 2ab20095b3 | ||
|   | f020e1d519 | ||
|   | 4b2d366c37 | ||
|   | 56fd4e4ef2 | ||
|   | 2c8b680b03 | ||
|   | 99b6b60085 | ||
|   | 74f00474e1 | ||
|   | e9a9580bdd | ||
|   | 4c33a9ac67 | ||
|   | 22885aeaee | ||
|   | ed969d2a06 | ||
|   | d9cf18e28d | ||
|   | 1556162c90 | ||
|   | 148f0225c0 | ||
|   | 4e07941b1e | ||
|   | 202c29c21a | ||
|   | c1c871620a | ||
|   | a21a8bef56 | ||
|   | 522726228a | ||
|   | 9770e3b325 | ||
|   | d617823355 | ||
|   | 6ed991c8e2 | ||
|   | e41576e768 | ||
|   | 155c1640f1 | ||
|   | f7d4947573 | ||
|   | 0d7a133b15 | ||
|   | e863066144 | ||
|   | 89a92477ad | ||
|   | 5cda9cdd13 | ||
|   | e5914eb320 | ||
|   | ab78f48ff8 | ||
|   | b1c88eb978 | ||
|   | efae43f932 | ||
|   | d3ee1329e9 | ||
|   | 700c719422 | ||
|   | 55aa4aaf0f | ||
|   | 820f95c4c4 | ||
|   | 3a05d3def7 | ||
|   | edac9c2446 | ||
|   | d9c2687fd0 | ||
|   | 6517bcc53c | ||
|   | 4f54f25b66 | ||
|   | 6a6828bddf | ||
|   | c0e7a3b90e | ||
|   | f27bc261cf | ||
|   | 21e6197c0b | ||
|   | 75d7d681c9 | ||
|   | 81d8d7b73f | ||
|   | 5c0de09a07 | ||
|   | 20bf000e55 | ||
|   | 40d0c4a1dc | ||
|   | be889b2f81 | ||
|   | 7e26a8df31 | ||
|   | 4ab1da38ba | ||
|   | be989d89d1 | ||
|   | bea683e3bf | ||
|   | 178237d37f | ||
|   | 76a678af34 | ||
|   | f65169b13e | ||
|   | 040a5b9750 | ||
|   | 37c9a8eea9 | ||
|   | 6de5d032e1 | ||
|   | d791df75dd | ||
|   | 020a3b3530 | ||
|   | fccf8d179f | ||
|   | 5b5cc9c9f1 | ||
|   | 4b3507f036 | ||
|   | 5ebce03c77 | ||
|   | 5e25f801ed | ||
|   | 8e1234b758 | ||
|   | 10885986b8 | ||
|   | 984c9c628c | ||
|   | 43c40c500e | ||
|   | c4861360ec | ||
|   | 9738ef85db | ||
|   | ac971c56d1 | ||
|   | 8228d166ce | ||
|   | 907e6c56b3 | ||
|   | 868e3b31c7 | ||
|   | 09d8bf6730 | ||
|   | 7a5f3616fd | ||
|   | cff002b824 | ||
|   | 55cf5021f0 | ||
|   | f58caa5ab5 | ||
|   | 82df473ec9 | ||
|   | e184c1d035 | ||
|   | 371d4e5df3 | ||
|   | 1f78e409b4 | ||
|   | 34a88cd776 | ||
|   | 1bee2347be | ||
|   | a027a7dd65 | ||
|   | 22986ccb38 | ||
|   | 884d78ceb3 | ||
|   | 3ceac05108 | ||
|   | 21ddcaa1f1 | ||
|   | f2074ed4c0 | ||
|   | a6f6d18f83 | ||
|   | 34a13a9d05 | ||
|   | 8713ac23a8 | ||
|   | 5eb712f962 | ||
|   | 4dc5b117dd | ||
|   | 931a5f3cb9 | ||
|   | 639288bf2b | ||
|   | d112c15d58 | ||
|   | 1267895e44 | ||
|   | 089d03bc8d | ||
|   | e37f4c4f42 | ||
|   | ab3ced9d32 | ||
|   | 0c52b4509b | ||
|   | 13aace3d34 | ||
|   | 2b3bb41598 | ||
|   | 93492f1e18 | ||
|   | 54ba3e2ceb | ||
|   | 4904cd8bcd | ||
|   | 8a45359ec6 | ||
|   | fb593b7bfc | ||
|   | 2544b8afa1 | ||
|   | ac1b04f271 | ||
|   | 123fdeb919 | ||
|   | 5c82bf95d1 | ||
|   | 38a9b1618c | ||
|   | c18be72a3b | ||
|   | a101fe51a7 | ||
|   | 06fc48ad66 | ||
|   | d93e2f9210 | ||
|   | 31edc829fc | ||
|   | b31104768c | ||
|   | b662d9fd8c | ||
|   | da36196d79 | ||
|   | b9f4d67554 | ||
|   | 42903973b7 | ||
|   | 8f2df948ab | ||
|   | e3fb1fd3f1 | ||
|   | 29b897f525 | ||
|   | 85aeb42869 | ||
|   | c5bcf32823 | ||
|   | a71ff3f6a2 | ||
|   | f0b365a478 | ||
|   | df8048fecd | ||
|   | da2459d519 | ||
|   | bd6d741d87 | ||
|   | 8b1e791820 | ||
|   | 03cff3a225 | ||
|   | cc509a994e | ||
|   | 0e79e52ddd | ||
|   | 6fbb380076 | ||
|   | 8f8b6288ac | ||
|   | b98096389d | ||
|   | 74a5f7e698 | ||
|   | 7a1c3e62dc | ||
|   | da52f5bfdd | ||
|   | 50e87c6691 | ||
|   | e4a970ece1 | ||
|   | 4ca43a694c | ||
|   | 765994362c | ||
|   | 40a25bf8c3 | ||
|   | 1c5a8770ee | ||
|   | daa0d1de7a | ||
|   | 58daeb962a | ||
|   | 528bafa585 | ||
|   | 81f75696e2 | ||
|   | 8bdcf894bd | ||
|   | fe530423a5 | ||
|   | 05e390205b | ||
|   | 872011630a | ||
|   | 203fdbc4b8 | ||
|   | 70e0ab6b3d | ||
|   | 319f078dd9 | ||
|   | 9968153729 | ||
|   | 7da249fcc1 | ||
|   | f529626c6c | ||
|   | 36d6081ed1 | ||
|   | aadedda486 | ||
|   | 671eec6da9 | ||
|   | e72fe7945f | ||
|   | d1c098b038 | ||
|   | 90ba0b80c7 | ||
|   | 39bb25d5f6 | ||
|   | eadee46840 | ||
|   | 2e2e624d21 | ||
|   | ed832ce3b7 | ||
|   | 227da16909 | ||
|   | bd58528fbd | ||
|   | c5e447a359 | ||
|   | fc40a4f166 | ||
|   | 9c7f30d31c | ||
|   | 6ed3ec0cb3 | ||
|   | 47bda0b860 | ||
|   | c75cafdb58 | ||
|   | f5cbcb08e6 | ||
|   | 67b6f8ba86 | ||
|   | 184ad8f057 | ||
|   | 822a0e36eb | ||
|   | 18b6b601ad | ||
|   | 0345070dfa | ||
|   | dffc8b6e09 | ||
|   | 0871083776 | ||
|   | e5b26c3aa2 | ||
|   | 3549676678 | ||
|   | 8fa477fadb | ||
|   | fadf75f99d | ||
|   | 01d155c969 | ||
|   | 5685c16d4e | ||
|   | db77dfe01f | ||
|   | ad3a7d0e2c | ||
|   | 18ffeeec45 | ||
|   | 688661ab9b | ||
|   | 36ad90e8e3 | ||
|   | 6fff59c637 | ||
|   | fee7687cf3 | ||
|   | d3bfb4889c | ||
|   | 1ac38ec89c | ||
|   | 1ad8266473 | ||
|   | f5ac8ddfb4 | ||
|   | cca61181cb | ||
|   | c490416189 | ||
|   | f62a882760 | ||
|   | 3003fc03fc | ||
|   | 32aec66e6a | ||
|   | 35af37a2cb | ||
|   | dbb3174cbc | ||
|   | 31673d26d0 | ||
|   | 8ba0f328af | ||
|   | d0e934b497 | ||
|   | e751e47d70 | ||
|   | 19d0f2b4cc | ||
|   | c48f07f821 | ||
|   | dc642aa07d | ||
|   | f1ff892fdd | ||
|   | 3f2a100465 | ||
|   | 95397416f3 | ||
|   | 8a86aae019 | ||
|   | 24c2c77057 | ||
|   | 5614984f06 | ||
|   | 4c1caa3733 | ||
|   | 12ab8f8f5f | ||
|   | 8ebbd12f21 | ||
|   | 07971759fa | ||
|   | f5f79049c2 | ||
|   | 726bc647b2 | ||
|   | af9039a167 | ||
|   | 07ed69bc37 | ||
|   | 0deb3767fc | ||
|   | cb55fa9270 | ||
|   | 93bc9f17a1 | ||
|   | 536028c35a | ||
|   | aedf3d1f38 | ||
|   | 91d927abc5 | ||
|   | ba8df10a43 | ||
|   | abf614804b | ||
|   | a0dbbb23c4 | ||
|   | 0fd6278446 | ||
|   | 29fe07f0cc | ||
|   | abfc73d31e | ||
|   | 5a5ca8e7ff | ||
|   | f24a6f5988 | ||
|   | fdbef6c95e | ||
|   | 24e43e3212 | ||
|   | 4cb42ca55e | ||
|   | ec5e22ac85 | ||
|   | ed89da92b4 | ||
|   | a3297fed41 | ||
|   | 88c55199f8 | ||
|   | c448443813 | ||
|   | efacd45fc5 | ||
|   | fa522695c4 | ||
|   | 8609db77ea | ||
|   | 65d93a86b2 | ||
|   | e6c427ce4d | ||
|   | b71c67b6ba | ||
|   | 6d6b0d3321 | ||
|   | 37324a0a00 | ||
|   | 20a5d99f77 | ||
|   | 3b43cc019a | ||
|   | b8421dce3d | ||
|   | 9f6e97865c | ||
|   | 9657314ae2 | ||
|   | 3f7d2336c7 | ||
|   | e0a73d7fbe | ||
|   | b08c4ca2bd | ||
|   | 734892f1e2 | ||
|   | d2bfaeac63 | ||
|   | 0768b1b907 | ||
|   | f5f0da06d9 | ||
|   | 52f04e39f2 | ||
|   | 3c8f4c03d7 | ||
|   | 7ba1308595 | ||
|   | 91cd54016c | ||
|   | e7a393de54 | ||
|   | 8454f298ac | ||
|   | a3badaf103 | ||
|   | 50e8e5bdbe | ||
|   | 8526e1f5f1 | ||
|   | 0cfdbb95cc | ||
|   | 6cea2061ec | ||
|   | 2832801c2a | ||
|   | 23a37dc466 | ||
|   | 992892866b | ||
|   | dde880290c | ||
|   | 1f27d7f1b8 | ||
|   | 00aaa05901 | ||
|   | a83eaa7a9f | ||
|   | 5156e48c2a | ||
|   | bf198c3918 | ||
|   | 09dc6273e3 | ||
|   | ebaa33ac28 | ||
|   | 3ec4ebc562 | ||
|   | 6a19724d5f | ||
|   | 924ce739f9 | ||
|   | e1973e6780 | ||
|   | f1b08ef40e | ||
|   | 31f0cb7742 | ||
|   | e4b2ccfb23 | ||
|   | a3d7bb0a30 | ||
|   | 77e49f3822 | ||
|   | 8945b25484 | ||
|   | 99ccf0c5d3 | ||
|   | d59b164fa2 | ||
|   | 55b5f5dc34 | ||
|   | 3b135ac963 | ||
|   | e6bae8d916 | ||
|   | d9f54300c3 | ||
|   | 1511219763 | ||
|   | ada0add89b | ||
|   | 75e508e1d6 | ||
|   | 6f046dbf18 | ||
|   | cd820c8bca | ||
|   | 88e755d7fd | ||
|   | 6984171cfd | ||
|   | 60b4db6389 | ||
|   | 7c6ea2a966 | ||
|   | c161aef5f9 | ||
|   | c47786c1b0 | ||
|   | df100ce540 | ||
|   | 5c5948b4e7 | ||
|   | 1c72e46e09 | ||
|   | ca210ba480 | ||
|   | df146c41e2 | ||
|   | 2d305fa99a | ||
|   | e4d7f3e287 | ||
|   | f2044b5838 | ||
|   | d53988f619 | ||
|   | ac88ab48d9 | ||
|   | 84c6ee8cc6 | ||
|   | dbc90576b8 | ||
|   | 84200dcde6 | ||
|   | e54c08da89 | ||
|   | 31413857ea | ||
|   | 25f874c030 | ||
|   | 10d502611f | ||
|   | 7fe4103b94 | ||
|   | 7fbdc8e2c1 | ||
|   | 9c5572d51f | ||
|   | 75eb28f574 | ||
|   | 56b6a1720f | ||
|   | dfceca48a7 | ||
|   | bbb67002c3 | ||
|   | 0294216ea9 | ||
|   | 7a62b2d2ab | ||
|   | f08c050e57 | ||
|   | 67c8d49757 | ||
|   | ffcd90e8a7 | ||
|   | 4ca7c4be1f | ||
|   | 17b7af78f0 | ||
|   | 4c1dc52083 | ||
|   | 572fc9099f | ||
|   | 3020f29041 | ||
|   | a6d03dd510 | ||
|   | 68df36ae50 | ||
|   | 5540305293 | ||
|   | d4cfee79d5 | ||
|   | 6e36f948df | ||
|   | 553fa39fe8 | ||
|   | 820e581ad8 | ||
|   | d14785738e | ||
|   | 9e15635c2d | ||
|   | 3e10f902f5 | ||
|   | aa6714f25c | ||
|   | 7f3a37aed4 | ||
|   | 7b08280355 | ||
|   | e3cc4d5eac | ||
|   | 8c85dfb735 | ||
|   | ac62a413e5 | ||
|   | d1f89778e9 | ||
|   | df67a90e64 | ||
|   | 576ae644de | ||
|   | 7e52e51db1 | ||
|   | f12df8d79a | ||
|   | 65de730bdb | ||
|   | 9658a5043b | ||
|   | 280fbe8019 | ||
|   | 2e339c2bab | ||
|   | 38f0c54c64 | ||
|   | f20426a768 | ||
|   | 885f67a471 | ||
|   | a9cc270b4d | ||
|   | aa281a30e5 | ||
|   | 760bc3366b | ||
|   | 5bea29f610 | ||
|   | 9310ee3967 | ||
|   | da7ddbb4dc | ||
|   | 4a28a2f093 | ||
|   | 3d9498dc95 | ||
|   | 1f45f7bb52 | ||
|   | 2e6c64a8f9 | ||
|   | c7dd52271c | ||
|   | e4300e1eb7 | ||
|   | aba706ea2d | ||
|   | 53d0052c6c | ||
|   | 28a136e9a3 | ||
|   | 529ff9ab6d | ||
|   | 41aca47d43 | ||
|   | 3862a51a6a | ||
|   | bcb612a30a | ||
|   | c05219aa0d | ||
|   | 508ffbbb15 | ||
|   | 59fa93cdd4 | ||
|   | 952abe029b | ||
|   | f923855906 | ||
|   | 9386073e96 | ||
|   | 52ea4d4bb2 | ||
|   | c4ba192187 | ||
|   | fe758ca319 | ||
|   | 08b933cc10 | ||
|   | 6746a00af8 | ||
|   | 2fb52261ad | ||
|   | 6fdea03049 | ||
|   | 38021ba494 | ||
|   | 6c9fa573ae | ||
|   | 40c9dc0a31 | ||
|   | 0142660bd4 | ||
|   | 743e957d88 | ||
|   | 560f36e6c8 | ||
|   | e88dd25bab | ||
|   | 567e74e7d7 | ||
|   | 5ade3db040 | ||
|   | 965f9ad033 | ||
|   | 5d1c6b7499 | ||
|   | 5fefaa5d4d | ||
|   | 1775647f76 | ||
|   | 77dc1a6d74 | ||
|   | 05e08d2310 | ||
|   | 31590284a7 | ||
|   | f2863cc7f8 | ||
|   | 4dd296e155 | ||
|   | 304f419429 | ||
|   | 2666d3c206 | 
| @@ -1,7 +1,8 @@ | ||||
| build | ||||
| llama/build | ||||
| .venv | ||||
| .vscode | ||||
| ollama | ||||
| app | ||||
| web | ||||
| dist | ||||
| scripts | ||||
| llm/llama.cpp/ggml | ||||
| llm/llama.cpp/gguf | ||||
| .env | ||||
|   | ||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -2,5 +2,7 @@ | ||||
| .vscode | ||||
| .env | ||||
| .venv | ||||
| .swp | ||||
| dist | ||||
| ollama | ||||
| ggml-metal.metal | ||||
|   | ||||
							
								
								
									
										10
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | ||||
| [submodule "llm/llama.cpp/ggml"] | ||||
|     path = llm/llama.cpp/ggml | ||||
|     url = https://github.com/ggerganov/llama.cpp.git | ||||
|     ignore = dirty | ||||
|     shallow = true | ||||
| [submodule "llm/llama.cpp/gguf"] | ||||
|     path = llm/llama.cpp/gguf | ||||
|     url = https://github.com/ggerganov/llama.cpp.git | ||||
|     ignore = dirty | ||||
|     shallow = true | ||||
							
								
								
									
										28
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						| @@ -1,15 +1,23 @@ | ||||
| FROM golang:1.20 | ||||
| WORKDIR /go/src/github.com/jmorganca/ollama | ||||
| COPY . . | ||||
| RUN CGO_ENABLED=1 go build -ldflags '-linkmode external -extldflags "-static"' . | ||||
| FROM nvidia/cuda:11.8.0-devel-ubuntu22.04 | ||||
|  | ||||
| FROM alpine | ||||
| ARG TARGETARCH | ||||
| ARG GOFLAGS="'-ldflags=-w -s'" | ||||
|  | ||||
| WORKDIR /go/src/github.com/jmorganca/ollama | ||||
| RUN apt-get update && apt-get install -y git build-essential cmake | ||||
| ADD https://dl.google.com/go/go1.21.1.linux-$TARGETARCH.tar.gz /tmp/go1.21.1.tar.gz | ||||
| RUN mkdir -p /usr/local && tar xz -C /usr/local </tmp/go1.21.1.tar.gz | ||||
|  | ||||
| COPY . . | ||||
| ENV GOARCH=$TARGETARCH | ||||
| ENV GOFLAGS=$GOFLAGS | ||||
| RUN /usr/local/go/bin/go generate ./... \ | ||||
|     && /usr/local/go/bin/go build . | ||||
|  | ||||
| FROM ubuntu:22.04 | ||||
| RUN apt-get update && apt-get install -y ca-certificates | ||||
| COPY --from=0 /go/src/github.com/jmorganca/ollama/ollama /bin/ollama | ||||
| EXPOSE 11434 | ||||
| ARG USER=ollama | ||||
| ARG GROUP=ollama | ||||
| RUN addgroup -g 1000 $GROUP && adduser -u 1000 -DG $GROUP $USER | ||||
| USER $USER:$GROUP | ||||
| ENTRYPOINT ["/bin/ollama"] | ||||
| ENV OLLAMA_HOST 0.0.0.0 | ||||
| ENTRYPOINT ["/bin/ollama"] | ||||
| CMD ["serve"] | ||||
|   | ||||
							
								
								
									
										32
									
								
								Dockerfile.build
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,32 @@ | ||||
|  | ||||
| # centos7 amd64 dependencies | ||||
| FROM --platform=linux/amd64 nvidia/cuda:11.8.0-devel-centos7 AS base-amd64 | ||||
| RUN yum install -y https://repo.ius.io/ius-release-el7.rpm centos-release-scl && \ | ||||
|     yum update -y && \ | ||||
|     yum install -y devtoolset-10-gcc devtoolset-10-gcc-c++ git236 wget | ||||
| RUN wget "https://github.com/Kitware/CMake/releases/download/v3.27.6/cmake-3.27.6-linux-x86_64.sh" -O cmake-installer.sh && chmod +x cmake-installer.sh && ./cmake-installer.sh --skip-license --prefix=/usr/local | ||||
| ENV PATH /opt/rh/devtoolset-10/root/usr/bin:$PATH | ||||
|  | ||||
| # centos8 arm64 dependencies | ||||
| FROM --platform=linux/arm64 nvidia/cuda:11.4.3-devel-centos8 AS base-arm64 | ||||
| RUN sed -i -e 's/mirrorlist/#mirrorlist/g' -e 's|#baseurl=http://mirror.centos.org|baseurl=http://vault.centos.org|g' /etc/yum.repos.d/CentOS-* | ||||
| RUN yum install -y git cmake | ||||
|  | ||||
| FROM base-${TARGETARCH} | ||||
| ARG TARGETARCH | ||||
| ARG GOFLAGS="'-ldflags -w -s'" | ||||
|  | ||||
| # install go | ||||
| ADD https://dl.google.com/go/go1.21.1.linux-$TARGETARCH.tar.gz /tmp/go1.21.1.tar.gz | ||||
| RUN mkdir -p /usr/local && tar xz -C /usr/local </tmp/go1.21.1.tar.gz | ||||
|  | ||||
| # build the final binary | ||||
| WORKDIR /go/src/github.com/jmorganca/ollama | ||||
| COPY . . | ||||
|  | ||||
| ENV GOOS=linux | ||||
| ENV GOARCH=$TARGETARCH | ||||
| ENV GOFLAGS=$GOFLAGS | ||||
|  | ||||
| RUN /usr/local/go/bin/go generate ./... && \ | ||||
|     /usr/local/go/bin/go build . | ||||
							
								
								
									
										212
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,108 +1,224 @@ | ||||
|  | ||||
| <div align="center"> | ||||
|   <picture> | ||||
|     <source media="(prefers-color-scheme: dark)" height="200px" srcset="https://github.com/jmorganca/ollama/assets/3325447/56ea1849-1284-4645-8970-956de6e51c3c"> | ||||
|     <img alt="logo" height="200px" src="https://github.com/jmorganca/ollama/assets/3325447/0d0b44e2-8f4a-4e99-9b52-a5c1c741c8f7"> | ||||
|   </picture> | ||||
| </div> | ||||
|  | ||||
| # Ollama | ||||
|  | ||||
| Run large language models with `llama.cpp`. | ||||
| [](https://discord.gg/ollama) | ||||
|  | ||||
| > Note: certain models that can be run with Ollama are intended for research and/or non-commercial use only. | ||||
| Get up and running with large language models locally. | ||||
|  | ||||
| ### Features | ||||
| ### macOS | ||||
|  | ||||
| - Download and run popular large language models | ||||
| - Switch between multiple models on the fly | ||||
| - Hardware acceleration where available (Metal, CUDA) | ||||
| - Fast inference server written in Go, powered by [llama.cpp](https://github.com/ggerganov/llama.cpp) | ||||
| - REST API to use with your application (python, typescript SDKs coming soon) | ||||
| [Download](https://ollama.ai/download/Ollama-darwin.zip) | ||||
|  | ||||
| ## Install | ||||
| ### Linux & WSL2 | ||||
|  | ||||
| - [Download](https://ollama.ai/download) for macOS | ||||
| - Download for Windows (coming soon) | ||||
| ``` | ||||
| curl https://ollama.ai/install.sh | sh | ||||
| ``` | ||||
|  | ||||
| You can also build the [binary from source](#building). | ||||
| [Manual install instructions](https://github.com/jmorganca/ollama/blob/main/docs/linux.md) | ||||
|  | ||||
| ### Windows | ||||
|  | ||||
| coming soon | ||||
|  | ||||
| ## Quickstart | ||||
|  | ||||
| Run a fast and simple model. | ||||
| To run and chat with [Llama 2](https://ollama.ai/library/llama2): | ||||
|  | ||||
| ``` | ||||
| ollama run orca | ||||
| ollama run llama2 | ||||
| ``` | ||||
|  | ||||
| ## Example models | ||||
| ## Model library | ||||
|  | ||||
| ### 💬 Chat | ||||
| Ollama supports a list of open-source models available on [ollama.ai/library](https://ollama.ai/library 'ollama model library') | ||||
|  | ||||
| Have a conversation. | ||||
| Here are some example open-source models that can be downloaded: | ||||
|  | ||||
| | Model              | Parameters | Size  | Download                       | | ||||
| | ------------------ | ---------- | ----- | ------------------------------ | | ||||
| | Mistral            | 7B         | 4.1GB | `ollama run mistral`           | | ||||
| | Llama 2            | 7B         | 3.8GB | `ollama run llama2`            | | ||||
| | Code Llama         | 7B         | 3.8GB | `ollama run codellama`         | | ||||
| | Llama 2 Uncensored | 7B         | 3.8GB | `ollama run llama2-uncensored` | | ||||
| | Llama 2 13B        | 13B        | 7.3GB | `ollama run llama2:13b`        | | ||||
| | Llama 2 70B        | 70B        | 39GB  | `ollama run llama2:70b`        | | ||||
| | Orca Mini          | 3B         | 1.9GB | `ollama run orca-mini`         | | ||||
| | Vicuna             | 7B         | 3.8GB | `ollama run vicuna`            | | ||||
|  | ||||
| > Note: You should have at least 8 GB of RAM to run the 3B models, 16 GB to run the 7B models, and 32 GB to run the 13B models. | ||||
|  | ||||
| ## Customize your own model | ||||
|  | ||||
| ### Import from GGUF or GGML | ||||
|  | ||||
| Ollama supports importing GGUF and GGML file formats in the Modelfile. This means if you have a model that is not in the Ollama library, you can create it, iterate on it, and upload it to the Ollama library to share with others when you are ready. | ||||
|  | ||||
| 1. Create a file named Modelfile, and add a `FROM` instruction with the local filepath to the model you want to import. | ||||
|  | ||||
|    ``` | ||||
| ollama run vicuna "Why is the sky blue?" | ||||
|    FROM ./vicuna-33b.Q4_0.gguf | ||||
|    ``` | ||||
|  | ||||
| ### 🗺️ Instructions | ||||
|  | ||||
| Get a helping hand. | ||||
| 2. Create the model in Ollama | ||||
|  | ||||
|    ``` | ||||
| ollama run orca "Write an email to my boss." | ||||
|    ollama create name -f path_to_modelfile | ||||
|    ``` | ||||
|  | ||||
| ### 🔎 Ask questions about documents | ||||
|  | ||||
| Send the contents of a document and ask questions about it. | ||||
| 3. Run the model | ||||
|  | ||||
|    ``` | ||||
| ollama run nous-hermes "$(cat input.txt)", please summarize this story | ||||
|    ollama run name | ||||
|    ``` | ||||
|  | ||||
| ### 📖 Storytelling | ||||
| ### Customize a prompt | ||||
|  | ||||
| Venture into the unknown. | ||||
| Models from the Ollama library can be customized with a prompt. The example | ||||
|  | ||||
| ``` | ||||
| ollama run nous-hermes "Once upon a time" | ||||
| ollama pull llama2 | ||||
| ``` | ||||
|  | ||||
| ## Advanced usage | ||||
|  | ||||
| ### Run a local model | ||||
| Create a `Modelfile`: | ||||
|  | ||||
| ``` | ||||
| ollama run ~/Downloads/vicuna-7b-v1.3.ggmlv3.q4_1.bin | ||||
| FROM llama2 | ||||
|  | ||||
| # set the temperature to 1 [higher is more creative, lower is more coherent] | ||||
| PARAMETER temperature 1 | ||||
|  | ||||
| # set the system prompt | ||||
| SYSTEM """ | ||||
| You are Mario from Super Mario Bros. Answer as Mario, the assistant, only. | ||||
| """ | ||||
| ``` | ||||
|  | ||||
| Next, create and run the model: | ||||
|  | ||||
| ``` | ||||
| ollama create mario -f ./Modelfile | ||||
| ollama run mario | ||||
| >>> hi | ||||
| Hello! It's your friend Mario. | ||||
| ``` | ||||
|  | ||||
| For more examples, see the [examples](examples) directory. For more information on working with a Modelfile, see the [Modelfile](docs/modelfile.md) documentation. | ||||
|  | ||||
| ## CLI Reference | ||||
|  | ||||
| ### Create a model | ||||
|  | ||||
| `ollama create` is used to create a model from a Modelfile. | ||||
|  | ||||
| ### Pull a model | ||||
|  | ||||
| ``` | ||||
| ollama pull llama2 | ||||
| ``` | ||||
|  | ||||
| > This command can also be used to update a local model. Only the diff will be pulled. | ||||
|  | ||||
| ### Remove a model | ||||
|  | ||||
| ``` | ||||
| ollama rm llama2 | ||||
| ``` | ||||
|  | ||||
| ### Copy a model | ||||
|  | ||||
| ``` | ||||
| ollama cp llama2 my-llama2 | ||||
| ``` | ||||
|  | ||||
| ### Multiline input | ||||
|  | ||||
| For multiline input, you can wrap text with `"""`: | ||||
|  | ||||
| ``` | ||||
| >>> """Hello, | ||||
| ... world! | ||||
| ... """ | ||||
| I'm a basic program that prints the famous "Hello, world!" message to the console. | ||||
| ``` | ||||
|  | ||||
| ### Pass in prompt as arguments | ||||
|  | ||||
| ``` | ||||
| $ ollama run llama2 "summarize this file:" "$(cat README.md)" | ||||
|  Ollama is a lightweight, extensible framework for building and running language models on the local machine. It provides a simple API for creating, running, and managing models, as well as a library of pre-built models that can be easily used in a variety of applications. | ||||
| ``` | ||||
|  | ||||
| ### List models on your computer | ||||
|  | ||||
| ``` | ||||
| ollama list | ||||
| ``` | ||||
|  | ||||
| ### Start Ollama | ||||
|  | ||||
| `ollama serve` is used when you want to start ollama without running the desktop application. | ||||
|  | ||||
| ## Building | ||||
|  | ||||
| Install `cmake` and `go`: | ||||
|  | ||||
| ``` | ||||
| brew install cmake | ||||
| brew install go | ||||
| ``` | ||||
|  | ||||
| Then generate dependencies and build: | ||||
|  | ||||
| ``` | ||||
| go generate ./... | ||||
| go build . | ||||
| ``` | ||||
|  | ||||
| To run it start the server: | ||||
| Next, start the server: | ||||
|  | ||||
| ``` | ||||
| ./ollama server & | ||||
| ./ollama serve | ||||
| ``` | ||||
|  | ||||
| Finally, run a model! | ||||
| Finally, in a separate shell, run a model: | ||||
|  | ||||
| ``` | ||||
| ./ollama run ~/Downloads/vicuna-7b-v1.3.ggmlv3.q4_1.bin | ||||
| ./ollama run llama2 | ||||
| ``` | ||||
|  | ||||
| ## API Reference | ||||
| ## REST API | ||||
|  | ||||
| ### `POST /api/pull` | ||||
| > See the [API documentation](docs/api.md) for all endpoints. | ||||
|  | ||||
| Download a model | ||||
| Ollama has an API for running and managing models. For example to generate text from a model: | ||||
|  | ||||
| ``` | ||||
| curl -X POST http://localhost:11343/api/pull -d '{"model": "orca"}' | ||||
| curl -X POST http://localhost:11434/api/generate -d '{ | ||||
|   "model": "llama2", | ||||
|   "prompt":"Why is the sky blue?" | ||||
| }' | ||||
| ``` | ||||
|  | ||||
| ### `POST /api/generate` | ||||
| ## Community Integrations | ||||
|  | ||||
| Complete a prompt | ||||
|  | ||||
| ``` | ||||
| curl -X POST http://localhost:11434/api/generate -d '{"model": "orca", "prompt": "hello!", "stream": true}' | ||||
| ``` | ||||
| - [LangChain](https://python.langchain.com/docs/integrations/llms/ollama) and [LangChain.js](https://js.langchain.com/docs/modules/model_io/models/llms/integrations/ollama) with [example](https://js.langchain.com/docs/use_cases/question_answering/local_retrieval_qa) | ||||
| - [LlamaIndex](https://gpt-index.readthedocs.io/en/stable/examples/llm/ollama.html) | ||||
| - [Raycast extension](https://github.com/MassimilianoPasquini97/raycast_ollama) | ||||
| - [Discollama](https://github.com/mxyng/discollama) (Discord bot inside the Ollama discord channel) | ||||
| - [Continue](https://github.com/continuedev/continue) | ||||
| - [Obsidian Ollama plugin](https://github.com/hinterdupfinger/obsidian-ollama) | ||||
| - [Dagger Chatbot](https://github.com/samalba/dagger-chatbot) | ||||
| - [LiteLLM](https://github.com/BerriAI/litellm) | ||||
| - [Discord AI Bot](https://github.com/mekb-turtle/discord-ai-bot) | ||||
| - [Chatbot UI](https://github.com/ivanfioravanti/chatbot-ollama) | ||||
| - [HTML UI](https://github.com/rtcfirefly/ollama-ui) | ||||
| - [Typescript UI](https://github.com/ollama-interface/Ollama-Gui?tab=readme-ov-file) | ||||
| - [Dumbar](https://github.com/JerrySievert/Dumbar) | ||||
| - [Emacs client](https://github.com/zweifisch/ollama) | ||||
|   | ||||
							
								
								
									
										214
									
								
								api/client.go
									
									
									
									
									
								
							
							
						
						| @@ -6,39 +6,129 @@ import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"os" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/jmorganca/ollama/version" | ||||
| ) | ||||
|  | ||||
| type StatusError struct { | ||||
| 	StatusCode int | ||||
| 	Status     string | ||||
| 	Message    string | ||||
| } | ||||
| const DefaultHost = "127.0.0.1:11434" | ||||
|  | ||||
| func (e StatusError) Error() string { | ||||
| 	if e.Message != "" { | ||||
| 		return fmt.Sprintf("%s: %s", e.Status, e.Message) | ||||
| 	} | ||||
|  | ||||
| 	return e.Status | ||||
| } | ||||
| var envHost = os.Getenv("OLLAMA_HOST") | ||||
|  | ||||
| type Client struct { | ||||
| 	base url.URL | ||||
| 	base *url.URL | ||||
| 	http http.Client | ||||
| } | ||||
|  | ||||
| func NewClient(hosts ...string) *Client { | ||||
| 	host := "127.0.0.1:11434" | ||||
| 	if len(hosts) > 0 { | ||||
| 		host = hosts[0] | ||||
| func checkError(resp *http.Response, body []byte) error { | ||||
| 	if resp.StatusCode < http.StatusBadRequest { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	return &Client{ | ||||
| 		base: url.URL{Scheme: "http", Host: host}, | ||||
| 	apiError := StatusError{StatusCode: resp.StatusCode} | ||||
|  | ||||
| 	err := json.Unmarshal(body, &apiError) | ||||
| 	if err != nil { | ||||
| 		// Use the full body as the message if we fail to decode a response. | ||||
| 		apiError.ErrorMessage = string(body) | ||||
| 	} | ||||
|  | ||||
| 	return apiError | ||||
| } | ||||
|  | ||||
| func ClientFromEnvironment() (*Client, error) { | ||||
| 	scheme, hostport, ok := strings.Cut(os.Getenv("OLLAMA_HOST"), "://") | ||||
| 	if !ok { | ||||
| 		scheme, hostport = "http", os.Getenv("OLLAMA_HOST") | ||||
| 	} | ||||
|  | ||||
| 	host, port, err := net.SplitHostPort(hostport) | ||||
| 	if err != nil { | ||||
| 		host, port = "127.0.0.1", "11434" | ||||
| 		if ip := net.ParseIP(strings.Trim(os.Getenv("OLLAMA_HOST"), "[]")); ip != nil { | ||||
| 			host = ip.String() | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	client := Client{ | ||||
| 		base: &url.URL{ | ||||
| 			Scheme: scheme, | ||||
| 			Host:   net.JoinHostPort(host, port), | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	mockRequest, err := http.NewRequest("HEAD", client.base.String(), nil) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	proxyURL, err := http.ProxyFromEnvironment(mockRequest) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	client.http = http.Client{ | ||||
| 		Transport: &http.Transport{ | ||||
| 			Proxy: http.ProxyURL(proxyURL), | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	return &client, nil | ||||
| } | ||||
|  | ||||
| func (c *Client) do(ctx context.Context, method, path string, reqData, respData any) error { | ||||
| 	var reqBody io.Reader | ||||
| 	var data []byte | ||||
| 	var err error | ||||
| 	if reqData != nil { | ||||
| 		data, err = json.Marshal(reqData) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		reqBody = bytes.NewReader(data) | ||||
| 	} | ||||
|  | ||||
| 	requestURL := c.base.JoinPath(path) | ||||
| 	request, err := http.NewRequestWithContext(ctx, method, requestURL.String(), reqBody) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	request.Header.Set("Content-Type", "application/json") | ||||
| 	request.Header.Set("Accept", "application/json") | ||||
| 	request.Header.Set("User-Agent", fmt.Sprintf("ollama/%s (%s %s) Go/%s", version.Version, runtime.GOARCH, runtime.GOOS, runtime.Version())) | ||||
|  | ||||
| 	respObj, err := c.http.Do(request) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	defer respObj.Body.Close() | ||||
|  | ||||
| 	respBody, err := io.ReadAll(respObj.Body) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if err := checkError(respObj, respBody); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if len(respBody) > 0 && respData != nil { | ||||
| 		if err := json.Unmarshal(respBody, respData); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| const maxBufferSize = 512 * 1000 // 512KB | ||||
|  | ||||
| func (c *Client) stream(ctx context.Context, method, path string, data any, fn func([]byte) error) error { | ||||
| 	var buf *bytes.Buffer | ||||
| 	if data != nil { | ||||
| @@ -50,21 +140,26 @@ func (c *Client) stream(ctx context.Context, method, path string, data any, fn f | ||||
| 		buf = bytes.NewBuffer(bts) | ||||
| 	} | ||||
|  | ||||
| 	request, err := http.NewRequestWithContext(ctx, method, c.base.JoinPath(path).String(), buf) | ||||
| 	requestURL := c.base.JoinPath(path) | ||||
| 	request, err := http.NewRequestWithContext(ctx, method, requestURL.String(), buf) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	request.Header.Set("Content-Type", "application/json") | ||||
| 	request.Header.Set("Accept", "application/json") | ||||
| 	request.Header.Set("Accept", "application/x-ndjson") | ||||
| 	request.Header.Set("User-Agent", fmt.Sprintf("ollama/%s (%s %s) Go/%s", version.Version, runtime.GOARCH, runtime.GOOS, runtime.Version())) | ||||
|  | ||||
| 	response, err := http.DefaultClient.Do(request) | ||||
| 	response, err := c.http.Do(request) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	defer response.Body.Close() | ||||
|  | ||||
| 	scanner := bufio.NewScanner(response.Body) | ||||
| 	// increase the buffer size to avoid running out of space | ||||
| 	scanBuf := make([]byte, 0, maxBufferSize) | ||||
| 	scanner.Buffer(scanBuf, maxBufferSize) | ||||
| 	for scanner.Scan() { | ||||
| 		var errorResponse struct { | ||||
| 			Error string `json:"error,omitempty"` | ||||
| @@ -75,11 +170,15 @@ func (c *Client) stream(ctx context.Context, method, path string, data any, fn f | ||||
| 			return fmt.Errorf("unmarshal: %w", err) | ||||
| 		} | ||||
|  | ||||
| 		if response.StatusCode >= 400 { | ||||
| 		if errorResponse.Error != "" { | ||||
| 			return fmt.Errorf(errorResponse.Error) | ||||
| 		} | ||||
|  | ||||
| 		if response.StatusCode >= http.StatusBadRequest { | ||||
| 			return StatusError{ | ||||
| 				StatusCode:   response.StatusCode, | ||||
| 				Status:       response.Status, | ||||
| 				Message:    errorResponse.Error, | ||||
| 				ErrorMessage: errorResponse.Error, | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| @@ -104,11 +203,11 @@ func (c *Client) Generate(ctx context.Context, req *GenerateRequest, fn Generate | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| type PullProgressFunc func(PullProgress) error | ||||
| type PullProgressFunc func(ProgressResponse) error | ||||
|  | ||||
| func (c *Client) Pull(ctx context.Context, req *PullRequest, fn PullProgressFunc) error { | ||||
| 	return c.stream(ctx, http.MethodPost, "/api/pull", req, func(bts []byte) error { | ||||
| 		var resp PullProgress | ||||
| 		var resp ProgressResponse | ||||
| 		if err := json.Unmarshal(bts, &resp); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| @@ -116,3 +215,66 @@ func (c *Client) Pull(ctx context.Context, req *PullRequest, fn PullProgressFunc | ||||
| 		return fn(resp) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| type PushProgressFunc func(ProgressResponse) error | ||||
|  | ||||
| func (c *Client) Push(ctx context.Context, req *PushRequest, fn PushProgressFunc) error { | ||||
| 	return c.stream(ctx, http.MethodPost, "/api/push", req, func(bts []byte) error { | ||||
| 		var resp ProgressResponse | ||||
| 		if err := json.Unmarshal(bts, &resp); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		return fn(resp) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| type CreateProgressFunc func(ProgressResponse) error | ||||
|  | ||||
| func (c *Client) Create(ctx context.Context, req *CreateRequest, fn CreateProgressFunc) error { | ||||
| 	return c.stream(ctx, http.MethodPost, "/api/create", req, func(bts []byte) error { | ||||
| 		var resp ProgressResponse | ||||
| 		if err := json.Unmarshal(bts, &resp); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		return fn(resp) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func (c *Client) List(ctx context.Context) (*ListResponse, error) { | ||||
| 	var lr ListResponse | ||||
| 	if err := c.do(ctx, http.MethodGet, "/api/tags", nil, &lr); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &lr, nil | ||||
| } | ||||
|  | ||||
| func (c *Client) Copy(ctx context.Context, req *CopyRequest) error { | ||||
| 	if err := c.do(ctx, http.MethodPost, "/api/copy", req, nil); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (c *Client) Delete(ctx context.Context, req *DeleteRequest) error { | ||||
| 	if err := c.do(ctx, http.MethodDelete, "/api/delete", req, nil); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (c *Client) Show(ctx context.Context, req *ShowRequest) (*ShowResponse, error) { | ||||
| 	var resp ShowResponse | ||||
| 	if err := c.do(ctx, http.MethodPost, "/api/show", req, &resp); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &resp, nil | ||||
| } | ||||
|  | ||||
| func (c *Client) Heartbeat(ctx context.Context) error { | ||||
| 	if err := c.do(ctx, http.MethodHead, "/", nil, nil); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
							
								
								
									
										225
									
								
								api/client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,225 @@ | ||||
| import os | ||||
| import json | ||||
| import requests | ||||
|  | ||||
| BASE_URL = os.environ.get('OLLAMA_HOST', 'http://localhost:11434') | ||||
|  | ||||
| # Generate a response for a given prompt with a provided model. This is a streaming endpoint, so will be a series of responses. | ||||
| # The final response object will include statistics and additional data from the request. Use the callback function to override | ||||
| # the default handler. | ||||
| def generate(model_name, prompt, system=None, template=None, context=None, options=None, callback=None): | ||||
|     try: | ||||
|         url = f"{BASE_URL}/api/generate" | ||||
|         payload = { | ||||
|             "model": model_name,  | ||||
|             "prompt": prompt,  | ||||
|             "system": system,  | ||||
|             "template": template,  | ||||
|             "context": context,  | ||||
|             "options": options | ||||
|         } | ||||
|          | ||||
|         # Remove keys with None values | ||||
|         payload = {k: v for k, v in payload.items() if v is not None} | ||||
|          | ||||
|         with requests.post(url, json=payload, stream=True) as response: | ||||
|             response.raise_for_status() | ||||
|              | ||||
|             # Creating a variable to hold the context history of the final chunk | ||||
|             final_context = None | ||||
|              | ||||
|             # Variable to hold concatenated response strings if no callback is provided | ||||
|             full_response = "" | ||||
|  | ||||
|             # Iterating over the response line by line and displaying the details | ||||
|             for line in response.iter_lines(): | ||||
|                 if line: | ||||
|                     # Parsing each line (JSON chunk) and extracting the details | ||||
|                     chunk = json.loads(line) | ||||
|                      | ||||
|                     # If a callback function is provided, call it with the chunk | ||||
|                     if callback: | ||||
|                         callback(chunk) | ||||
|                     else: | ||||
|                         # If this is not the last chunk, add the "response" field value to full_response and print it | ||||
|                         if not chunk.get("done"): | ||||
|                             response_piece = chunk.get("response", "") | ||||
|                             full_response += response_piece | ||||
|                             print(response_piece, end="", flush=True) | ||||
|                      | ||||
|                     # Check if it's the last chunk (done is true) | ||||
|                     if chunk.get("done"): | ||||
|                         final_context = chunk.get("context") | ||||
|              | ||||
|             # Return the full response and the final context | ||||
|             return full_response, final_context | ||||
|     except requests.exceptions.RequestException as e: | ||||
|         print(f"An error occurred: {e}") | ||||
|         return None, None | ||||
|  | ||||
| # Create a model from a Modelfile. Use the callback function to override the default handler. | ||||
| def create(model_name, model_path, callback=None): | ||||
|     try: | ||||
|         url = f"{BASE_URL}/api/create" | ||||
|         payload = {"name": model_name, "path": model_path} | ||||
|          | ||||
|         # Making a POST request with the stream parameter set to True to handle streaming responses | ||||
|         with requests.post(url, json=payload, stream=True) as response: | ||||
|             response.raise_for_status() | ||||
|  | ||||
|             # Iterating over the response line by line and displaying the status | ||||
|             for line in response.iter_lines(): | ||||
|                 if line: | ||||
|                     # Parsing each line (JSON chunk) and extracting the status | ||||
|                     chunk = json.loads(line) | ||||
|  | ||||
|                     if callback: | ||||
|                         callback(chunk) | ||||
|                     else: | ||||
|                         print(f"Status: {chunk.get('status')}") | ||||
|     except requests.exceptions.RequestException as e: | ||||
|         print(f"An error occurred: {e}") | ||||
|  | ||||
| # Pull a model from a the model registry. Cancelled pulls are resumed from where they left off, and multiple | ||||
| # calls to will share the same download progress. Use the callback function to override the default handler. | ||||
| def pull(model_name, insecure=False, callback=None): | ||||
|     try: | ||||
|         url = f"{BASE_URL}/api/pull" | ||||
|         payload = { | ||||
|             "name": model_name, | ||||
|             "insecure": insecure | ||||
|         } | ||||
|  | ||||
|         # Making a POST request with the stream parameter set to True to handle streaming responses | ||||
|         with requests.post(url, json=payload, stream=True) as response: | ||||
|             response.raise_for_status() | ||||
|  | ||||
|             # Iterating over the response line by line and displaying the details | ||||
|             for line in response.iter_lines(): | ||||
|                 if line: | ||||
|                     # Parsing each line (JSON chunk) and extracting the details | ||||
|                     chunk = json.loads(line) | ||||
|  | ||||
|                     # If a callback function is provided, call it with the chunk | ||||
|                     if callback: | ||||
|                         callback(chunk) | ||||
|                     else: | ||||
|                         # Print the status message directly to the console | ||||
|                         print(chunk.get('status', ''), end='', flush=True) | ||||
|                      | ||||
|                     # If there's layer data, you might also want to print that (adjust as necessary) | ||||
|                     if 'digest' in chunk: | ||||
|                         print(f" - Digest: {chunk['digest']}", end='', flush=True) | ||||
|                         print(f" - Total: {chunk['total']}", end='', flush=True) | ||||
|                         print(f" - Completed: {chunk['completed']}", end='\n', flush=True) | ||||
|                     else: | ||||
|                         print() | ||||
|     except requests.exceptions.RequestException as e: | ||||
|         print(f"An error occurred: {e}") | ||||
|  | ||||
| # Push a model to the model registry. Use the callback function to override the default handler. | ||||
| def push(model_name, insecure=False, callback=None): | ||||
|     try: | ||||
|         url = f"{BASE_URL}/api/push" | ||||
|         payload = { | ||||
|             "name": model_name, | ||||
|             "insecure": insecure | ||||
|         } | ||||
|  | ||||
|         # Making a POST request with the stream parameter set to True to handle streaming responses | ||||
|         with requests.post(url, json=payload, stream=True) as response: | ||||
|             response.raise_for_status() | ||||
|  | ||||
|             # Iterating over the response line by line and displaying the details | ||||
|             for line in response.iter_lines(): | ||||
|                 if line: | ||||
|                     # Parsing each line (JSON chunk) and extracting the details | ||||
|                     chunk = json.loads(line) | ||||
|  | ||||
|                     # If a callback function is provided, call it with the chunk | ||||
|                     if callback: | ||||
|                         callback(chunk) | ||||
|                     else: | ||||
|                         # Print the status message directly to the console | ||||
|                         print(chunk.get('status', ''), end='', flush=True) | ||||
|                      | ||||
|                     # If there's layer data, you might also want to print that (adjust as necessary) | ||||
|                     if 'digest' in chunk: | ||||
|                         print(f" - Digest: {chunk['digest']}", end='', flush=True) | ||||
|                         print(f" - Total: {chunk['total']}", end='', flush=True) | ||||
|                         print(f" - Completed: {chunk['completed']}", end='\n', flush=True) | ||||
|                     else: | ||||
|                         print() | ||||
|     except requests.exceptions.RequestException as e: | ||||
|         print(f"An error occurred: {e}") | ||||
|  | ||||
| # List models that are available locally. | ||||
| def list(): | ||||
|     try: | ||||
|         response = requests.get(f"{BASE_URL}/api/tags") | ||||
|         response.raise_for_status() | ||||
|         data = response.json() | ||||
|         models = data.get('models', []) | ||||
|         return models | ||||
|  | ||||
|     except requests.exceptions.RequestException as e: | ||||
|         print(f"An error occurred: {e}") | ||||
|         return None | ||||
|  | ||||
| # Copy a model. Creates a model with another name from an existing model. | ||||
| def copy(source, destination): | ||||
|     try: | ||||
|         # Create the JSON payload | ||||
|         payload = { | ||||
|             "source": source, | ||||
|             "destination": destination | ||||
|         } | ||||
|          | ||||
|         response = requests.post(f"{BASE_URL}/api/copy", json=payload) | ||||
|         response.raise_for_status() | ||||
|          | ||||
|         # If the request was successful, return a message indicating that the copy was successful | ||||
|         return "Copy successful" | ||||
|  | ||||
|     except requests.exceptions.RequestException as e: | ||||
|         print(f"An error occurred: {e}") | ||||
|         return None | ||||
|  | ||||
| # Delete a model and its data. | ||||
| def delete(model_name): | ||||
|     try: | ||||
|         url = f"{BASE_URL}/api/delete" | ||||
|         payload = {"name": model_name} | ||||
|         response = requests.delete(url, json=payload) | ||||
|         response.raise_for_status() | ||||
|         return "Delete successful" | ||||
|     except requests.exceptions.RequestException as e: | ||||
|         print(f"An error occurred: {e}") | ||||
|         return None | ||||
|  | ||||
| # Show info about a model. | ||||
| def show(model_name): | ||||
|     try: | ||||
|         url = f"{BASE_URL}/api/show" | ||||
|         payload = {"name": model_name} | ||||
|         response = requests.post(url, json=payload) | ||||
|         response.raise_for_status() | ||||
|          | ||||
|         # Parse the JSON response and return it | ||||
|         data = response.json() | ||||
|         return data | ||||
|     except requests.exceptions.RequestException as e: | ||||
|         print(f"An error occurred: {e}") | ||||
|         return None | ||||
|  | ||||
| def heartbeat(): | ||||
|     try: | ||||
|         url = f"{BASE_URL}/" | ||||
|         response = requests.head(url) | ||||
|         response.raise_for_status() | ||||
|         return "Ollama is running" | ||||
|     except requests.exceptions.RequestException as e: | ||||
|         print(f"An error occurred: {e}") | ||||
|         return "Ollama is not running" | ||||
|  | ||||
|  | ||||
							
								
								
									
										331
									
								
								api/types.go
									
									
									
									
									
								
							
							
						
						| @@ -1,26 +1,165 @@ | ||||
| package api | ||||
|  | ||||
| import "runtime" | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"log" | ||||
| 	"math" | ||||
| 	"os" | ||||
| 	"reflect" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type PullRequest struct { | ||||
| 	Model string `json:"model"` | ||||
| type StatusError struct { | ||||
| 	StatusCode   int | ||||
| 	Status       string | ||||
| 	ErrorMessage string `json:"error"` | ||||
| } | ||||
|  | ||||
| type PullProgress struct { | ||||
| 	Total     int64   `json:"total"` | ||||
| 	Completed int64   `json:"completed"` | ||||
| 	Percent   float64 `json:"percent"` | ||||
| func (e StatusError) Error() string { | ||||
| 	switch { | ||||
| 	case e.Status != "" && e.ErrorMessage != "": | ||||
| 		return fmt.Sprintf("%s: %s", e.Status, e.ErrorMessage) | ||||
| 	case e.Status != "": | ||||
| 		return e.Status | ||||
| 	case e.ErrorMessage != "": | ||||
| 		return e.ErrorMessage | ||||
| 	default: | ||||
| 		// this should not happen | ||||
| 		return "something went wrong, please see the ollama server logs for details" | ||||
| 	} | ||||
| } | ||||
|  | ||||
| type GenerateRequest struct { | ||||
| 	Model    string `json:"model"` | ||||
| 	Prompt   string `json:"prompt"` | ||||
| 	System   string `json:"system"` | ||||
| 	Template string `json:"template"` | ||||
| 	Context  []int  `json:"context,omitempty"` | ||||
| 	Stream   *bool  `json:"stream,omitempty"` | ||||
|  | ||||
| 	Options `json:"options"` | ||||
| 	Options map[string]interface{} `json:"options"` | ||||
| } | ||||
|  | ||||
| type EmbeddingRequest struct { | ||||
| 	Model  string `json:"model"` | ||||
| 	Prompt string `json:"prompt"` | ||||
|  | ||||
| 	Options map[string]interface{} `json:"options"` | ||||
| } | ||||
|  | ||||
| type EmbeddingResponse struct { | ||||
| 	Embedding []float64 `json:"embedding"` | ||||
| } | ||||
|  | ||||
| type CreateRequest struct { | ||||
| 	Name   string `json:"name"` | ||||
| 	Path   string `json:"path"` | ||||
| 	Stream *bool  `json:"stream,omitempty"` | ||||
| } | ||||
|  | ||||
| type DeleteRequest struct { | ||||
| 	Name string `json:"name"` | ||||
| } | ||||
|  | ||||
| type ShowRequest struct { | ||||
| 	Name string `json:"name"` | ||||
| } | ||||
|  | ||||
| type ShowResponse struct { | ||||
| 	License    string `json:"license,omitempty"` | ||||
| 	Modelfile  string `json:"modelfile,omitempty"` | ||||
| 	Parameters string `json:"parameters,omitempty"` | ||||
| 	Template   string `json:"template,omitempty"` | ||||
| 	System     string `json:"system,omitempty"` | ||||
| } | ||||
|  | ||||
| type CopyRequest struct { | ||||
| 	Source      string `json:"source"` | ||||
| 	Destination string `json:"destination"` | ||||
| } | ||||
|  | ||||
| type PullRequest struct { | ||||
| 	Name     string `json:"name"` | ||||
| 	Insecure bool   `json:"insecure,omitempty"` | ||||
| 	Username string `json:"username"` | ||||
| 	Password string `json:"password"` | ||||
| 	Stream   *bool  `json:"stream,omitempty"` | ||||
| } | ||||
|  | ||||
| type ProgressResponse struct { | ||||
| 	Status    string `json:"status"` | ||||
| 	Digest    string `json:"digest,omitempty"` | ||||
| 	Total     int64  `json:"total,omitempty"` | ||||
| 	Completed int64  `json:"completed,omitempty"` | ||||
| } | ||||
|  | ||||
| type PushRequest struct { | ||||
| 	Name     string `json:"name"` | ||||
| 	Insecure bool   `json:"insecure,omitempty"` | ||||
| 	Username string `json:"username"` | ||||
| 	Password string `json:"password"` | ||||
| 	Stream   *bool  `json:"stream,omitempty"` | ||||
| } | ||||
|  | ||||
| type ListResponse struct { | ||||
| 	Models []ModelResponse `json:"models"` | ||||
| } | ||||
|  | ||||
| type ModelResponse struct { | ||||
| 	Name       string    `json:"name"` | ||||
| 	ModifiedAt time.Time `json:"modified_at"` | ||||
| 	Size       int64     `json:"size"` | ||||
| 	Digest     string    `json:"digest"` | ||||
| } | ||||
|  | ||||
| type TokenResponse struct { | ||||
| 	Token string `json:"token"` | ||||
| } | ||||
|  | ||||
| type GenerateResponse struct { | ||||
| 	Model     string    `json:"model"` | ||||
| 	CreatedAt time.Time `json:"created_at"` | ||||
| 	Response  string    `json:"response"` | ||||
|  | ||||
| 	Done    bool  `json:"done"` | ||||
| 	Context []int `json:"context,omitempty"` | ||||
|  | ||||
| 	TotalDuration      time.Duration `json:"total_duration,omitempty"` | ||||
| 	LoadDuration       time.Duration `json:"load_duration,omitempty"` | ||||
| 	PromptEvalCount    int           `json:"prompt_eval_count,omitempty"` | ||||
| 	PromptEvalDuration time.Duration `json:"prompt_eval_duration,omitempty"` | ||||
| 	EvalCount          int           `json:"eval_count,omitempty"` | ||||
| 	EvalDuration       time.Duration `json:"eval_duration,omitempty"` | ||||
| } | ||||
|  | ||||
| func (r *GenerateResponse) Summary() { | ||||
| 	if r.TotalDuration > 0 { | ||||
| 		fmt.Fprintf(os.Stderr, "total duration:       %v\n", r.TotalDuration) | ||||
| 	} | ||||
|  | ||||
| 	if r.LoadDuration > 0 { | ||||
| 		fmt.Fprintf(os.Stderr, "load duration:        %v\n", r.LoadDuration) | ||||
| 	} | ||||
|  | ||||
| 	if r.PromptEvalCount > 0 { | ||||
| 		fmt.Fprintf(os.Stderr, "prompt eval count:    %d token(s)\n", r.PromptEvalCount) | ||||
| 	} | ||||
|  | ||||
| 	if r.PromptEvalDuration > 0 { | ||||
| 		fmt.Fprintf(os.Stderr, "prompt eval duration: %s\n", r.PromptEvalDuration) | ||||
| 		fmt.Fprintf(os.Stderr, "prompt eval rate:     %.2f tokens/s\n", float64(r.PromptEvalCount)/r.PromptEvalDuration.Seconds()) | ||||
| 	} | ||||
|  | ||||
| 	if r.EvalCount > 0 { | ||||
| 		fmt.Fprintf(os.Stderr, "eval count:           %d token(s)\n", r.EvalCount) | ||||
| 	} | ||||
|  | ||||
| 	if r.EvalDuration > 0 { | ||||
| 		fmt.Fprintf(os.Stderr, "eval duration:        %s\n", r.EvalDuration) | ||||
| 		fmt.Fprintf(os.Stderr, "eval rate:            %.2f tokens/s\n", float64(r.EvalCount)/r.EvalDuration.Seconds()) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| type Options struct { | ||||
| @@ -31,7 +170,9 @@ type Options struct { | ||||
|  | ||||
| 	// Model options | ||||
| 	NumCtx             int     `json:"num_ctx,omitempty"` | ||||
| 	NumKeep            int     `json:"num_keep,omitempty"` | ||||
| 	NumBatch           int     `json:"num_batch,omitempty"` | ||||
| 	NumGQA             int     `json:"num_gqa,omitempty"` | ||||
| 	NumGPU             int     `json:"num_gpu,omitempty"` | ||||
| 	MainGPU            int     `json:"main_gpu,omitempty"` | ||||
| 	LowVRAM            bool    `json:"low_vram,omitempty"` | ||||
| @@ -41,51 +182,181 @@ type Options struct { | ||||
| 	UseMMap            bool    `json:"use_mmap,omitempty"` | ||||
| 	UseMLock           bool    `json:"use_mlock,omitempty"` | ||||
| 	EmbeddingOnly      bool    `json:"embedding_only,omitempty"` | ||||
| 	RopeFrequencyBase  float32 `json:"rope_frequency_base,omitempty"` | ||||
| 	RopeFrequencyScale float32 `json:"rope_frequency_scale,omitempty"` | ||||
|  | ||||
| 	// Predict options | ||||
| 	RepeatLastN      int     `json:"repeat_last_n,omitempty"` | ||||
| 	RepeatPenalty    float32 `json:"repeat_penalty,omitempty"` | ||||
| 	FrequencyPenalty float32 `json:"frequency_penalty,omitempty"` | ||||
| 	PresencePenalty  float32 `json:"presence_penalty,omitempty"` | ||||
| 	Temperature      float32 `json:"temperature,omitempty"` | ||||
| 	NumPredict       int      `json:"num_predict,omitempty"` | ||||
| 	TopK             int      `json:"top_k,omitempty"` | ||||
| 	TopP             float32  `json:"top_p,omitempty"` | ||||
| 	TFSZ             float32  `json:"tfs_z,omitempty"` | ||||
| 	TypicalP         float32  `json:"typical_p,omitempty"` | ||||
| 	RepeatLastN      int      `json:"repeat_last_n,omitempty"` | ||||
| 	Temperature      float32  `json:"temperature,omitempty"` | ||||
| 	RepeatPenalty    float32  `json:"repeat_penalty,omitempty"` | ||||
| 	PresencePenalty  float32  `json:"presence_penalty,omitempty"` | ||||
| 	FrequencyPenalty float32  `json:"frequency_penalty,omitempty"` | ||||
| 	Mirostat         int      `json:"mirostat,omitempty"` | ||||
| 	MirostatTau      float32  `json:"mirostat_tau,omitempty"` | ||||
| 	MirostatEta      float32  `json:"mirostat_eta,omitempty"` | ||||
| 	PenalizeNewline  bool     `json:"penalize_newline,omitempty"` | ||||
| 	Stop             []string `json:"stop,omitempty"` | ||||
|  | ||||
| 	NumThread int `json:"num_thread,omitempty"` | ||||
| } | ||||
|  | ||||
| var ErrInvalidOpts = fmt.Errorf("invalid options") | ||||
|  | ||||
| func (opts *Options) FromMap(m map[string]interface{}) error { | ||||
| 	valueOpts := reflect.ValueOf(opts).Elem() // names of the fields in the options struct | ||||
| 	typeOpts := reflect.TypeOf(opts).Elem()   // types of the fields in the options struct | ||||
|  | ||||
| 	// build map of json struct tags to their types | ||||
| 	jsonOpts := make(map[string]reflect.StructField) | ||||
| 	for _, field := range reflect.VisibleFields(typeOpts) { | ||||
| 		jsonTag := strings.Split(field.Tag.Get("json"), ",")[0] | ||||
| 		if jsonTag != "" { | ||||
| 			jsonOpts[jsonTag] = field | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	invalidOpts := []string{} | ||||
| 	for key, val := range m { | ||||
| 		if opt, ok := jsonOpts[key]; ok { | ||||
| 			field := valueOpts.FieldByName(opt.Name) | ||||
| 			if field.IsValid() && field.CanSet() { | ||||
| 				if val == nil { | ||||
| 					continue | ||||
| 				} | ||||
|  | ||||
| 				switch field.Kind() { | ||||
| 				case reflect.Int: | ||||
| 					switch t := val.(type) { | ||||
| 					case int64: | ||||
| 						field.SetInt(t) | ||||
| 					case float64: | ||||
| 						// when JSON unmarshals numbers, it uses float64, not int | ||||
| 						field.SetInt(int64(t)) | ||||
| 					default: | ||||
| 						log.Printf("could not convert model parameter %v of type %T to int, skipped", key, val) | ||||
| 					} | ||||
| 				case reflect.Bool: | ||||
| 					val, ok := val.(bool) | ||||
| 					if !ok { | ||||
| 						log.Printf("could not convert model parameter %v of type %T to bool, skipped", key, val) | ||||
| 						continue | ||||
| 					} | ||||
| 					field.SetBool(val) | ||||
| 				case reflect.Float32: | ||||
| 					// JSON unmarshals to float64 | ||||
| 					val, ok := val.(float64) | ||||
| 					if !ok { | ||||
| 						log.Printf("could not convert model parameter %v of type %T to float32, skipped", key, val) | ||||
| 						continue | ||||
| 					} | ||||
| 					field.SetFloat(val) | ||||
| 				case reflect.String: | ||||
| 					val, ok := val.(string) | ||||
| 					if !ok { | ||||
| 						log.Printf("could not convert model parameter %v of type %T to string, skipped", key, val) | ||||
| 						continue | ||||
| 					} | ||||
| 					field.SetString(val) | ||||
| 				case reflect.Slice: | ||||
| 					// JSON unmarshals to []interface{}, not []string | ||||
| 					val, ok := val.([]interface{}) | ||||
| 					if !ok { | ||||
| 						log.Printf("could not convert model parameter %v of type %T to slice, skipped", key, val) | ||||
| 						continue | ||||
| 					} | ||||
| 					// convert []interface{} to []string | ||||
| 					slice := make([]string, len(val)) | ||||
| 					for i, item := range val { | ||||
| 						str, ok := item.(string) | ||||
| 						if !ok { | ||||
| 							log.Printf("could not convert model parameter %v of type %T to slice of strings, skipped", key, item) | ||||
| 							continue | ||||
| 						} | ||||
| 						slice[i] = str | ||||
| 					} | ||||
| 					field.Set(reflect.ValueOf(slice)) | ||||
| 				default: | ||||
| 					return fmt.Errorf("unknown type loading config params: %v", field.Kind()) | ||||
| 				} | ||||
| 			} | ||||
| 		} else { | ||||
| 			invalidOpts = append(invalidOpts, key) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if len(invalidOpts) > 0 { | ||||
| 		return fmt.Errorf("%w: %v", ErrInvalidOpts, strings.Join(invalidOpts, ", ")) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func DefaultOptions() Options { | ||||
| 	return Options{ | ||||
| 		Seed: -1, | ||||
|  | ||||
| 		UseNUMA: false, | ||||
|  | ||||
| 		NumCtx:   512, | ||||
| 		NumBatch: 512, | ||||
| 		NumGPU:   1, | ||||
| 		LowVRAM:  false, | ||||
| 		F16KV:    true, | ||||
| 		UseMMap:  true, | ||||
| 		UseMLock: false, | ||||
|  | ||||
| 		RepeatLastN:      512, | ||||
| 		RepeatPenalty:    1.1, | ||||
| 		FrequencyPenalty: 0.0, | ||||
| 		PresencePenalty:  0.0, | ||||
| 		// options set on request to runner | ||||
| 		NumPredict:       -1, | ||||
| 		NumKeep:          -1, | ||||
| 		Temperature:      0.8, | ||||
| 		TopK:             40, | ||||
| 		TopP:             0.9, | ||||
| 		TFSZ:             1.0, | ||||
| 		TypicalP:         1.0, | ||||
| 		RepeatLastN:      64, | ||||
| 		RepeatPenalty:    1.1, | ||||
| 		PresencePenalty:  0.0, | ||||
| 		FrequencyPenalty: 0.0, | ||||
| 		Mirostat:         0, | ||||
| 		MirostatTau:      5.0, | ||||
| 		MirostatEta:      0.1, | ||||
| 		PenalizeNewline:  true, | ||||
| 		Seed:             -1, | ||||
|  | ||||
| 		NumThread: runtime.NumCPU(), | ||||
| 		// options set when the model is loaded | ||||
| 		NumCtx:             2048, | ||||
| 		RopeFrequencyBase:  10000.0, | ||||
| 		RopeFrequencyScale: 1.0, | ||||
| 		NumBatch:           512, | ||||
| 		NumGPU:             -1, // -1 here indicates that NumGPU should be set dynamically | ||||
| 		NumGQA:             1, | ||||
| 		NumThread:          0, // let the runtime decide | ||||
| 		LowVRAM:            false, | ||||
| 		F16KV:              true, | ||||
| 		UseMLock:           false, | ||||
| 		UseMMap:            true, | ||||
| 		UseNUMA:            false, | ||||
| 		EmbeddingOnly:      true, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| type Duration struct { | ||||
| 	time.Duration | ||||
| } | ||||
|  | ||||
| func (d *Duration) UnmarshalJSON(b []byte) (err error) { | ||||
| 	var v any | ||||
| 	if err := json.Unmarshal(b, &v); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	d.Duration = 5 * time.Minute | ||||
|  | ||||
| 	switch t := v.(type) { | ||||
| 	case float64: | ||||
| 		if t < 0 { | ||||
| 			t = math.MaxFloat64 | ||||
| 		} | ||||
|  | ||||
| 		d.Duration = time.Duration(t) | ||||
| 	case string: | ||||
| 		d.Duration, err = time.ParseDuration(t) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
| @@ -1,7 +1,5 @@ | ||||
| # Desktop | ||||
|  | ||||
| _Note: the Ollama desktop app is a work in progress and is not ready yet for general use._ | ||||
|  | ||||
| This app builds upon Ollama to provide a desktop experience for running models. | ||||
|  | ||||
| ## Developing | ||||
| @@ -9,19 +7,15 @@ This app builds upon Ollama to provide a desktop experience for running models. | ||||
| First, build the `ollama` binary: | ||||
|  | ||||
| ``` | ||||
| make -C .. | ||||
| cd .. | ||||
| go build . | ||||
| ``` | ||||
|  | ||||
| Then run the desktop app with `npm start`: | ||||
|  | ||||
| ``` | ||||
| cd app | ||||
| npm install | ||||
| npm start | ||||
| ``` | ||||
|  | ||||
| ## Coming soon | ||||
|  | ||||
| - Browse the latest available models on Hugging Face and other sources | ||||
| - Keep track of previous conversations with models | ||||
| - Switch quickly between models | ||||
| - Connect to remote Ollama servers to run models | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								app/assets/iconDarkTemplate.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 402 B | 
							
								
								
									
										
											BIN
										
									
								
								app/assets/iconDarkTemplate@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 741 B | 
							
								
								
									
										
											BIN
										
									
								
								app/assets/iconDarkUpdateTemplate.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 440 B | 
							
								
								
									
										
											BIN
										
									
								
								app/assets/iconDarkUpdateTemplate@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 763 B | 
							
								
								
									
										
											BIN
										
									
								
								app/assets/iconTemplate.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 447 B | 
							
								
								
									
										
											BIN
										
									
								
								app/assets/iconTemplate@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 891 B | 
							
								
								
									
										
											BIN
										
									
								
								app/assets/iconUpdateTemplate.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 443 B | 
							
								
								
									
										
											BIN
										
									
								
								app/assets/iconUpdateTemplate@2x.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 844 B | 
| Before Width: | Height: | Size: 442 B | 
| Before Width: | Height: | Size: 889 B | 
| @@ -1,4 +1,4 @@ | ||||
| import type { ForgeConfig, ResolvedForgeConfig, ForgeMakeResult } from '@electron-forge/shared-types' | ||||
| import type { ForgeConfig } from '@electron-forge/shared-types' | ||||
| import { MakerSquirrel } from '@electron-forge/maker-squirrel' | ||||
| import { MakerZIP } from '@electron-forge/maker-zip' | ||||
| import { PublisherGithub } from '@electron-forge/publisher-github' | ||||
| @@ -18,10 +18,15 @@ const config: ForgeConfig = { | ||||
|     asar: true, | ||||
|     icon: './assets/icon.icns', | ||||
|     extraResource: [ | ||||
|       '../ollama', | ||||
|       path.join(__dirname, './assets/ollama_icon_16x16Template.png'), | ||||
|       path.join(__dirname, './assets/ollama_icon_16x16Template@2x.png'), | ||||
|       ...(process.platform === 'darwin' ? ['../llama/ggml-metal.metal'] : []), | ||||
|       '../dist/ollama', | ||||
|       path.join(__dirname, './assets/iconTemplate.png'), | ||||
|       path.join(__dirname, './assets/iconTemplate@2x.png'), | ||||
|       path.join(__dirname, './assets/iconUpdateTemplate.png'), | ||||
|       path.join(__dirname, './assets/iconUpdateTemplate@2x.png'), | ||||
|       path.join(__dirname, './assets/iconDarkTemplate.png'), | ||||
|       path.join(__dirname, './assets/iconDarkTemplate@2x.png'), | ||||
|       path.join(__dirname, './assets/iconDarkUpdateTemplate.png'), | ||||
|       path.join(__dirname, './assets/iconDarkUpdateTemplate@2x.png'), | ||||
|     ], | ||||
|     ...(process.env.SIGN | ||||
|       ? { | ||||
| @@ -36,6 +41,9 @@ const config: ForgeConfig = { | ||||
|           }, | ||||
|         } | ||||
|       : {}), | ||||
|     osxUniversal: { | ||||
|       x64ArchFiles: '**/ollama', | ||||
|     }, | ||||
|   }, | ||||
|   rebuildConfig: {}, | ||||
|   makers: [new MakerSquirrel({}), new MakerZIP({}, ['darwin'])], | ||||
| @@ -58,7 +66,7 @@ const config: ForgeConfig = { | ||||
|     new AutoUnpackNativesPlugin({}), | ||||
|     new WebpackPlugin({ | ||||
|       mainConfig, | ||||
|       devContentSecurityPolicy: `default-src * 'unsafe-eval' 'unsafe-inline'`, | ||||
|       devContentSecurityPolicy: `default-src * 'unsafe-eval' 'unsafe-inline'; img-src data: 'self'`, | ||||
|       renderer: { | ||||
|         config: rendererConfig, | ||||
|         nodeIntegration: true, | ||||
|   | ||||
							
								
								
									
										2281
									
								
								app/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						| @@ -6,12 +6,14 @@ | ||||
|   "main": ".webpack/main", | ||||
|   "scripts": { | ||||
|     "start": "electron-forge start", | ||||
|     "package": "electron-forge package", | ||||
|     "package:sign": "SIGN=1 electron-forge package", | ||||
|     "make": "electron-forge make", | ||||
|     "make:sign": "SIGN=1 electron-forge make", | ||||
|     "package": "electron-forge package --arch universal", | ||||
|     "package:sign": "SIGN=1 electron-forge package --arch universal", | ||||
|     "make": "electron-forge make --arch universal", | ||||
|     "make:sign": "SIGN=1 electron-forge make --arch universal", | ||||
|     "publish": "SIGN=1 electron-forge publish", | ||||
|     "lint": "eslint --ext .ts,.tsx ." | ||||
|     "lint": "eslint --ext .ts,.tsx .", | ||||
|     "format": "prettier --check . --ignore-path .gitignore", | ||||
|     "format:fix": "prettier --write . --ignore-path .gitignore" | ||||
|   }, | ||||
|   "keywords": [], | ||||
|   "author": { | ||||
| @@ -30,6 +32,8 @@ | ||||
|     "@electron-forge/plugin-auto-unpack-natives": "^6.2.1", | ||||
|     "@electron-forge/plugin-webpack": "^6.2.1", | ||||
|     "@electron-forge/publisher-github": "^6.2.1", | ||||
|     "@electron/universal": "^1.4.1", | ||||
|     "@svgr/webpack": "^8.0.1", | ||||
|     "@types/chmodr": "^1.0.0", | ||||
|     "@types/node": "^20.4.0", | ||||
|     "@types/react": "^18.2.14", | ||||
| @@ -54,17 +58,21 @@ | ||||
|     "prettier": "^2.8.8", | ||||
|     "prettier-plugin-tailwindcss": "^0.3.0", | ||||
|     "style-loader": "^3.3.3", | ||||
|     "svg-inline-loader": "^0.8.2", | ||||
|     "tailwindcss": "^3.3.2", | ||||
|     "ts-loader": "^9.4.3", | ||||
|     "ts-node": "^10.9.1", | ||||
|     "typescript": "~4.5.4", | ||||
|     "url-loader": "^4.1.1", | ||||
|     "webpack": "^5.88.0", | ||||
|     "webpack-cli": "^5.1.4", | ||||
|     "webpack-dev-server": "^4.15.1" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "@electron/remote": "^2.0.10", | ||||
|     "@heroicons/react": "^2.0.18", | ||||
|     "@segment/analytics-node": "^1.0.0", | ||||
|     "copy-to-clipboard": "^3.3.3", | ||||
|     "electron-squirrel-startup": "^1.0.0", | ||||
|     "electron-store": "^8.1.0", | ||||
|     "react": "^18.2.0", | ||||
|   | ||||
| @@ -11,6 +11,10 @@ body { | ||||
|   -webkit-app-region: drag; | ||||
| } | ||||
|  | ||||
| .no-drag { | ||||
|   -webkit-app-region: no-drag; | ||||
| } | ||||
|  | ||||
| .blink { | ||||
|   -webkit-animation: 1s blink step-end infinite; | ||||
|   -moz-animation: 1s blink step-end infinite; | ||||
|   | ||||
							
								
								
									
										232
									
								
								app/src/app.tsx
									
									
									
									
									
								
							
							
						
						| @@ -1,158 +1,120 @@ | ||||
| import { useState } from 'react' | ||||
| import path from 'path' | ||||
| import os from 'os' | ||||
| import { dialog, getCurrentWindow } from '@electron/remote' | ||||
| import copy from 'copy-to-clipboard' | ||||
| import { CheckIcon, DocumentDuplicateIcon } from '@heroicons/react/24/outline' | ||||
| import Store from 'electron-store' | ||||
| import { getCurrentWindow, app } from '@electron/remote' | ||||
|  | ||||
| const API_URL = 'http://127.0.0.1:7734' | ||||
| import { install } from './install' | ||||
| import OllamaIcon from './ollama.svg' | ||||
|  | ||||
| type Message = { | ||||
|   sender: 'bot' | 'human' | ||||
|   content: string | ||||
| } | ||||
| const store = new Store() | ||||
|  | ||||
| const userInfo = os.userInfo() | ||||
|  | ||||
| async function generate(prompt: string, model: string, callback: (res: string) => void) { | ||||
|   const result = await fetch(`${API_URL}/generate`, { | ||||
|     method: 'POST', | ||||
|     headers: { | ||||
|       'Content-Type': 'application/json', | ||||
|     }, | ||||
|     body: JSON.stringify({ | ||||
|       prompt, | ||||
|       model, | ||||
|     }), | ||||
|   }) | ||||
|  | ||||
|   if (!result.ok) { | ||||
|     return | ||||
|   } | ||||
|  | ||||
|   let reader = result.body.getReader() | ||||
|  | ||||
|   while (true) { | ||||
|     const { done, value } = await reader.read() | ||||
|  | ||||
|     if (done) { | ||||
|       break | ||||
|     } | ||||
|  | ||||
|     let decoder = new TextDecoder() | ||||
|     let str = decoder.decode(value) | ||||
|  | ||||
|     let re = /}\s*{/g | ||||
|     str = '[' + str.replace(re, '},{') + ']' | ||||
|     let messages = JSON.parse(str) | ||||
|  | ||||
|     for (const message of messages) { | ||||
|       const choice = message.choices[0] | ||||
|  | ||||
|       callback(choice.text) | ||||
|  | ||||
|       if (choice.finish_reason === 'stop') { | ||||
|         break | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   return | ||||
| enum Step { | ||||
|   WELCOME = 0, | ||||
|   CLI, | ||||
|   FINISH, | ||||
| } | ||||
|  | ||||
| export default function () { | ||||
|   const [prompt, setPrompt] = useState('') | ||||
|   const [messages, setMessages] = useState<Message[]>([]) | ||||
|   const [model, setModel] = useState('') | ||||
|   const [generating, setGenerating] = useState(false) | ||||
|   const [step, setStep] = useState<Step>(Step.WELCOME) | ||||
|   const [commandCopied, setCommandCopied] = useState<boolean>(false) | ||||
|  | ||||
|   const command = 'ollama run llama2' | ||||
|  | ||||
|   return ( | ||||
|     <div className='flex min-h-screen flex-1 flex-col justify-between bg-white'> | ||||
|       <header className='drag sticky top-0 z-50 flex h-14 w-full flex-row items-center border-b border-black/10 bg-white/75 backdrop-blur-md'> | ||||
|         <div className='mx-auto w-full max-w-xl leading-none'> | ||||
|           <h1 className='text-sm font-medium'>{path.basename(model).replace('.bin', '')}</h1> | ||||
|     <div className='drag'> | ||||
|       <div className='mx-auto flex min-h-screen w-full flex-col justify-between bg-white px-4 pt-16'> | ||||
|         {step === Step.WELCOME && ( | ||||
|           <> | ||||
|             <div className='mx-auto text-center'> | ||||
|               <h1 className='mb-6 mt-4 text-2xl tracking-tight text-gray-900'>Welcome to Ollama</h1> | ||||
|               <p className='mx-auto w-[65%] text-sm text-gray-400'> | ||||
|                 Let's get you up and running with your own large language models. | ||||
|               </p> | ||||
|               <button | ||||
|                 onClick={() => setStep(Step.CLI)} | ||||
|                 className='no-drag rounded-dm mx-auto my-8 w-[40%] rounded-md bg-black px-4 py-2 text-sm text-white hover:brightness-110' | ||||
|               > | ||||
|                 Next | ||||
|               </button> | ||||
|             </div> | ||||
|       </header> | ||||
|       {model ? ( | ||||
|         <section className='mx-auto mb-10 w-full max-w-xl flex-1 break-words'> | ||||
|           {messages.map((m, i) => ( | ||||
|             <div className='my-4 flex gap-4' key={i}> | ||||
|               <div className='flex-none pr-1 text-lg'> | ||||
|                 {m.sender === 'human' ? ( | ||||
|                   <div className='mt-px flex h-6 w-6 items-center justify-center rounded-md bg-neutral-200 text-sm text-neutral-700'> | ||||
|                     {userInfo.username[0].toUpperCase()} | ||||
|                   </div> | ||||
|                 ) : ( | ||||
|                   <div className='mt-0.5 flex h-6 w-6 items-center justify-center rounded-md bg-blue-600 text-sm text-white'> | ||||
|                     {path.basename(model)[0].toUpperCase()} | ||||
|             <div className='mx-auto'> | ||||
|               <OllamaIcon /> | ||||
|             </div> | ||||
|           </> | ||||
|         )} | ||||
|               </div> | ||||
|               <div className='flex-1 text-gray-800'> | ||||
|                 {m.content} | ||||
|                 {m.sender === 'bot' && generating && i === messages.length - 1 && ( | ||||
|                   <span className='blink relative -top-[3px] left-1 text-[10px]'>█</span> | ||||
|                 )} | ||||
|               </div> | ||||
|             </div> | ||||
|           ))} | ||||
|         </section> | ||||
|       ) : ( | ||||
|         <section className='flex flex-1 select-none flex-col items-center justify-center pb-20'> | ||||
|           <h2 className='text-3xl font-light text-neutral-400'>No model selected</h2> | ||||
|         {step === Step.CLI && ( | ||||
|           <> | ||||
|             <div className='mx-auto flex flex-col space-y-28 text-center'> | ||||
|               <h1 className='mt-4 text-2xl tracking-tight text-gray-900'>Install the command line</h1> | ||||
|               <pre className='mx-auto text-4xl text-gray-400'>> ollama</pre> | ||||
|               <div className='mx-auto'> | ||||
|                 <button | ||||
|                   onClick={async () => { | ||||
|               const res = await dialog.showOpenDialog(getCurrentWindow(), { | ||||
|                 properties: ['openFile', 'multiSelections'], | ||||
|               }) | ||||
|               if (res.canceled) { | ||||
|                 return | ||||
|                     try { | ||||
|                       await install() | ||||
|                       setStep(Step.FINISH) | ||||
|                     } catch (e) { | ||||
|                       console.error('could not install: ', e) | ||||
|                     } finally { | ||||
|                       getCurrentWindow().show() | ||||
|                       getCurrentWindow().focus() | ||||
|                     } | ||||
|  | ||||
|               setModel(res.filePaths[0]) | ||||
|                   }} | ||||
|             className='rounded-dm my-8 rounded-md bg-blue-600 px-4 py-2 text-sm text-white hover:brightness-110' | ||||
|                   className='no-drag rounded-dm mx-auto w-[60%] rounded-md bg-black px-4 py-2 text-sm text-white hover:brightness-110' | ||||
|                 > | ||||
|             Open file... | ||||
|                   Install | ||||
|                 </button> | ||||
|         </section> | ||||
|                 <p className='mx-auto my-4 w-[70%] text-xs text-gray-400'> | ||||
|                   You will be prompted for administrator access | ||||
|                 </p> | ||||
|               </div> | ||||
|             </div> | ||||
|           </> | ||||
|         )} | ||||
|       <div className='sticky bottom-0 bg-gradient-to-b from-transparent to-white'> | ||||
|         {model && ( | ||||
|           <textarea | ||||
|             autoFocus | ||||
|             rows={1} | ||||
|             value={prompt} | ||||
|             placeholder='Send a message...' | ||||
|             onChange={e => setPrompt(e.target.value)} | ||||
|             className='mx-auto my-4 block w-full max-w-xl resize-none rounded-xl border border-gray-200 px-5 py-3.5 text-[15px] shadow-lg shadow-black/5 focus:outline-none' | ||||
|             onKeyDownCapture={async e => { | ||||
|               if (e.key === 'Enter' && !e.shiftKey) { | ||||
|                 e.preventDefault() | ||||
|  | ||||
|                 if (generating) { | ||||
|                   return | ||||
|                 } | ||||
|  | ||||
|                 if (!prompt) { | ||||
|                   return | ||||
|                 } | ||||
|  | ||||
|                 await setMessages(messages => { | ||||
|                   return [...messages, { sender: 'human', content: prompt }, { sender: 'bot', content: '' }] | ||||
|                 }) | ||||
|  | ||||
|                 setPrompt('') | ||||
|  | ||||
|                 setGenerating(true) | ||||
|                 await generate(prompt, model, res => { | ||||
|                   setMessages(messages => { | ||||
|                     let last = messages[messages.length - 1] | ||||
|                     return [...messages.slice(0, messages.length - 1), { ...last, content: last.content + res }] | ||||
|                   }) | ||||
|                 }) | ||||
|                 setGenerating(false) | ||||
|               } | ||||
|         {step === Step.FINISH && ( | ||||
|           <> | ||||
|             <div className='mx-auto flex flex-col space-y-20 text-center'> | ||||
|               <h1 className='mt-4 text-2xl tracking-tight text-gray-900'>Run your first model</h1> | ||||
|               <div className='flex flex-col'> | ||||
|                 <div className='group relative flex items-center'> | ||||
|                   <pre className='language-none text-2xs w-full rounded-md bg-gray-100 px-4 py-3 text-start leading-normal'> | ||||
|                     {command} | ||||
|                   </pre> | ||||
|                   <button | ||||
|                     className={`no-drag absolute right-[5px] px-2 py-2 ${ | ||||
|                       commandCopied | ||||
|                         ? 'text-gray-900 opacity-100 hover:cursor-auto' | ||||
|                         : 'text-gray-200 opacity-50 hover:cursor-pointer' | ||||
|                     } hover:font-bold hover:text-gray-900 group-hover:opacity-100`} | ||||
|                     onClick={() => { | ||||
|                       copy(command) | ||||
|                       setCommandCopied(true) | ||||
|                       setTimeout(() => setCommandCopied(false), 3000) | ||||
|                     }} | ||||
|           ></textarea> | ||||
|                   > | ||||
|                     {commandCopied ? ( | ||||
|                       <CheckIcon className='h-4 w-4 font-bold text-gray-500' /> | ||||
|                     ) : ( | ||||
|                       <DocumentDuplicateIcon className='h-4 w-4 text-gray-500' /> | ||||
|                     )} | ||||
|                   </button> | ||||
|                 </div> | ||||
|                 <p className='mx-auto my-4 w-[70%] text-xs text-gray-400'> | ||||
|                   Run this command in your favorite terminal. | ||||
|                 </p> | ||||
|               </div> | ||||
|               <button | ||||
|                 onClick={() => { | ||||
|                   store.set('first-time-run', true) | ||||
|                   window.close() | ||||
|                 }} | ||||
|                 className='no-drag rounded-dm mx-auto w-[60%] rounded-md bg-black px-4 py-2 text-sm text-white hover:brightness-110' | ||||
|               > | ||||
|                 Finish | ||||
|               </button> | ||||
|             </div> | ||||
|           </> | ||||
|         )} | ||||
|       </div> | ||||
|     </div> | ||||
|   | ||||
							
								
								
									
										4
									
								
								app/src/declarations.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| declare module '*.svg' { | ||||
|   const content: string | ||||
|   export default content | ||||
| } | ||||
							
								
								
									
										270
									
								
								app/src/index.ts
									
									
									
									
									
								
							
							
						
						| @@ -1,17 +1,24 @@ | ||||
| import { spawn, exec } from 'child_process' | ||||
| import { app, autoUpdater, dialog, Tray, Menu } from 'electron' | ||||
| import { spawn, ChildProcess } from 'child_process' | ||||
| import { app, autoUpdater, dialog, Tray, Menu, BrowserWindow, MenuItemConstructorOptions, nativeTheme } from 'electron' | ||||
| import Store from 'electron-store' | ||||
| import winston from 'winston' | ||||
| import 'winston-daily-rotate-file' | ||||
| import * as path from 'path' | ||||
| import * as fs from 'fs' | ||||
|  | ||||
| import { analytics, id } from './telemetry' | ||||
| import { v4 as uuidv4 } from 'uuid' | ||||
| import { installed } from './install' | ||||
|  | ||||
| require('@electron/remote/main').initialize() | ||||
|  | ||||
| if (require('electron-squirrel-startup')) { | ||||
|   app.quit() | ||||
| } | ||||
|  | ||||
| const store = new Store() | ||||
| let tray: Tray | null = null | ||||
|  | ||||
| let welcomeWindow: BrowserWindow | null = null | ||||
|  | ||||
| declare const MAIN_WINDOW_WEBPACK_ENTRY: string | ||||
|  | ||||
| const logger = winston.createLogger({ | ||||
|   transports: [ | ||||
| @@ -22,41 +29,116 @@ const logger = winston.createLogger({ | ||||
|       maxFiles: 5, | ||||
|     }), | ||||
|   ], | ||||
|   format: winston.format.printf(info => `${info.message}`), | ||||
|   format: winston.format.printf(info => info.message), | ||||
| }) | ||||
|  | ||||
| const SingleInstanceLock = app.requestSingleInstanceLock() | ||||
| if (!SingleInstanceLock) { | ||||
|   app.quit() | ||||
| app.on('ready', () => { | ||||
|   const gotTheLock = app.requestSingleInstanceLock() | ||||
|   if (!gotTheLock) { | ||||
|     app.exit(0) | ||||
|     return | ||||
|   } | ||||
|  | ||||
| const createSystemtray = () => { | ||||
|   let iconPath = path.join(__dirname, '..', '..', 'assets', 'ollama_icon_16x16Template.png') | ||||
|  | ||||
|   if (app.isPackaged) { | ||||
|     iconPath = path.join(process.resourcesPath, 'ollama_icon_16x16Template.png') | ||||
|   app.on('second-instance', () => { | ||||
|     if (app.hasSingleInstanceLock()) { | ||||
|       app.releaseSingleInstanceLock() | ||||
|     } | ||||
|  | ||||
|   tray = new Tray(iconPath) | ||||
|  | ||||
|   const contextMenu = Menu.buildFromTemplate([{ role: 'quit', label: 'Quit Ollama', accelerator: 'Command+Q' }]) | ||||
|  | ||||
|   tray.setContextMenu(contextMenu) | ||||
|   tray.setToolTip('Ollama') | ||||
|     if (proc) { | ||||
|       proc.off('exit', restart) | ||||
|       proc.kill() | ||||
|     } | ||||
|  | ||||
| if (require('electron-squirrel-startup')) { | ||||
|   app.quit() | ||||
|     app.exit(0) | ||||
|   }) | ||||
|  | ||||
|   app.focus({ steal: true }) | ||||
|  | ||||
|   init() | ||||
| }) | ||||
|  | ||||
| function firstRunWindow() { | ||||
|   // Create the browser window. | ||||
|   welcomeWindow = new BrowserWindow({ | ||||
|     width: 400, | ||||
|     height: 500, | ||||
|     frame: false, | ||||
|     fullscreenable: false, | ||||
|     resizable: false, | ||||
|     movable: true, | ||||
|     show: false, | ||||
|     webPreferences: { | ||||
|       nodeIntegration: true, | ||||
|       contextIsolation: false, | ||||
|     }, | ||||
|   }) | ||||
|  | ||||
|   require('@electron/remote/main').enable(welcomeWindow.webContents) | ||||
|  | ||||
|   welcomeWindow.loadURL(MAIN_WINDOW_WEBPACK_ENTRY) | ||||
|   welcomeWindow.on('ready-to-show', () => welcomeWindow.show()) | ||||
|   welcomeWindow.on('closed', () => { | ||||
|     if (process.platform === 'darwin') { | ||||
|       app.dock.hide() | ||||
|     } | ||||
|   }) | ||||
| } | ||||
|  | ||||
| const ollama = path.join(process.resourcesPath, 'ollama') | ||||
| let tray: Tray | null = null | ||||
| let updateAvailable = false | ||||
| const assetPath = app.isPackaged ? process.resourcesPath : path.join(__dirname, '..', '..', 'assets') | ||||
|  | ||||
| function trayIconPath() { | ||||
|   return nativeTheme.shouldUseDarkColors | ||||
|     ? updateAvailable | ||||
|       ? path.join(assetPath, 'iconDarkUpdateTemplate.png') | ||||
|       : path.join(assetPath, 'iconDarkTemplate.png') | ||||
|     : updateAvailable | ||||
|     ? path.join(assetPath, 'iconUpdateTemplate.png') | ||||
|     : path.join(assetPath, 'iconTemplate.png') | ||||
| } | ||||
|  | ||||
| function updateTrayIcon() { | ||||
|   if (tray) { | ||||
|     tray.setImage(trayIconPath()) | ||||
|   } | ||||
| } | ||||
|  | ||||
| function updateTray() { | ||||
|   const updateItems: MenuItemConstructorOptions[] = [ | ||||
|     { label: 'An update is available', enabled: false }, | ||||
|     { | ||||
|       label: 'Restart to update', | ||||
|       click: () => autoUpdater.quitAndInstall(), | ||||
|     }, | ||||
|     { type: 'separator' }, | ||||
|   ] | ||||
|  | ||||
|   const menu = Menu.buildFromTemplate([ | ||||
|     ...(updateAvailable ? updateItems : []), | ||||
|     { role: 'quit', label: 'Quit Ollama', accelerator: 'Command+Q' }, | ||||
|   ]) | ||||
|  | ||||
|   if (!tray) { | ||||
|     tray = new Tray(trayIconPath()) | ||||
|   } | ||||
|  | ||||
|   tray.setToolTip(updateAvailable ? 'An update is available' : 'Ollama') | ||||
|   tray.setContextMenu(menu) | ||||
|   tray.setImage(trayIconPath()) | ||||
|  | ||||
|   nativeTheme.off('updated', updateTrayIcon) | ||||
|   nativeTheme.on('updated', updateTrayIcon) | ||||
| } | ||||
|  | ||||
| let proc: ChildProcess = null | ||||
|  | ||||
| function server() { | ||||
|   const binary = app.isPackaged | ||||
|     ? path.join(process.resourcesPath, 'ollama') | ||||
|     : path.resolve(process.cwd(), '..', 'ollama') | ||||
|  | ||||
|   const proc = spawn(binary, ['serve']) | ||||
|   proc = spawn(binary, ['serve']) | ||||
|  | ||||
|   proc.stdout.on('data', data => { | ||||
|     logger.info(data.toString().trim()) | ||||
| @@ -66,66 +148,33 @@ function server() { | ||||
|     logger.error(data.toString().trim()) | ||||
|   }) | ||||
|  | ||||
|   proc.on('exit', () => { | ||||
|     logger.info('Restarting the server...') | ||||
|     server() | ||||
|   }) | ||||
|  | ||||
|   proc.on('disconnect', () => { | ||||
|     logger.info('Server disconnected. Reconnecting...') | ||||
|     server() | ||||
|   }) | ||||
|  | ||||
|   process.on('exit', () => { | ||||
|     proc.kill() | ||||
|   }) | ||||
|   proc.on('exit', restart) | ||||
| } | ||||
|  | ||||
| function installCLI() { | ||||
|   const symlinkPath = '/usr/local/bin/ollama' | ||||
|  | ||||
|   if (fs.existsSync(symlinkPath) && fs.readlinkSync(symlinkPath) === ollama) { | ||||
|     return | ||||
| function restart() { | ||||
|   setTimeout(server, 1000) | ||||
| } | ||||
|  | ||||
|   dialog | ||||
|     .showMessageBox({ | ||||
|       type: 'info', | ||||
|       title: 'Ollama CLI installation', | ||||
|       message: 'To make the Ollama command work in your terminal, it needs administrator privileges.', | ||||
|       buttons: ['OK'], | ||||
|     }) | ||||
|     .then(result => { | ||||
|       if (result.response === 0) { | ||||
|         const command = ` | ||||
|     do shell script "ln -F -s ${ollama} /usr/local/bin/ollama" with administrator privileges | ||||
|     ` | ||||
|         exec(`osascript -e '${command}'`, (error: Error | null, stdout: string, stderr: string) => { | ||||
|           if (error) { | ||||
|             logger.error(`cli: failed to install cli: ${error.message}`) | ||||
|             return | ||||
|           } | ||||
|  | ||||
|           logger.info(stdout) | ||||
|           logger.error(stderr) | ||||
|         }) | ||||
| app.on('before-quit', () => { | ||||
|   if (proc) { | ||||
|     proc.off('exit', restart) | ||||
|     proc.kill('SIGINT') // send SIGINT signal to the server, which also stops any loaded llms | ||||
|   } | ||||
| }) | ||||
|  | ||||
| function init() { | ||||
|   if (app.isPackaged) { | ||||
|     autoUpdater.checkForUpdates() | ||||
|     setInterval(() => { | ||||
|       if (!updateAvailable) { | ||||
|         autoUpdater.checkForUpdates() | ||||
|       } | ||||
|     }, 60 * 60 * 1000) | ||||
|   } | ||||
|  | ||||
| app.on('ready', () => { | ||||
|   updateTray() | ||||
|  | ||||
|   if (process.platform === 'darwin') { | ||||
|     app.dock.hide() | ||||
|  | ||||
|     if (!store.has('first-time-run')) { | ||||
|       // This is the first run | ||||
|       app.setLoginItemSettings({ openAtLogin: true }) | ||||
|       store.set('first-time-run', false) | ||||
|     } else { | ||||
|       // The app has been run before | ||||
|       app.setLoginItemSettings({ openAtLogin: app.getLoginItemSettings().openAtLogin }) | ||||
|     } | ||||
|  | ||||
|     if (app.isPackaged) { | ||||
|       if (!app.isInApplicationsFolder()) { | ||||
|         const chosen = dialog.showMessageBoxSync({ | ||||
| @@ -157,14 +206,24 @@ app.on('ready', () => { | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       installCLI() | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   createSystemtray() | ||||
|   server() | ||||
| }) | ||||
|  | ||||
|   if (store.get('first-time-run') && installed()) { | ||||
|     if (process.platform === 'darwin') { | ||||
|       app.dock.hide() | ||||
|     } | ||||
|  | ||||
|     app.setLoginItemSettings({ openAtLogin: app.getLoginItemSettings().openAtLogin }) | ||||
|     return | ||||
|   } | ||||
|  | ||||
|   // This is the first run or the CLI is no longer installed | ||||
|   app.setLoginItemSettings({ openAtLogin: true }) | ||||
|   firstRunWindow() | ||||
| } | ||||
|  | ||||
| // Quit when all windows are closed, except on macOS. There, it's common | ||||
| // for applications and their menu bar to stay active until the user quits | ||||
| @@ -175,45 +234,30 @@ app.on('window-all-closed', () => { | ||||
|   } | ||||
| }) | ||||
|  | ||||
| // In this file you can include the rest of your app's specific main process | ||||
| // code. You can also put them in separate files and import them here. | ||||
| function id(): string { | ||||
|   const id = store.get('id') as string | ||||
|  | ||||
|   if (id) { | ||||
|     return id | ||||
|   } | ||||
|  | ||||
|   const uuid = uuidv4() | ||||
|   store.set('id', uuid) | ||||
|   return uuid | ||||
| } | ||||
|  | ||||
| autoUpdater.setFeedURL({ | ||||
|   url: `https://ollama.ai/api/update?os=${process.platform}&arch=${process.arch}&version=${app.getVersion()}`, | ||||
|   url: `https://ollama.ai/api/update?os=${process.platform}&arch=${ | ||||
|     process.arch | ||||
|   }&version=${app.getVersion()}&id=${id()}`, | ||||
| }) | ||||
|  | ||||
| async function heartbeat() { | ||||
|   analytics.track({ | ||||
|     anonymousId: id(), | ||||
|     event: 'heartbeat', | ||||
|     properties: { | ||||
|       version: app.getVersion(), | ||||
|     }, | ||||
|   }) | ||||
| } | ||||
|  | ||||
| if (app.isPackaged) { | ||||
|   heartbeat() | ||||
|   autoUpdater.checkForUpdates() | ||||
|   setInterval(() => { | ||||
|     heartbeat() | ||||
|     autoUpdater.checkForUpdates() | ||||
|   }, 60 * 60 * 1000) | ||||
| } | ||||
|  | ||||
| autoUpdater.on('error', e => { | ||||
|   logger.error(`update check failed - ${e.message}`) | ||||
|   console.error(`update check failed - ${e.message}`) | ||||
| }) | ||||
|  | ||||
| autoUpdater.on('update-downloaded', (event, releaseNotes, releaseName) => { | ||||
|   dialog | ||||
|     .showMessageBox({ | ||||
|       type: 'info', | ||||
|       buttons: ['Restart Now', 'Later'], | ||||
|       title: 'New update available', | ||||
|       message: process.platform === 'win32' ? releaseNotes : releaseName, | ||||
|       detail: 'A new version of Ollama is available. Restart to apply the update.', | ||||
|     }) | ||||
|     .then(returnValue => { | ||||
|       if (returnValue.response === 0) autoUpdater.quitAndInstall() | ||||
|     }) | ||||
| autoUpdater.on('update-downloaded', () => { | ||||
|   updateAvailable = true | ||||
|   updateTray() | ||||
| }) | ||||
|   | ||||
							
								
								
									
										21
									
								
								app/src/install.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| import * as fs from 'fs' | ||||
| import { exec as cbExec } from 'child_process' | ||||
| import * as path from 'path' | ||||
| import { promisify } from 'util' | ||||
|  | ||||
| const app = process && process.type === 'renderer' ? require('@electron/remote').app : require('electron').app | ||||
| const ollama = app.isPackaged ? path.join(process.resourcesPath, 'ollama') : path.resolve(process.cwd(), '..', 'ollama') | ||||
| const exec = promisify(cbExec) | ||||
| const symlinkPath = '/usr/local/bin/ollama' | ||||
|  | ||||
| export function installed() { | ||||
|   return fs.existsSync(symlinkPath) && fs.readlinkSync(symlinkPath) === ollama | ||||
| } | ||||
|  | ||||
| export async function install() { | ||||
|   const command = `do shell script "mkdir -p ${path.dirname( | ||||
|     symlinkPath | ||||
|   )} && ln -F -s \\"${ollama}\\" \\"${symlinkPath}\\"" with administrator privileges` | ||||
|  | ||||
|   await exec(`osascript -e '${command}'`) | ||||
| } | ||||
							
								
								
									
										9
									
								
								app/src/ollama.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 17 KiB | 
| @@ -1,19 +0,0 @@ | ||||
| import { Analytics } from '@segment/analytics-node' | ||||
| import { v4 as uuidv4 } from 'uuid' | ||||
| import Store from 'electron-store' | ||||
|  | ||||
| const store = new Store() | ||||
|  | ||||
| export const analytics = new Analytics({ writeKey: process.env.TELEMETRY_WRITE_KEY || '<empty>' }) | ||||
|  | ||||
| export function id(): string { | ||||
|   const id = store.get('id') as string | ||||
|  | ||||
|   if (id) { | ||||
|     return id | ||||
|   } | ||||
|  | ||||
|   const uuid = uuidv4() | ||||
|   store.set('id', uuid) | ||||
|   return uuid | ||||
| } | ||||
| @@ -28,4 +28,8 @@ export const rules: Required<ModuleOptions>['rules'] = [ | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
|   { | ||||
|     test: /\.svg$/, | ||||
|     use: ['@svgr/webpack'], | ||||
|   }, | ||||
| ] | ||||
|   | ||||
							
								
								
									
										1093
									
								
								cmd/cmd.go
									
									
									
									
									
								
							
							
						
						
							
								
								
									
										44
									
								
								cmd/spinner.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,44 @@ | ||||
| package cmd | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/jmorganca/ollama/progressbar" | ||||
| ) | ||||
|  | ||||
| type Spinner struct { | ||||
| 	description string | ||||
| 	*progressbar.ProgressBar | ||||
| } | ||||
|  | ||||
| func NewSpinner(description string) *Spinner { | ||||
| 	return &Spinner{ | ||||
| 		description: description, | ||||
| 		ProgressBar: progressbar.NewOptions(-1, | ||||
| 			progressbar.OptionSetWriter(os.Stderr), | ||||
| 			progressbar.OptionThrottle(60*time.Millisecond), | ||||
| 			progressbar.OptionSpinnerType(14), | ||||
| 			progressbar.OptionSetRenderBlankState(true), | ||||
| 			progressbar.OptionSetElapsedTime(false), | ||||
| 			progressbar.OptionClearOnFinish(), | ||||
| 			progressbar.OptionSetDescription(description), | ||||
| 		), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (s *Spinner) Spin(tick time.Duration) { | ||||
| 	for range time.Tick(tick) { | ||||
| 		if s.IsFinished() { | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		s.Add(1) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (s *Spinner) Stop() { | ||||
| 	s.Finish() | ||||
| 	fmt.Println(s.description) | ||||
| } | ||||
							
								
								
									
										6
									
								
								docs/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,6 @@ | ||||
| # Documentation | ||||
|  | ||||
| - [Modelfile](./modelfile.md) | ||||
| - [How to develop Ollama](./development.md) | ||||
| - [API](./api.md) | ||||
| - [Tutorials](./tutorials.md) | ||||
							
								
								
									
										363
									
								
								docs/api.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,363 @@ | ||||
| # API | ||||
|  | ||||
| ## Endpoints | ||||
|  | ||||
| - [Generate a completion](#generate-a-completion) | ||||
| - [Create a Model](#create-a-model) | ||||
| - [List Local Models](#list-local-models) | ||||
| - [Show Model Information](#show-model-information) | ||||
| - [Copy a Model](#copy-a-model) | ||||
| - [Delete a Model](#delete-a-model) | ||||
| - [Pull a Model](#pull-a-model) | ||||
| - [Push a Model](#push-a-model) | ||||
| - [Generate Embeddings](#generate-embeddings) | ||||
|  | ||||
| ## Conventions | ||||
|  | ||||
| ### Model names | ||||
|  | ||||
| Model names follow a `model:tag` format. Some examples are `orca-mini:3b-q4_1` and `llama2:70b`. The tag is optional and, if not provided, will default to `latest`. The tag is used to identify a specific version. | ||||
|  | ||||
| ### Durations | ||||
|  | ||||
| All durations are returned in nanoseconds. | ||||
|  | ||||
| ### Streaming responses | ||||
|  | ||||
| Certain endpoints stream responses as JSON objects delineated with the newline (`\n`) character. | ||||
|  | ||||
| ## Generate a completion | ||||
|  | ||||
| ```shell | ||||
| POST /api/generate | ||||
| ``` | ||||
|  | ||||
| Generate a response for a given prompt with a provided model. This is a streaming endpoint, so will be a series of responses. The final response object will include statistics and additional data from the request. | ||||
|  | ||||
| ### Parameters | ||||
|  | ||||
| - `model`: (required) the [model name](#model-names) | ||||
| - `prompt`: the prompt to generate a response for | ||||
|  | ||||
| Advanced parameters (optional): | ||||
|  | ||||
| - `options`: additional model parameters listed in the documentation for the [Modelfile](./modelfile.md#valid-parameters-and-values) such as `temperature` | ||||
| - `system`: system prompt to (overrides what is defined in the `Modelfile`) | ||||
| - `template`: the full prompt or prompt template (overrides what is defined in the `Modelfile`) | ||||
| - `context`: the context parameter returned from a previous request to `/generate`, this can be used to keep a short conversational memory | ||||
| - `stream`: if `false` the response will be be returned as a single response object, rather than a stream of objects | ||||
|  | ||||
| ### Request | ||||
|  | ||||
| ```shell | ||||
| curl -X POST http://localhost:11434/api/generate -d '{ | ||||
|   "model": "llama2:7b", | ||||
|   "prompt": "Why is the sky blue?" | ||||
| }' | ||||
| ``` | ||||
|  | ||||
| ### Response | ||||
|  | ||||
| A stream of JSON objects: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "model": "llama2:7b", | ||||
|   "created_at": "2023-08-04T08:52:19.385406455-07:00", | ||||
|   "response": "The", | ||||
|   "done": false | ||||
| } | ||||
| ``` | ||||
|  | ||||
| The final response in the stream also includes additional data about the generation: | ||||
|  | ||||
| - `total_duration`: time spent generating the response | ||||
| - `load_duration`: time spent in nanoseconds loading the model | ||||
| - `sample_count`: number of samples generated | ||||
| - `sample_duration`: time spent generating samples | ||||
| - `prompt_eval_count`: number of tokens in the prompt | ||||
| - `prompt_eval_duration`: time spent in nanoseconds evaluating the prompt | ||||
| - `eval_count`: number of tokens the response | ||||
| - `eval_duration`: time in nanoseconds spent generating the response | ||||
| - `context`: an encoding of the conversation used in this response, this can be sent in the next request to keep a conversational memory | ||||
| - `response`: empty if the response was streamed, if not streamed, this will contain the full response | ||||
|  | ||||
| To calculate how fast the response is generated in tokens per second (token/s), divide `eval_count` / `eval_duration`. | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "model": "llama2:7b", | ||||
|   "created_at": "2023-08-04T19:22:45.499127Z", | ||||
|   "response": "", | ||||
|   "context": [1, 2, 3], | ||||
|   "done": true, | ||||
|   "total_duration": 5589157167, | ||||
|   "load_duration": 3013701500, | ||||
|   "sample_count": 114, | ||||
|   "sample_duration": 81442000, | ||||
|   "prompt_eval_count": 46, | ||||
|   "prompt_eval_duration": 1160282000, | ||||
|   "eval_count": 113, | ||||
|   "eval_duration": 1325948000 | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Create a Model | ||||
|  | ||||
| ```shell | ||||
| POST /api/create | ||||
| ``` | ||||
|  | ||||
| Create a model from a [`Modelfile`](./modelfile.md) | ||||
|  | ||||
| ### Parameters | ||||
|  | ||||
| - `name`: name of the model to create | ||||
| - `path`: path to the Modelfile | ||||
| - `stream`: (optional) if `false` the response will be be returned as a single response object, rather than a stream of objects | ||||
|  | ||||
| ### Request | ||||
|  | ||||
| ```shell | ||||
| curl -X POST http://localhost:11434/api/create -d '{ | ||||
|   "name": "mario", | ||||
|   "path": "~/Modelfile" | ||||
| }' | ||||
| ``` | ||||
|  | ||||
| ### Response | ||||
|  | ||||
| A stream of JSON objects. When finished, `status` is `success`. | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "status": "parsing modelfile" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## List Local Models | ||||
|  | ||||
| ```shell | ||||
| GET /api/tags | ||||
| ``` | ||||
|  | ||||
| List models that are available locally. | ||||
|  | ||||
| ### Request | ||||
|  | ||||
| ```shell | ||||
| curl http://localhost:11434/api/tags | ||||
| ``` | ||||
|  | ||||
| ### Response | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "models": [ | ||||
|     { | ||||
|       "name": "llama2:7b", | ||||
|       "modified_at": "2023-08-02T17:02:23.713454393-07:00", | ||||
|       "size": 3791730596 | ||||
|     }, | ||||
|     { | ||||
|       "name": "llama2:13b", | ||||
|       "modified_at": "2023-08-08T12:08:38.093596297-07:00", | ||||
|       "size": 7323310500 | ||||
|     } | ||||
|   ] | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Show Model Information | ||||
|  | ||||
| ```shell | ||||
| POST /api/show | ||||
| ``` | ||||
|  | ||||
| Show details about a model including modelfile, template, parameters, license, and system prompt. | ||||
|  | ||||
| ### Parameters | ||||
|  | ||||
| - `name`: name of the model to show | ||||
|  | ||||
| ### Request | ||||
|  | ||||
| ```shell | ||||
| curl http://localhost:11434/api/show -d '{ | ||||
|   "name": "llama2:7b" | ||||
| }' | ||||
| ``` | ||||
|  | ||||
| ### Response | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "license": "<contents of license block>", | ||||
|   "modelfile": "# Modelfile generated by \"ollama show\"\n# To build a new Modelfile based on this one, replace the FROM line with:\n# FROM llama2:latest\n\nFROM /Users/username/.ollama/models/blobs/sha256:8daa9615cce30c259a9555b1cc250d461d1bc69980a274b44d7eda0be78076d8\nTEMPLATE \"\"\"[INST] {{ if and .First .System }}<<SYS>>{{ .System }}<</SYS>>\n\n{{ end }}{{ .Prompt }} [/INST] \"\"\"\nSYSTEM \"\"\"\"\"\"\nPARAMETER stop [INST]\nPARAMETER stop [/INST]\nPARAMETER stop <<SYS>>\nPARAMETER stop <</SYS>>\n", | ||||
|   "parameters": "stop                           [INST]\nstop                           [/INST]\nstop                           <<SYS>>\nstop                           <</SYS>>", | ||||
|   "template": "[INST] {{ if and .First .System }}<<SYS>>{{ .System }}<</SYS>>\n\n{{ end }}{{ .Prompt }} [/INST] " | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Copy a Model | ||||
|  | ||||
| ```shell | ||||
| POST /api/copy | ||||
| ``` | ||||
|  | ||||
| Copy a model. Creates a model with another name from an existing model. | ||||
|  | ||||
| ### Request | ||||
|  | ||||
| ```shell | ||||
| curl http://localhost:11434/api/copy -d '{ | ||||
|   "source": "llama2:7b", | ||||
|   "destination": "llama2-backup" | ||||
| }' | ||||
| ``` | ||||
|  | ||||
| ## Delete a Model | ||||
|  | ||||
| ```shell | ||||
| DELETE /api/delete | ||||
| ``` | ||||
|  | ||||
| Delete a model and its data. | ||||
|  | ||||
| ### Parameters | ||||
|  | ||||
| - `model`: model name to delete | ||||
|  | ||||
| ### Request | ||||
|  | ||||
| ```shell | ||||
| curl -X DELETE http://localhost:11434/api/delete -d '{ | ||||
|   "name": "llama2:13b" | ||||
| }' | ||||
| ``` | ||||
|  | ||||
| ## Pull a Model | ||||
|  | ||||
| ```shell | ||||
| POST /api/pull | ||||
| ``` | ||||
|  | ||||
| Download a model from the ollama library. Cancelled pulls are resumed from where they left off, and multiple calls will share the same download progress. | ||||
|  | ||||
| ### Parameters | ||||
|  | ||||
| - `name`: name of the model to pull | ||||
| - `insecure`: (optional) allow insecure connections to the library. Only use this if you are pulling from your own library during development. | ||||
| - `stream`: (optional) if `false` the response will be be returned as a single response object, rather than a stream of objects | ||||
|  | ||||
| ### Request | ||||
|  | ||||
| ```shell | ||||
| curl -X POST http://localhost:11434/api/pull -d '{ | ||||
|   "name": "llama2:7b" | ||||
| }' | ||||
| ``` | ||||
|  | ||||
| ### Response | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "status": "downloading digestname", | ||||
|   "digest": "digestname", | ||||
|   "total": 2142590208 | ||||
| } | ||||
| ``` | ||||
|  | ||||
| ## Push a Model | ||||
|  | ||||
| ```shell | ||||
| POST /api/push | ||||
| ``` | ||||
|  | ||||
| Upload a model to a model library. Requires registering for ollama.ai and adding a public key first. | ||||
|  | ||||
| ### Parameters | ||||
|  | ||||
| - `name`: name of the model to push in the form of `<namespace>/<model>:<tag>` | ||||
| - `insecure`: (optional) allow insecure connections to the library. Only use this if you are pushing to your library during development. | ||||
| - `stream`: (optional) if `false` the response will be be returned as a single response object, rather than a stream of objects | ||||
|  | ||||
| ### Request | ||||
|  | ||||
| ```shell | ||||
| curl -X POST http://localhost:11434/api/push -d '{ | ||||
|   "name": "mattw/pygmalion:latest" | ||||
| }' | ||||
| ``` | ||||
|  | ||||
| ### Response | ||||
|  | ||||
| Streaming response that starts with: | ||||
|  | ||||
| ```json | ||||
| { "status": "retrieving manifest" } | ||||
| ``` | ||||
|  | ||||
| and then: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "status": "starting upload", | ||||
|   "digest": "sha256:bc07c81de745696fdf5afca05e065818a8149fb0c77266fb584d9b2cba3711ab", | ||||
|   "total": 1928429856 | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Then there is a series of uploading responses: | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "status": "starting upload", | ||||
|   "digest": "sha256:bc07c81de745696fdf5afca05e065818a8149fb0c77266fb584d9b2cba3711ab", | ||||
|   "total": 1928429856 | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Finally, when the upload is complete: | ||||
|  | ||||
| ```json | ||||
| {"status":"pushing manifest"} | ||||
| {"status":"success"} | ||||
| ``` | ||||
|  | ||||
| ## Generate Embeddings | ||||
|  | ||||
| ```shell | ||||
| POST /api/embeddings | ||||
| ``` | ||||
|  | ||||
| Generate embeddings from a model | ||||
|  | ||||
| ### Parameters | ||||
|  | ||||
| - `model`: name of model to generate embeddings from | ||||
| - `prompt`: text to generate embeddings for | ||||
|  | ||||
| Advanced parameters: | ||||
|  | ||||
| - `options`: additional model parameters listed in the documentation for the [Modelfile](./modelfile.md#valid-parameters-and-values) such as `temperature` | ||||
|  | ||||
| ### Request | ||||
|  | ||||
| ```shell | ||||
| curl -X POST http://localhost:11434/api/embeddings -d '{ | ||||
|   "model": "llama2:7b", | ||||
|   "prompt": "Here is an article about llamas..." | ||||
| }' | ||||
| ``` | ||||
|  | ||||
| ### Response | ||||
|  | ||||
| ```json | ||||
| { | ||||
|   "embeddings": [ | ||||
|     0.5670403838157654, 0.009260174818336964, 0.23178744316101074, -0.2916173040866852, -0.8924556970596313, | ||||
|     0.8785552978515625, -0.34576427936553955, 0.5742510557174683, -0.04222835972905159, -0.137906014919281 | ||||
|   ] | ||||
| } | ||||
| ``` | ||||
| @@ -1,40 +1,39 @@ | ||||
| # Development | ||||
|  | ||||
| - Install cmake or (optionally, required tools for GPUs) | ||||
| - run `go generate ./...` | ||||
| - run `go build .` | ||||
|  | ||||
| Install required tools: | ||||
|  | ||||
| ``` | ||||
| brew install cmake go node | ||||
| - cmake version 3.24 or higher | ||||
| - go version 1.20 or higher | ||||
| - gcc version 11.4.0 or higher | ||||
|  | ||||
| ```bash | ||||
| brew install go cmake gcc | ||||
| ``` | ||||
|  | ||||
| Then run `make`: | ||||
| Get the required libraries: | ||||
|  | ||||
| ```bash | ||||
| go generate ./... | ||||
| ``` | ||||
| make | ||||
|  | ||||
| Then build ollama: | ||||
|  | ||||
| ```bash | ||||
| go build . | ||||
| ``` | ||||
|  | ||||
| Now you can run `ollama`: | ||||
|  | ||||
| ``` | ||||
| ```bash | ||||
| ./ollama | ||||
| ``` | ||||
|  | ||||
| ## Releasing | ||||
|  | ||||
| To release a new version of Ollama you'll need to set some environment variables: | ||||
|  | ||||
| * `GITHUB_TOKEN`: your GitHub token | ||||
| * `APPLE_IDENTITY`: the Apple signing identity (macOS only) | ||||
| * `APPLE_ID`: your Apple ID | ||||
| * `APPLE_PASSWORD`: your Apple ID app-specific password | ||||
| * `APPLE_TEAM_ID`: the Apple team ID for the signing identity | ||||
| * `TELEMETRY_WRITE_KEY`: segment write key for telemetry | ||||
|  | ||||
| Then run the publish script with the target version: | ||||
|  | ||||
| ``` | ||||
| VERSION=0.0.2 ./scripts/publish.sh | ||||
| ``` | ||||
|  | ||||
|  | ||||
|  | ||||
| ## Building on Linux with GPU support | ||||
|  | ||||
| - Install cmake and nvidia-cuda-toolkit | ||||
| - run `go generate ./...` | ||||
| - run `go build .` | ||||
|   | ||||
							
								
								
									
										18
									
								
								docs/faq.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,18 @@ | ||||
| # FAQ | ||||
|  | ||||
| ## How can I expose the Ollama server? | ||||
|  | ||||
| ```bash | ||||
| OLLAMA_HOST=0.0.0.0:11435 ollama serve | ||||
| ``` | ||||
|  | ||||
| By default, Ollama allows cross origin requests from `127.0.0.1` and `0.0.0.0`. To support more origins, you can use the `OLLAMA_ORIGINS` environment variable: | ||||
|  | ||||
| ```bash | ||||
| OLLAMA_ORIGINS=http://192.168.1.1:*,https://example.com ollama serve | ||||
| ``` | ||||
|  | ||||
| ## Where are models stored? | ||||
|  | ||||
| * macOS: Raw model data is stored under `~/.ollama/models`. | ||||
| * Linux: Raw model data is stored under `/usr/share/ollama/.ollama/models` | ||||
							
								
								
									
										83
									
								
								docs/linux.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,83 @@ | ||||
| # Installing Ollama on Linux | ||||
|  | ||||
| > Note: A one line installer for Ollama is available by running: | ||||
| > | ||||
| > ```bash | ||||
| > curl https://ollama.ai/install.sh | sh | ||||
| > ``` | ||||
|  | ||||
| ## Download the `ollama` binary | ||||
|  | ||||
| Ollama is distributed as a self-contained binary. Download it to a directory in your PATH: | ||||
|  | ||||
| ```bash | ||||
| sudo curl -L https://ollama.ai/download/ollama-linux-amd64 -o /usr/bin/ollama | ||||
| sudo chmod +x /usr/bin/ollama | ||||
| ``` | ||||
|  | ||||
| ## Start Ollama | ||||
|  | ||||
| Start Ollama by running `ollama serve`: | ||||
|  | ||||
| ```bash | ||||
| ollama serve | ||||
| ``` | ||||
|  | ||||
| Once Ollama is running, run a model in another terminal session: | ||||
|  | ||||
| ```bash | ||||
| ollama run llama2 | ||||
| ``` | ||||
|  | ||||
| ## Install CUDA drivers (optional – for Nvidia GPUs) | ||||
|  | ||||
| [Download and install](https://developer.nvidia.com/cuda-downloads) CUDA. | ||||
|  | ||||
| Verify that the drivers are installed by running the following command, which should print details about your GPU: | ||||
|  | ||||
| ```bash | ||||
| nvidia-smi | ||||
| ``` | ||||
|  | ||||
| ## Adding Ollama as a startup service (optional) | ||||
|  | ||||
| Create a user for Ollama: | ||||
|  | ||||
| ```bash | ||||
| sudo useradd -r -s /bin/false -m -d /usr/share/ollama ollama | ||||
| ``` | ||||
|  | ||||
| Create a service file in `/etc/systemd/system/ollama.service`: | ||||
|  | ||||
| ```ini | ||||
| [Unit] | ||||
| Description=Ollama Service | ||||
| After=network-online.target | ||||
|  | ||||
| [Service] | ||||
| ExecStart=/usr/bin/ollama serve | ||||
| User=ollama | ||||
| Group=ollama | ||||
| Restart=always | ||||
| RestartSec=3 | ||||
| Environment="HOME=/usr/share/ollama" | ||||
|  | ||||
| [Install] | ||||
| WantedBy=default.target | ||||
| ``` | ||||
|  | ||||
| Then start the service: | ||||
|  | ||||
| ```bash | ||||
| sudo systemctl daemon-reload | ||||
| sudo systemctl enable ollama | ||||
| ``` | ||||
|  | ||||
| ### Viewing logs | ||||
|  | ||||
| To view logs of Ollama running as a startup service, run: | ||||
|  | ||||
| ```bash | ||||
| journalctl -u ollama | ||||
| ``` | ||||
|  | ||||
							
								
								
									
										191
									
								
								docs/modelfile.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,191 @@ | ||||
| # Ollama Model File | ||||
|  | ||||
| > Note: this `Modelfile` syntax is in development | ||||
|  | ||||
| A model file is the blueprint to create and share models with Ollama. | ||||
|  | ||||
| ## Table of Contents | ||||
|  | ||||
| - [Format](#format) | ||||
| - [Examples](#examples) | ||||
| - [Instructions](#instructions) | ||||
|   - [FROM (Required)](#from-required) | ||||
|     - [Build from llama2](#build-from-llama2) | ||||
|     - [Build from a bin file](#build-from-a-bin-file) | ||||
|   - [EMBED](#embed) | ||||
|   - [PARAMETER](#parameter) | ||||
|     - [Valid Parameters and Values](#valid-parameters-and-values) | ||||
|   - [TEMPLATE](#template) | ||||
|     - [Template Variables](#template-variables) | ||||
|   - [SYSTEM](#system) | ||||
|   - [ADAPTER](#adapter) | ||||
|   - [LICENSE](#license) | ||||
| - [Notes](#notes) | ||||
|  | ||||
| ## Format | ||||
|  | ||||
| The format of the `Modelfile`: | ||||
|  | ||||
| ```modelfile | ||||
| # comment | ||||
| INSTRUCTION arguments | ||||
| ``` | ||||
|  | ||||
| | Instruction                         | Description                                                   | | ||||
| | ----------------------------------- | ------------------------------------------------------------- | | ||||
| | [`FROM`](#from-required) (required) | Defines the base model to use.                                | | ||||
| | [`PARAMETER`](#parameter)           | Sets the parameters for how Ollama will run the model.        | | ||||
| | [`TEMPLATE`](#template)             | The full prompt template to be sent to the model.             | | ||||
| | [`SYSTEM`](#system)                 | Specifies the system prompt that will be set in the template. | | ||||
| | [`ADAPTER`](#adapter)               | Defines the (Q)LoRA adapters to apply to the model.           | | ||||
| | [`LICENSE`](#license)               | Specifies the legal license.                                  | | ||||
|  | ||||
| ## Examples | ||||
|  | ||||
| An example of a `Modelfile` creating a mario blueprint: | ||||
|  | ||||
| ```modelfile | ||||
| FROM llama2 | ||||
| # sets the temperature to 1 [higher is more creative, lower is more coherent] | ||||
| PARAMETER temperature 1 | ||||
| # sets the context window size to 4096, this controls how many tokens the LLM can use as context to generate the next token | ||||
| PARAMETER num_ctx 4096 | ||||
|  | ||||
| # sets a custom system prompt to specify the behavior of the chat assistant | ||||
| SYSTEM You are Mario from super mario bros, acting as an assistant. | ||||
| ``` | ||||
|  | ||||
| To use this: | ||||
|  | ||||
| 1. Save it as a file (e.g. `Modelfile`) | ||||
| 2. `ollama create choose-a-model-name -f <location of the file e.g. ./Modelfile>'` | ||||
| 3. `ollama run choose-a-model-name` | ||||
| 4. Start using the model! | ||||
|  | ||||
| More examples are available in the [examples directory](../examples). | ||||
|  | ||||
| ## Instructions | ||||
|  | ||||
| ### FROM (Required) | ||||
|  | ||||
| The `FROM` instruction defines the base model to use when creating a model. | ||||
|  | ||||
| ```modelfile | ||||
| FROM <model name>:<tag> | ||||
| ``` | ||||
|  | ||||
| #### Build from llama2 | ||||
|  | ||||
| ```modelfile | ||||
| FROM llama2 | ||||
| ``` | ||||
|  | ||||
| A list of available base models: | ||||
| <https://github.com/jmorganca/ollama#model-library> | ||||
|  | ||||
| #### Build from a `bin` file | ||||
|  | ||||
| ```modelfile | ||||
| FROM ./ollama-model.bin | ||||
| ``` | ||||
|  | ||||
| This bin file location should be specified as an absolute path or relative to the `Modelfile` location. | ||||
|  | ||||
| ### EMBED | ||||
|  | ||||
| The `EMBED` instruction is used to add embeddings of files to a model. This is useful for adding custom data that the model can reference when generating an answer. Note that currently only text files are supported, formatted with each line as one embedding. | ||||
|  | ||||
| ```modelfile | ||||
| FROM <model name>:<tag> | ||||
| EMBED <file path>.txt | ||||
| EMBED <different file path>.txt | ||||
| EMBED <path to directory>/*.txt | ||||
| ``` | ||||
|  | ||||
| ### PARAMETER | ||||
|  | ||||
| The `PARAMETER` instruction defines a parameter that can be set when the model is run. | ||||
|  | ||||
| ```modelfile | ||||
| PARAMETER <parameter> <parametervalue> | ||||
| ``` | ||||
|  | ||||
| ### Valid Parameters and Values | ||||
|  | ||||
| | Parameter      | Description                                                                                                                                                                                                                                             | Value Type | Example Usage        | | ||||
| | -------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ---------- | -------------------- | | ||||
| | mirostat       | Enable Mirostat sampling for controlling perplexity. (default: 0, 0 = disabled, 1 = Mirostat, 2 = Mirostat 2.0)                                                                                                                                         | int        | mirostat 0           | | ||||
| | mirostat_eta   | Influences how quickly the algorithm responds to feedback from the generated text. A lower learning rate will result in slower adjustments, while a higher learning rate will make the algorithm more responsive. (Default: 0.1)                        | float      | mirostat_eta 0.1     | | ||||
| | mirostat_tau   | Controls the balance between coherence and diversity of the output. A lower value will result in more focused and coherent text. (Default: 5.0)                                                                                                         | float      | mirostat_tau 5.0     | | ||||
| | num_ctx        | Sets the size of the context window used to generate the next token. (Default: 2048)                                                                                                                                                                    | int        | num_ctx 4096         | | ||||
| | num_gqa        | The number of GQA groups in the transformer layer. Required for some models, for example it is 8 for llama2:70b                                                                                                                                         | int        | num_gqa 1            | | ||||
| | num_gpu        | The number of layers to send to the GPU(s). On macOS it defaults to 1 to enable metal support, 0 to disable.                                                                                                                                            | int        | num_gpu 50           | | ||||
| | num_thread     | Sets the number of threads to use during computation. By default, Ollama will detect this for optimal performance. It is recommended to set this value to the number of physical CPU cores your system has (as opposed to the logical number of cores). | int        | num_thread 8         | | ||||
| | repeat_last_n  | Sets how far back for the model to look back to prevent repetition. (Default: 64, 0 = disabled, -1 = num_ctx)                                                                                                                                           | int        | repeat_last_n 64     | | ||||
| | repeat_penalty | Sets how strongly to penalize repetitions. A higher value (e.g., 1.5) will penalize repetitions more strongly, while a lower value (e.g., 0.9) will be more lenient. (Default: 1.1)                                                                     | float      | repeat_penalty 1.1   | | ||||
| | temperature    | The temperature of the model. Increasing the temperature will make the model answer more creatively. (Default: 0.8)                                                                                                                                     | float      | temperature 0.7      | | ||||
| | seed | Sets the random number seed to use for generation. Setting this to a specific number will make the model generate the same text for the same prompt. | int | seed 42 | | ||||
| | stop           | Sets the stop sequences to use.                                                                                                                                                                                                                         | string     | stop "AI assistant:" | | ||||
| | tfs_z          | Tail free sampling is used to reduce the impact of less probable tokens from the output. A higher value (e.g., 2.0) will reduce the impact more, while a value of 1.0 disables this setting. (default: 1)                                               | float      | tfs_z 1              | | ||||
| | num_predict    | Maximum number of tokens to predict when generating text. (Default: 128, -1 = infinite generation, -2 = fill context)                                                                                                                                   | int        | num_predict 42       | | ||||
| | top_k          | Reduces the probability of generating nonsense. A higher value (e.g. 100) will give more diverse answers, while a lower value (e.g. 10) will be more conservative. (Default: 40)                                                                        | int        | top_k 40             | | ||||
| | top_p          | Works together with top-k. A higher value (e.g., 0.95) will lead to more diverse text, while a lower value (e.g., 0.5) will generate more focused and conservative text. (Default: 0.9)                                                                 | float      | top_p 0.9            | | ||||
|  | ||||
| ### TEMPLATE | ||||
|  | ||||
| `TEMPLATE` of the full prompt template to be passed into the model. It may include (optionally) a system prompt and a user's prompt. This is used to create a full custom prompt, and syntax may be model specific. You can usually find the template for a given model in the readme for that model. | ||||
|  | ||||
| #### Template Variables | ||||
|  | ||||
| | Variable        | Description                                                                                                  | | ||||
| | --------------- | ------------------------------------------------------------------------------------------------------------ | | ||||
| | `{{ .System }}` | The system prompt used to specify custom behavior, this must also be set in the Modelfile as an instruction. | | ||||
| | `{{ .Prompt }}` | The incoming prompt, this is not specified in the model file and will be set based on input.                 | | ||||
| | `{{ .First }}`  | A boolean value used to render specific template information for the first generation of a session.          | | ||||
|  | ||||
| ```modelfile | ||||
| TEMPLATE """ | ||||
| {{- if .First }} | ||||
| ### System: | ||||
| {{ .System }} | ||||
| {{- end }} | ||||
|  | ||||
| ### User: | ||||
| {{ .Prompt }} | ||||
|  | ||||
| ### Response: | ||||
| """ | ||||
|  | ||||
| SYSTEM """<system message>""" | ||||
| ``` | ||||
|  | ||||
| ### SYSTEM | ||||
|  | ||||
| The `SYSTEM` instruction specifies the system prompt to be used in the template, if applicable. | ||||
|  | ||||
| ```modelfile | ||||
| SYSTEM """<system message>""" | ||||
| ``` | ||||
|  | ||||
| ### ADAPTER | ||||
|  | ||||
| The `ADAPTER` instruction specifies the LoRA adapter to apply to the base model. The value of this instruction should be an absolute path or a path relative to the Modelfile and the file must be in a GGML file format. The adapter should be tuned from the base model otherwise the behaviour is undefined. | ||||
|  | ||||
| ```modelfile | ||||
| ADAPTER ./ollama-lora.bin | ||||
| ``` | ||||
|  | ||||
| ### LICENSE | ||||
|  | ||||
| The `LICENSE` instruction allows you to specify the legal license under which the model used with this Modelfile is shared or distributed. | ||||
|  | ||||
| ```modelfile | ||||
| LICENSE """ | ||||
| <license text> | ||||
| """ | ||||
| ``` | ||||
|  | ||||
| ## Notes | ||||
|  | ||||
| - the **`Modelfile` is not case sensitive**. In the examples, we use uppercase for instructions to make it easier to distinguish it from arguments. | ||||
| - Instructions can be in any order. In the examples, we start with FROM instruction to keep it easily readable. | ||||
| @@ -1,64 +0,0 @@ | ||||
| # Python SDK | ||||
|  | ||||
| ## Install | ||||
|  | ||||
| ``` | ||||
| pip install ollama | ||||
| ``` | ||||
|  | ||||
| ## Example | ||||
|  | ||||
| ```python | ||||
| import ollama | ||||
| ollama.generate("orca-mini-3b", "hi") | ||||
| ``` | ||||
|  | ||||
| ## Reference | ||||
|  | ||||
| ### `ollama.generate(model, message)` | ||||
|  | ||||
| Generate a completion | ||||
|  | ||||
| ```python | ||||
| ollama.generate("./llama-7b-ggml.bin", "hi") | ||||
| ``` | ||||
|  | ||||
| ### `ollama.models()` | ||||
|  | ||||
| List available local models | ||||
|  | ||||
| ```python | ||||
| models = ollama.models() | ||||
| ``` | ||||
|  | ||||
| ### `ollama.load(model)` | ||||
|  | ||||
| Manually a model for generation | ||||
|  | ||||
| ```python | ||||
| ollama.load("model") | ||||
| ``` | ||||
|  | ||||
| ### `ollama.unload(model)` | ||||
|  | ||||
| Unload a model | ||||
|  | ||||
| ```python | ||||
| ollama.unload("model") | ||||
| ``` | ||||
|  | ||||
| ### `ollama.pull(model)` | ||||
|  | ||||
| Download a model | ||||
|  | ||||
| ```python | ||||
| ollama.pull("huggingface.co/thebloke/llama-7b-ggml") | ||||
| ``` | ||||
|  | ||||
| ### `ollama.search(query)` | ||||
|  | ||||
| Search for compatible models that Ollama can run | ||||
|  | ||||
| ```python | ||||
| ollama.search("llama-7b") | ||||
| ``` | ||||
							
								
								
									
										111
									
								
								docs/quantize.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,111 @@ | ||||
| # How to Quantize a Model | ||||
|  | ||||
| Sometimes the model you want to work with is not available at [https://ollama.ai/library](https://ollama.ai/library). | ||||
|  | ||||
| ## Figure out if we can run the model? | ||||
|  | ||||
| Not all models will work with Ollama. There are a number of factors that go into whether we are able to work with the next cool model. First it has to work with llama.cpp. Then we have to have implemented the features of llama.cpp that it requires. And then, sometimes, even with both of those, the model might not work... | ||||
|  | ||||
| 1. What is the model you want to convert and upload? | ||||
| 2. Visit the model's page on HuggingFace. | ||||
| 3. Switch to the **Files and versions** tab. | ||||
| 4. Click on the **config.json** file. If there is no config.json file, it may not work. | ||||
| 5. Take note of the **architecture** list in the json file. | ||||
| 6. Does any entry in the list match one of the following architectures? | ||||
|     1. LlamaForCausalLM | ||||
|     2. MistralForCausalLM | ||||
|     3. RWForCausalLM | ||||
|     4. FalconForCausalLM | ||||
|     5. GPTNeoXForCausalLM | ||||
|     6. GPTBigCodeForCausalLM | ||||
| 7. If the answer is yes, then there is a good chance the model will run after being converted and quantized. | ||||
| 8. An alternative to this process is to visit [https://caniquant.tvl.st](https://caniquant.tvl.st) and enter the org/modelname in the box and submit. | ||||
|  | ||||
| At this point there are two processes you can use. You can either use a Docker container to convert and quantize, OR you can manually run the scripts. The Docker container is the easiest way to do it, but it requires you to have Docker installed on your machine. If you don't have Docker installed, you can follow the manual process. | ||||
|  | ||||
| ## Convert and Quantize with Docker | ||||
|  | ||||
| Run `docker run --rm -v /path/to/model/repo:/repo ollama/quantize -q quantlevel /repo`. For instance, if you have downloaded the latest Mistral 7B model, then clone it to your machine. Then change into that directory and you can run: | ||||
|  | ||||
| ```shell | ||||
| docker run --rm -v .:/repo ollama/quantize -q q4_0 /repo | ||||
| ``` | ||||
|  | ||||
| You can find the different quantization levels below under **Quantize the Model**. | ||||
|  | ||||
| This will output two files into the directory. First is a f16.bin file that is the model converted to GGUF. The second file is a q4_0.bin file which is the model quantized to a 4 bit quantization. You should rename it to something more descriptive. | ||||
|  | ||||
| You can find the repository for the Docker container here: [https://github.com/mxyng/quantize](https://github.com/mxyng/quantize) | ||||
|  | ||||
| For instance, if you wanted to convert the Mistral 7B model to a Q4 quantized model, then you could go through the following steps: | ||||
|  | ||||
| 1. First verify the model will potentially work. | ||||
| 2. Now clone Mistral 7B to your machine. You can find the command to run when you click the three vertical dots button on the model page, then click **Clone Repository**. | ||||
|    1. For this repo, the command is: | ||||
|  | ||||
|       ```shell | ||||
|       git lfs install | ||||
|       git clone https://huggingface.co/mistralai/Mistral-7B-v0.1 | ||||
|       ``` | ||||
|  | ||||
|    2. Navigate into the new directory and run `docker run --rm -v .:/repo ollama/quantize -q q4_0 /repo` | ||||
|    3. Now you can create a modelfile using the q4_0.bin file that was created. | ||||
|  | ||||
| ## Convert and Quantize Manually | ||||
|  | ||||
| ### Clone llama.cpp to your machine | ||||
|  | ||||
| If we know the model has a chance of working, then we need to convert and quantize. This is a matter of running two separate scripts in the llama.cpp project. | ||||
|  | ||||
| 1. Decide where you want the llama.cpp repository on your machine. | ||||
| 2. Navigate to that location and then run: | ||||
|  [`git clone https://github.com/ggerganov/llama.cpp.git`](https://github.com/ggerganov/llama.cpp.git) | ||||
|     1. If you don't have git installed, download this zip file and unzip it to that location: https://github.com/ggerganov/llama.cpp/archive/refs/heads/master.zip | ||||
| 3. Install the Python dependencies: `pip install torch transformers sentencepiece` | ||||
| 4. Run 'make' to build the project and the quantize executable. | ||||
|  | ||||
| ### Convert the model to GGUF | ||||
|  | ||||
| 1. Decide on the right convert script to run. What was the model architecture you found in the first section. | ||||
|     1. LlamaForCausalLM or MistralForCausalLM: | ||||
|     run `python3 convert.py <modelfilename>` | ||||
|     No need to specify fp16 or fp32. | ||||
|     2. FalconForCausalLM or RWForCausalLM: | ||||
|     run `python3 convert-falcon-hf-to-gguf.py <modelfilename> <fpsize>`   | ||||
|     fpsize depends on the weight size. 1 for fp16, 0 for fp32 | ||||
|     3. GPTNeoXForCausalLM: | ||||
|     run `python3 convert-gptneox-hf-to-gguf.py <modelfilename> <fpsize>` | ||||
|     fpsize depends on the weight size. 1 for fp16, 0 for fp32 | ||||
|     4. GPTBigCodeForCausalLM: | ||||
|     run `python3 convert-starcoder-hf-to-gguf.py <modelfilename> <fpsize>` | ||||
|     fpsize depends on the weight size. 1 for fp16, 0 for fp32 | ||||
|  | ||||
| ### Quantize the model | ||||
|  | ||||
| If the model converted successfully, there is a good chance it will also quantize successfully. Now you need to decide on the quantization to use. We will always try to create all the quantizations and upload them to the library. You should decide which level is more important to you and quantize accordingly. | ||||
|  | ||||
| The quantization options are as follows. Note that some architectures such as Falcon do not support K quants. | ||||
|  | ||||
| - Q4_0 | ||||
| - Q4_1 | ||||
| - Q5_0 | ||||
| - Q5_1 | ||||
| - Q2_K | ||||
| - Q3_K | ||||
| - Q3_K_S | ||||
| - Q3_K_M | ||||
| - Q3_K_L | ||||
| - Q4_K | ||||
| - Q4_K_S | ||||
| - Q4_K_M | ||||
| - Q5_K | ||||
| - Q5_K_S | ||||
| - Q5_K_M | ||||
| - Q6_K | ||||
| - Q8_0 | ||||
|  | ||||
| Run the following command `quantize <converted model from above> <output file> <quantization type>` | ||||
|  | ||||
| ## Now Create the Model | ||||
|  | ||||
| Now you can create the Ollama model. Refer to the [modelfile](./modelfile.md) doc for more information on doing that. | ||||
							
								
								
									
										8
									
								
								docs/tutorials.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | ||||
| # Tutorials | ||||
|  | ||||
| Here is a list of ways you can use Ollama with other tools to build interesting applications. | ||||
|  | ||||
| - [Using LangChain with Ollama in JavaScript](./tutorials/langchainjs.md) | ||||
| - [Using LangChain with Ollama in Python](./tutorials/langchainpy.md) | ||||
|  | ||||
| Also be sure to check out the [examples](../examples) directory for more ways to use Ollama. | ||||
							
								
								
									
										73
									
								
								docs/tutorials/langchainjs.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,73 @@ | ||||
| # Using LangChain with Ollama using JavaScript | ||||
|  | ||||
| In this tutorial, we are going to use JavaScript with LangChain and Ollama to learn about something just a touch more recent. In August 2023, there was a series of wildfires on Maui. There is no way an LLM trained before that time can know about this, since their training data would not include anything as recent as that. So we can find the [Wikipedia article about the fires](https://en.wikipedia.org/wiki/2023_Hawaii_wildfires) and ask questions about the contents. | ||||
|  | ||||
| To get started, let's just use **LangChain** to ask a simple question to a model. To do this with JavaScript, we need to install **LangChain**: | ||||
|  | ||||
| ```bash | ||||
| npm install langchain | ||||
| ``` | ||||
|  | ||||
| Now we can start building out our JavaScript: | ||||
|  | ||||
| ```javascript | ||||
| import { Ollama } from "langchain/llms/ollama"; | ||||
|  | ||||
| const ollama = new Ollama({ | ||||
|   baseUrl: "http://localhost:11434", | ||||
|   model: "llama2", | ||||
| }); | ||||
|  | ||||
| const answer = await ollama.call(`why is the sky blue?`); | ||||
|  | ||||
| console.log(answer); | ||||
| ``` | ||||
|  | ||||
| That will get us the same thing as if we ran `ollama run llama2 "why is the sky blue"` in the terminal. But we want to load a document from the web to ask a question against. **Cheerio** is a great library for ingesting a webpage, and **LangChain** uses it in their **CheerioWebBaseLoader**. So let's build that part of the app. | ||||
|  | ||||
| ```javascript | ||||
| import { CheerioWebBaseLoader } from "langchain/document_loaders/web/cheerio"; | ||||
|  | ||||
| const loader = new CheerioWebBaseLoader("https://en.wikipedia.org/wiki/2023_Hawaii_wildfires"); | ||||
| const data = loader.load(); | ||||
| ``` | ||||
|  | ||||
| That will load the document. Although this page is smaller than the Odyssey, it is certainly bigger than the context size for most LLMs. So we are going to need to split into smaller pieces, and then select just the pieces relevant to our question. This is a great use for a vector datastore. In this example, we will use the **MemoryVectorStore** that is part of **LangChain**. But there is one more thing we need to get the content into the datastore. We have to run an embeddings process that converts the tokens in the text into a series of vectors. And for that, we are going to use **Tensorflow**. There is a lot of stuff going on in this one. First, install the **Tensorflow** components that we need. | ||||
|  | ||||
| ```javascript | ||||
| npm install @tensorflow/tfjs-core@3.6.0 @tensorflow/tfjs-converter@3.6.0 @tensorflow-models/universal-sentence-encoder@1.3.3 @tensorflow/tfjs-node@4.10.0 | ||||
| ``` | ||||
|  | ||||
| If you just install those components without the version numbers, it will install the latest versions, but there are conflicts within **Tensorflow**, so you need to install the compatible versions. | ||||
|  | ||||
| ```javascript | ||||
| import { RecursiveCharacterTextSplitter } from "langchain/text_splitter" | ||||
| import { MemoryVectorStore } from "langchain/vectorstores/memory"; | ||||
| import "@tensorflow/tfjs-node"; | ||||
| import { TensorFlowEmbeddings } from "langchain/embeddings/tensorflow"; | ||||
|  | ||||
| // Split the text into 500 character chunks. And overlap each chunk by 20 characters | ||||
| const textSplitter = new RecursiveCharacterTextSplitter({ | ||||
|  chunkSize: 500, | ||||
|  chunkOverlap: 20 | ||||
| }); | ||||
| const splitDocs = await textSplitter.splitDocuments(data); | ||||
|  | ||||
| // Then use the TensorFlow Embedding to store these chunks in the datastore | ||||
| const vectorStore = await MemoryVectorStore.fromDocuments(splitDocs, new TensorFlowEmbeddings()); | ||||
| ``` | ||||
|  | ||||
| To connect the datastore to a question asked to a LLM, we need to use the concept at the heart of **LangChain**: the chain. Chains are a way to connect a number of activities together to accomplish a particular tasks. There are a number of chain types available, but for this tutorial we are using the **RetrievalQAChain**. | ||||
|  | ||||
| ```javascript | ||||
| import { RetrievalQAChain } from "langchain/chains"; | ||||
|  | ||||
| const retriever = vectorStore.asRetriever(); | ||||
| const chain = RetrievalQAChain.fromLLM(ollama, retriever); | ||||
| const result = await chain.call({query: "When was Hawaii's request for a major disaster declaration approved?"}); | ||||
| console.log(result.text) | ||||
| ``` | ||||
|  | ||||
| So we created a retriever, which is a way to return the chunks that match a query from a datastore. And then connect the retriever and the model via a chain. Finally, we send a query to the chain, which results in an answer using our document as a source. The answer it returned was correct, August 10, 2023. | ||||
|  | ||||
| And that is a simple introduction to what you can do with **LangChain** and **Ollama.** | ||||
							
								
								
									
										81
									
								
								docs/tutorials/langchainpy.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,81 @@ | ||||
| # Using LangChain with Ollama in Python | ||||
|  | ||||
| Let's imagine we are studying the classics, such as **the Odyssey** by **Homer**. We might have a question about Neleus and his family. If you ask llama2 for that info, you may get something like: | ||||
|  | ||||
| > I apologize, but I'm a large language model, I cannot provide information on individuals or families that do not exist in reality. Neleus is not a real person or character, and therefore does not have a family or any other personal details. My apologies for any confusion. Is there anything else I can help you with? | ||||
|  | ||||
| This sounds like a typical censored response, but even llama2-uncensored gives a mediocre answer: | ||||
|  | ||||
| > Neleus was a legendary king of Pylos and the father of Nestor, one of the Argonauts. His mother was Clymene, a sea nymph, while his father was Neptune, the god of the sea. | ||||
|  | ||||
| So let's figure out how we can use **LangChain** with Ollama to ask our question to the actual document, the Odyssey by Homer, using Python. | ||||
|  | ||||
| Let's start by asking a simple question that we can get an answer to from the **Llama2** model using **Ollama**. First, we need to install the **LangChain** package: | ||||
|  | ||||
| `pip install langchain` | ||||
|  | ||||
| Then we can create a model and ask the question: | ||||
|  | ||||
| ```python | ||||
| from langchain.llms import Ollama | ||||
| ollama = Ollama(base_url='http://localhost:11434', | ||||
| model="llama2") | ||||
| print(ollama("why is the sky blue")) | ||||
| ``` | ||||
|  | ||||
| Notice that we are defining the model and the base URL for Ollama. | ||||
|  | ||||
| Now let's load a document to ask questions against. I'll load up the Odyssey by Homer, which you can find at Project Gutenberg. We will need **WebBaseLoader** which is part of **LangChain** and loads text from any webpage. On my machine, I also needed to install **bs4** to get that to work, so run `pip install bs4`. | ||||
|  | ||||
| ```python | ||||
| from langchain.document_loaders import WebBaseLoader | ||||
| loader = WebBaseLoader("https://www.gutenberg.org/files/1727/1727-h/1727-h.htm") | ||||
| data = loader.load() | ||||
| ``` | ||||
|  | ||||
| This file is pretty big. Just the preface is 3000 tokens. Which means the full document won't fit into the context for the model. So we need to split it up into smaller pieces. | ||||
|  | ||||
| ```python | ||||
| from langchain.text_splitter import RecursiveCharacterTextSplitter | ||||
|  | ||||
| text_splitter=RecursiveCharacterTextSplitter(chunk_size=500, chunk_overlap=0) | ||||
| all_splits = text_splitter.split_documents(data) | ||||
| ``` | ||||
|  | ||||
| It's split up, but we have to find the relevant splits and then submit those to the model. We can do this by creating embeddings and storing them in a vector database. For now, we don't have embeddings built in to Ollama, though we will be adding that soon, so for now, we can use the GPT4All library for that. We will use ChromaDB in this example for a vector database. `pip install GPT4All chromadb` | ||||
|  | ||||
| ```python | ||||
| from langchain.embeddings import GPT4AllEmbeddings | ||||
| from langchain.vectorstores import Chroma | ||||
| vectorstore = Chroma.from_documents(documents=all_splits, embedding=GPT4AllEmbeddings()) | ||||
| ``` | ||||
|  | ||||
| Now let's ask a question from the document. **Who was Neleus, and who is in his family?** Neleus is a character in the Odyssey, and the answer can be found in our text. | ||||
|  | ||||
| ```python | ||||
| question="Who is Neleus and who is in Neleus' family?" | ||||
| docs = vectorstore.similarity_search(question) | ||||
| len(docs) | ||||
| ``` | ||||
|  | ||||
| This will output the number of matches for chunks of data similar to the search. | ||||
|  | ||||
| The next thing is to send the question and the relevant parts of the docs to the model to see if we can get a good answer. But we are stitching two parts of the process together, and that is called a chain. This means we need to define a chain: | ||||
|  | ||||
| ```python | ||||
| from langchain.chains import RetrievalQA | ||||
| qachain=RetrievalQA.from_chain_type(ollama, retriever=vectorstore.as_retriever()) | ||||
| qachain({"query": question}) | ||||
| ``` | ||||
|  | ||||
| The answer received from this chain was: | ||||
|  | ||||
| > Neleus is a character in Homer's "Odyssey" and is mentioned in the context of Penelope's suitors. Neleus is the father of Chloris, who is married to Neleus and bears him several children, including Nestor, Chromius, Periclymenus, and Pero. Amphinomus, the son of Nisus, is also mentioned as a suitor of Penelope and is known for his good natural disposition and agreeable conversation. | ||||
|  | ||||
| It's not a perfect answer, as it implies Neleus married his daughter when actually Chloris "was the youngest daughter to Amphion son of Iasus and king of Minyan Orchomenus, and was Queen in Pylos". | ||||
|  | ||||
| I updated the chunk_overlap for the text splitter to 20 and tried again and got a much better answer: | ||||
|  | ||||
| > Neleus is a character in Homer's epic poem "The Odyssey." He is the husband of Chloris, who is the youngest daughter of Amphion son of Iasus and king of Minyan Orchomenus. Neleus has several children with Chloris, including Nestor, Chromius, Periclymenus, and Pero. | ||||
|  | ||||
| And that is a much better answer. | ||||
							
								
								
									
										171
									
								
								examples/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,171 @@ | ||||
| node_modules | ||||
| # OSX | ||||
| .DS_STORE | ||||
|  | ||||
| # Models | ||||
| models/ | ||||
|  | ||||
| # Local Chroma db | ||||
| .chroma/ | ||||
| db/ | ||||
|  | ||||
| # Byte-compiled / optimized / DLL files | ||||
| __pycache__/ | ||||
| *.py[cod] | ||||
| *$py.class | ||||
|  | ||||
| # C extensions | ||||
| *.so | ||||
|  | ||||
| # Distribution / packaging | ||||
| .Python | ||||
| build/ | ||||
| develop-eggs/ | ||||
| dist/ | ||||
| downloads/ | ||||
| eggs/ | ||||
| .eggs/ | ||||
| lib/ | ||||
| lib64/ | ||||
| parts/ | ||||
| sdist/ | ||||
| var/ | ||||
| wheels/ | ||||
| share/python-wheels/ | ||||
| *.egg-info/ | ||||
| .installed.cfg | ||||
| *.egg | ||||
| MANIFEST | ||||
|  | ||||
| # PyInstaller | ||||
| #  Usually these files are written by a python script from a template | ||||
| #  before PyInstaller builds the exe, so as to inject date/other infos into it. | ||||
| *.manifest | ||||
| *.spec | ||||
|  | ||||
| # Installer logs | ||||
| pip-log.txt | ||||
| pip-delete-this-directory.txt | ||||
|  | ||||
| # Unit test / coverage reports | ||||
| htmlcov/ | ||||
| .tox/ | ||||
| .nox/ | ||||
| .coverage | ||||
| .coverage.* | ||||
| .cache | ||||
| nosetests.xml | ||||
| coverage.xml | ||||
| *.cover | ||||
| *.py,cover | ||||
| .hypothesis/ | ||||
| .pytest_cache/ | ||||
| cover/ | ||||
|  | ||||
| # Translations | ||||
| *.mo | ||||
| *.pot | ||||
|  | ||||
| # Django stuff: | ||||
| *.log | ||||
| local_settings.py | ||||
| db.sqlite3 | ||||
| db.sqlite3-journal | ||||
|  | ||||
| # Flask stuff: | ||||
| instance/ | ||||
| .webassets-cache | ||||
|  | ||||
| # Scrapy stuff: | ||||
| .scrapy | ||||
|  | ||||
| # Sphinx documentation | ||||
| docs/_build/ | ||||
|  | ||||
| # PyBuilder | ||||
| .pybuilder/ | ||||
| target/ | ||||
|  | ||||
| # Jupyter Notebook | ||||
| .ipynb_checkpoints | ||||
|  | ||||
| # IPython | ||||
| profile_default/ | ||||
| ipython_config.py | ||||
|  | ||||
| # pyenv | ||||
| #   For a library or package, you might want to ignore these files since the code is | ||||
| #   intended to run in multiple environments; otherwise, check them in: | ||||
| # .python-version | ||||
|  | ||||
| # pipenv | ||||
| #   According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. | ||||
| #   However, in case of collaboration, if having platform-specific dependencies or dependencies | ||||
| #   having no cross-platform support, pipenv may install dependencies that don't work, or not | ||||
| #   install all needed dependencies. | ||||
| #Pipfile.lock | ||||
|  | ||||
| # poetry | ||||
| #   Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control. | ||||
| #   This is especially recommended for binary packages to ensure reproducibility, and is more | ||||
| #   commonly ignored for libraries. | ||||
| #   https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control | ||||
| #poetry.lock | ||||
|  | ||||
| # pdm | ||||
| #   Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control. | ||||
| #pdm.lock | ||||
| #   pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it | ||||
| #   in version control. | ||||
| #   https://pdm.fming.dev/#use-with-ide | ||||
| .pdm.toml | ||||
|  | ||||
| # PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm | ||||
| __pypackages__/ | ||||
|  | ||||
| # Celery stuff | ||||
| celerybeat-schedule | ||||
| celerybeat.pid | ||||
|  | ||||
| # SageMath parsed files | ||||
| *.sage.py | ||||
|  | ||||
| # Environments | ||||
| .env | ||||
| .venv | ||||
| env/ | ||||
| venv/ | ||||
| ENV/ | ||||
| env.bak/ | ||||
| venv.bak/ | ||||
|  | ||||
| # Spyder project settings | ||||
| .spyderproject | ||||
| .spyproject | ||||
|  | ||||
| # Rope project settings | ||||
| .ropeproject | ||||
|  | ||||
| # mkdocs documentation | ||||
| /site | ||||
|  | ||||
| # mypy | ||||
| .mypy_cache/ | ||||
| .dmypy.json | ||||
| dmypy.json | ||||
|  | ||||
| # Pyre type checker | ||||
| .pyre/ | ||||
|  | ||||
| # pytype static type analyzer | ||||
| .pytype/ | ||||
|  | ||||
| # Cython debug symbols | ||||
| cython_debug/ | ||||
|  | ||||
| # PyCharm | ||||
| #  JetBrains specific template is maintained in a separate JetBrains.gitignore that can | ||||
| #  be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore | ||||
| #  and can be added to the global gitignore or merged into this file.  For a more nuclear | ||||
| #  option (not recommended) you can uncomment the following to ignore the entire idea folder. | ||||
| #.idea/ | ||||
							
								
								
									
										3
									
								
								examples/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| # Examples | ||||
|  | ||||
| This directory contains different examples of using Ollama. | ||||
							
								
								
									
										0
									
								
								examples/golang-simplegenerate/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										27
									
								
								examples/golang-simplegenerate/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"io" | ||||
| 	"log" | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
| 	body := []byte(`{"model":"mistral"}`) | ||||
| 	resp, err := http.Post("http://localhost:11434/api/generate", "application/json", bytes.NewBuffer(body)) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		fmt.Print(err.Error()) | ||||
| 		os.Exit(1) | ||||
| 	}  | ||||
|  | ||||
| 	responseData, err := io.ReadAll(resp.Body) | ||||
| 	if err != nil { | ||||
| 		log.Fatal(err) | ||||
| 	} | ||||
| 	fmt.Println(string(responseData)) | ||||
|  | ||||
| } | ||||
							
								
								
									
										21
									
								
								examples/langchain-python-rag-document/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| # LangChain Document QA | ||||
|  | ||||
| This example provides an interface for asking questions to a PDF document. | ||||
|  | ||||
| ## Setup | ||||
|  | ||||
| ``` | ||||
| pip install -r requirements.txt | ||||
| ``` | ||||
|  | ||||
| ## Run | ||||
|  | ||||
| ``` | ||||
| python main.py | ||||
| ``` | ||||
|  | ||||
| A prompt will appear, where questions may be asked: | ||||
|  | ||||
| ``` | ||||
| Query: How many locations does WeWork have? | ||||
| ``` | ||||
							
								
								
									
										61
									
								
								examples/langchain-python-rag-document/main.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,61 @@ | ||||
| from langchain.document_loaders import OnlinePDFLoader | ||||
| from langchain.vectorstores import Chroma | ||||
| from langchain.embeddings import GPT4AllEmbeddings | ||||
| from langchain import PromptTemplate | ||||
| from langchain.llms import Ollama | ||||
| from langchain.callbacks.manager import CallbackManager | ||||
| from langchain.callbacks.streaming_stdout import StreamingStdOutCallbackHandler | ||||
| from langchain.chains import RetrievalQA | ||||
| import sys | ||||
| import os | ||||
|  | ||||
| class SuppressStdout: | ||||
|     def __enter__(self): | ||||
|         self._original_stdout = sys.stdout | ||||
|         self._original_stderr = sys.stderr | ||||
|         sys.stdout = open(os.devnull, 'w') | ||||
|         sys.stderr = open(os.devnull, 'w') | ||||
|  | ||||
|     def __exit__(self, exc_type, exc_val, exc_tb): | ||||
|         sys.stdout.close() | ||||
|         sys.stdout = self._original_stdout | ||||
|         sys.stderr = self._original_stderr | ||||
|  | ||||
| # load the pdf and split it into chunks | ||||
| loader = OnlinePDFLoader("https://d18rn0p25nwr6d.cloudfront.net/CIK-0001813756/975b3e9b-268e-4798-a9e4-2a9a7c92dc10.pdf") | ||||
| data = loader.load() | ||||
|  | ||||
| from langchain.text_splitter import RecursiveCharacterTextSplitter | ||||
| text_splitter = RecursiveCharacterTextSplitter(chunk_size=500, chunk_overlap=0) | ||||
| all_splits = text_splitter.split_documents(data) | ||||
|  | ||||
| with SuppressStdout(): | ||||
|     vectorstore = Chroma.from_documents(documents=all_splits, embedding=GPT4AllEmbeddings()) | ||||
|  | ||||
| while True: | ||||
|     query = input("\nQuery: ") | ||||
|     if query == "exit": | ||||
|         break | ||||
|     if query.strip() == "": | ||||
|         continue | ||||
|  | ||||
|     # Prompt | ||||
|     template = """Use the following pieces of context to answer the question at the end.  | ||||
|     If you don't know the answer, just say that you don't know, don't try to make up an answer.  | ||||
|     Use three sentences maximum and keep the answer as concise as possible.  | ||||
|     {context} | ||||
|     Question: {question} | ||||
|     Helpful Answer:""" | ||||
|     QA_CHAIN_PROMPT = PromptTemplate( | ||||
|         input_variables=["context", "question"], | ||||
|         template=template, | ||||
|     ) | ||||
|  | ||||
|     llm = Ollama(model="llama2:13b", callback_manager=CallbackManager([StreamingStdOutCallbackHandler()])) | ||||
|     qa_chain = RetrievalQA.from_chain_type( | ||||
|         llm, | ||||
|         retriever=vectorstore.as_retriever(), | ||||
|         chain_type_kwargs={"prompt": QA_CHAIN_PROMPT}, | ||||
|     ) | ||||
|  | ||||
|     result = qa_chain({"query": query}) | ||||
							
								
								
									
										109
									
								
								examples/langchain-python-rag-document/requirements.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,109 @@ | ||||
| absl-py==1.4.0 | ||||
| aiohttp==3.8.5 | ||||
| aiosignal==1.3.1 | ||||
| anyio==3.7.1 | ||||
| astunparse==1.6.3 | ||||
| async-timeout==4.0.3 | ||||
| attrs==23.1.0 | ||||
| backoff==2.2.1 | ||||
| beautifulsoup4==4.12.2 | ||||
| bs4==0.0.1 | ||||
| cachetools==5.3.1 | ||||
| certifi==2023.7.22 | ||||
| cffi==1.15.1 | ||||
| chardet==5.2.0 | ||||
| charset-normalizer==3.2.0 | ||||
| Chroma==0.2.0 | ||||
| chroma-hnswlib==0.7.2 | ||||
| chromadb==0.4.5 | ||||
| click==8.1.6 | ||||
| coloredlogs==15.0.1 | ||||
| cryptography==41.0.3 | ||||
| dataclasses-json==0.5.14 | ||||
| fastapi==0.99.1 | ||||
| filetype==1.2.0 | ||||
| flatbuffers==23.5.26 | ||||
| frozenlist==1.4.0 | ||||
| gast==0.4.0 | ||||
| google-auth==2.22.0 | ||||
| google-auth-oauthlib==1.0.0 | ||||
| google-pasta==0.2.0 | ||||
| gpt4all==1.0.8 | ||||
| grpcio==1.57.0 | ||||
| h11==0.14.0 | ||||
| h5py==3.9.0 | ||||
| httptools==0.6.0 | ||||
| humanfriendly==10.0 | ||||
| idna==3.4 | ||||
| importlib-resources==6.0.1 | ||||
| joblib==1.3.2 | ||||
| keras==2.13.1 | ||||
| langchain==0.0.261 | ||||
| langsmith==0.0.21 | ||||
| libclang==16.0.6 | ||||
| lxml==4.9.3 | ||||
| Markdown==3.4.4 | ||||
| MarkupSafe==2.1.3 | ||||
| marshmallow==3.20.1 | ||||
| monotonic==1.6 | ||||
| mpmath==1.3.0 | ||||
| multidict==6.0.4 | ||||
| mypy-extensions==1.0.0 | ||||
| nltk==3.8.1 | ||||
| numexpr==2.8.5 | ||||
| numpy==1.24.3 | ||||
| oauthlib==3.2.2 | ||||
| onnxruntime==1.15.1 | ||||
| openapi-schema-pydantic==1.2.4 | ||||
| opt-einsum==3.3.0 | ||||
| overrides==7.4.0 | ||||
| packaging==23.1 | ||||
| pdf2image==1.16.3 | ||||
| pdfminer==20191125 | ||||
| pdfminer.six==20221105 | ||||
| Pillow==10.0.0 | ||||
| posthog==3.0.1 | ||||
| protobuf==4.24.0 | ||||
| pulsar-client==3.2.0 | ||||
| pyasn1==0.5.0 | ||||
| pyasn1-modules==0.3.0 | ||||
| pycparser==2.21 | ||||
| pycryptodome==3.18.0 | ||||
| pydantic==1.10.12 | ||||
| PyPika==0.48.9 | ||||
| python-dateutil==2.8.2 | ||||
| python-dotenv==1.0.0 | ||||
| python-magic==0.4.27 | ||||
| PyYAML==6.0.1 | ||||
| regex==2023.8.8 | ||||
| requests==2.31.0 | ||||
| requests-oauthlib==1.3.1 | ||||
| rsa==4.9 | ||||
| six==1.16.0 | ||||
| sniffio==1.3.0 | ||||
| soupsieve==2.4.1 | ||||
| SQLAlchemy==2.0.19 | ||||
| starlette==0.27.0 | ||||
| sympy==1.12 | ||||
| tabulate==0.9.0 | ||||
| tenacity==8.2.2 | ||||
| tensorboard==2.13.0 | ||||
| tensorboard-data-server==0.7.1 | ||||
| tensorflow==2.13.0 | ||||
| tensorflow-estimator==2.13.0 | ||||
| tensorflow-hub==0.14.0 | ||||
| tensorflow-macos==2.13.0 | ||||
| termcolor==2.3.0 | ||||
| tokenizers==0.13.3 | ||||
| tqdm==4.66.1 | ||||
| typing-inspect==0.9.0 | ||||
| typing_extensions==4.5.0 | ||||
| unstructured==0.9.2 | ||||
| urllib3==1.26.16 | ||||
| uvicorn==0.23.2 | ||||
| uvloop==0.17.0 | ||||
| watchfiles==0.19.0 | ||||
| websockets==11.0.3 | ||||
| Werkzeug==2.3.6 | ||||
| wrapt==1.15.0 | ||||
| yarl==1.9.2 | ||||
							
								
								
									
										170
									
								
								examples/langchain-python-rag-privategpt/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,170 @@ | ||||
| # OSX | ||||
| .DS_STORE | ||||
|  | ||||
| # Models | ||||
| models/ | ||||
|  | ||||
| # Local Chroma db | ||||
| .chroma/ | ||||
| db/ | ||||
|  | ||||
| # Byte-compiled / optimized / DLL files | ||||
| __pycache__/ | ||||
| *.py[cod] | ||||
| *$py.class | ||||
|  | ||||
| # C extensions | ||||
| *.so | ||||
|  | ||||
| # Distribution / packaging | ||||
| .Python | ||||
| build/ | ||||
| develop-eggs/ | ||||
| dist/ | ||||
| downloads/ | ||||
| eggs/ | ||||
| .eggs/ | ||||
| lib/ | ||||
| lib64/ | ||||
| parts/ | ||||
| sdist/ | ||||
| var/ | ||||
| wheels/ | ||||
| share/python-wheels/ | ||||
| *.egg-info/ | ||||
| .installed.cfg | ||||
| *.egg | ||||
| MANIFEST | ||||
|  | ||||
| # PyInstaller | ||||
| #  Usually these files are written by a python script from a template | ||||
| #  before PyInstaller builds the exe, so as to inject date/other infos into it. | ||||
| *.manifest | ||||
| *.spec | ||||
|  | ||||
| # Installer logs | ||||
| pip-log.txt | ||||
| pip-delete-this-directory.txt | ||||
|  | ||||
| # Unit test / coverage reports | ||||
| htmlcov/ | ||||
| .tox/ | ||||
| .nox/ | ||||
| .coverage | ||||
| .coverage.* | ||||
| .cache | ||||
| nosetests.xml | ||||
| coverage.xml | ||||
| *.cover | ||||
| *.py,cover | ||||
| .hypothesis/ | ||||
| .pytest_cache/ | ||||
| cover/ | ||||
|  | ||||
| # Translations | ||||
| *.mo | ||||
| *.pot | ||||
|  | ||||
| # Django stuff: | ||||
| *.log | ||||
| local_settings.py | ||||
| db.sqlite3 | ||||
| db.sqlite3-journal | ||||
|  | ||||
| # Flask stuff: | ||||
| instance/ | ||||
| .webassets-cache | ||||
|  | ||||
| # Scrapy stuff: | ||||
| .scrapy | ||||
|  | ||||
| # Sphinx documentation | ||||
| docs/_build/ | ||||
|  | ||||
| # PyBuilder | ||||
| .pybuilder/ | ||||
| target/ | ||||
|  | ||||
| # Jupyter Notebook | ||||
| .ipynb_checkpoints | ||||
|  | ||||
| # IPython | ||||
| profile_default/ | ||||
| ipython_config.py | ||||
|  | ||||
| # pyenv | ||||
| #   For a library or package, you might want to ignore these files since the code is | ||||
| #   intended to run in multiple environments; otherwise, check them in: | ||||
| # .python-version | ||||
|  | ||||
| # pipenv | ||||
| #   According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. | ||||
| #   However, in case of collaboration, if having platform-specific dependencies or dependencies | ||||
| #   having no cross-platform support, pipenv may install dependencies that don't work, or not | ||||
| #   install all needed dependencies. | ||||
| #Pipfile.lock | ||||
|  | ||||
| # poetry | ||||
| #   Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control. | ||||
| #   This is especially recommended for binary packages to ensure reproducibility, and is more | ||||
| #   commonly ignored for libraries. | ||||
| #   https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control | ||||
| #poetry.lock | ||||
|  | ||||
| # pdm | ||||
| #   Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control. | ||||
| #pdm.lock | ||||
| #   pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it | ||||
| #   in version control. | ||||
| #   https://pdm.fming.dev/#use-with-ide | ||||
| .pdm.toml | ||||
|  | ||||
| # PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm | ||||
| __pypackages__/ | ||||
|  | ||||
| # Celery stuff | ||||
| celerybeat-schedule | ||||
| celerybeat.pid | ||||
|  | ||||
| # SageMath parsed files | ||||
| *.sage.py | ||||
|  | ||||
| # Environments | ||||
| .env | ||||
| .venv | ||||
| env/ | ||||
| venv/ | ||||
| ENV/ | ||||
| env.bak/ | ||||
| venv.bak/ | ||||
|  | ||||
| # Spyder project settings | ||||
| .spyderproject | ||||
| .spyproject | ||||
|  | ||||
| # Rope project settings | ||||
| .ropeproject | ||||
|  | ||||
| # mkdocs documentation | ||||
| /site | ||||
|  | ||||
| # mypy | ||||
| .mypy_cache/ | ||||
| .dmypy.json | ||||
| dmypy.json | ||||
|  | ||||
| # Pyre type checker | ||||
| .pyre/ | ||||
|  | ||||
| # pytype static type analyzer | ||||
| .pytype/ | ||||
|  | ||||
| # Cython debug symbols | ||||
| cython_debug/ | ||||
|  | ||||
| # PyCharm | ||||
| #  JetBrains specific template is maintained in a separate JetBrains.gitignore that can | ||||
| #  be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore | ||||
| #  and can be added to the global gitignore or merged into this file.  For a more nuclear | ||||
| #  option (not recommended) you can uncomment the following to ignore the entire idea folder. | ||||
| #.idea/ | ||||
							
								
								
									
										201
									
								
								examples/langchain-python-rag-privategpt/LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,201 @@ | ||||
|                                  Apache License | ||||
|                            Version 2.0, January 2004 | ||||
|                         http://www.apache.org/licenses/ | ||||
|  | ||||
|    TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | ||||
|  | ||||
|    1. Definitions. | ||||
|  | ||||
|       "License" shall mean the terms and conditions for use, reproduction, | ||||
|       and distribution as defined by Sections 1 through 9 of this document. | ||||
|  | ||||
|       "Licensor" shall mean the copyright owner or entity authorized by | ||||
|       the copyright owner that is granting the License. | ||||
|  | ||||
|       "Legal Entity" shall mean the union of the acting entity and all | ||||
|       other entities that control, are controlled by, or are under common | ||||
|       control with that entity. For the purposes of this definition, | ||||
|       "control" means (i) the power, direct or indirect, to cause the | ||||
|       direction or management of such entity, whether by contract or | ||||
|       otherwise, or (ii) ownership of fifty percent (50%) or more of the | ||||
|       outstanding shares, or (iii) beneficial ownership of such entity. | ||||
|  | ||||
|       "You" (or "Your") shall mean an individual or Legal Entity | ||||
|       exercising permissions granted by this License. | ||||
|  | ||||
|       "Source" form shall mean the preferred form for making modifications, | ||||
|       including but not limited to software source code, documentation | ||||
|       source, and configuration files. | ||||
|  | ||||
|       "Object" form shall mean any form resulting from mechanical | ||||
|       transformation or translation of a Source form, including but | ||||
|       not limited to compiled object code, generated documentation, | ||||
|       and conversions to other media types. | ||||
|  | ||||
|       "Work" shall mean the work of authorship, whether in Source or | ||||
|       Object form, made available under the License, as indicated by a | ||||
|       copyright notice that is included in or attached to the work | ||||
|       (an example is provided in the Appendix below). | ||||
|  | ||||
|       "Derivative Works" shall mean any work, whether in Source or Object | ||||
|       form, that is based on (or derived from) the Work and for which the | ||||
|       editorial revisions, annotations, elaborations, or other modifications | ||||
|       represent, as a whole, an original work of authorship. For the purposes | ||||
|       of this License, Derivative Works shall not include works that remain | ||||
|       separable from, or merely link (or bind by name) to the interfaces of, | ||||
|       the Work and Derivative Works thereof. | ||||
|  | ||||
|       "Contribution" shall mean any work of authorship, including | ||||
|       the original version of the Work and any modifications or additions | ||||
|       to that Work or Derivative Works thereof, that is intentionally | ||||
|       submitted to Licensor for inclusion in the Work by the copyright owner | ||||
|       or by an individual or Legal Entity authorized to submit on behalf of | ||||
|       the copyright owner. For the purposes of this definition, "submitted" | ||||
|       means any form of electronic, verbal, or written communication sent | ||||
|       to the Licensor or its representatives, including but not limited to | ||||
|       communication on electronic mailing lists, source code control systems, | ||||
|       and issue tracking systems that are managed by, or on behalf of, the | ||||
|       Licensor for the purpose of discussing and improving the Work, but | ||||
|       excluding communication that is conspicuously marked or otherwise | ||||
|       designated in writing by the copyright owner as "Not a Contribution." | ||||
|  | ||||
|       "Contributor" shall mean Licensor and any individual or Legal Entity | ||||
|       on behalf of whom a Contribution has been received by Licensor and | ||||
|       subsequently incorporated within the Work. | ||||
|  | ||||
|    2. Grant of Copyright License. Subject to the terms and conditions of | ||||
|       this License, each Contributor hereby grants to You a perpetual, | ||||
|       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||
|       copyright license to reproduce, prepare Derivative Works of, | ||||
|       publicly display, publicly perform, sublicense, and distribute the | ||||
|       Work and such Derivative Works in Source or Object form. | ||||
|  | ||||
|    3. Grant of Patent License. Subject to the terms and conditions of | ||||
|       this License, each Contributor hereby grants to You a perpetual, | ||||
|       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||
|       (except as stated in this section) patent license to make, have made, | ||||
|       use, offer to sell, sell, import, and otherwise transfer the Work, | ||||
|       where such license applies only to those patent claims licensable | ||||
|       by such Contributor that are necessarily infringed by their | ||||
|       Contribution(s) alone or by combination of their Contribution(s) | ||||
|       with the Work to which such Contribution(s) was submitted. If You | ||||
|       institute patent litigation against any entity (including a | ||||
|       cross-claim or counterclaim in a lawsuit) alleging that the Work | ||||
|       or a Contribution incorporated within the Work constitutes direct | ||||
|       or contributory patent infringement, then any patent licenses | ||||
|       granted to You under this License for that Work shall terminate | ||||
|       as of the date such litigation is filed. | ||||
|  | ||||
|    4. Redistribution. You may reproduce and distribute copies of the | ||||
|       Work or Derivative Works thereof in any medium, with or without | ||||
|       modifications, and in Source or Object form, provided that You | ||||
|       meet the following conditions: | ||||
|  | ||||
|       (a) You must give any other recipients of the Work or | ||||
|           Derivative Works a copy of this License; and | ||||
|  | ||||
|       (b) You must cause any modified files to carry prominent notices | ||||
|           stating that You changed the files; and | ||||
|  | ||||
|       (c) You must retain, in the Source form of any Derivative Works | ||||
|           that You distribute, all copyright, patent, trademark, and | ||||
|           attribution notices from the Source form of the Work, | ||||
|           excluding those notices that do not pertain to any part of | ||||
|           the Derivative Works; and | ||||
|  | ||||
|       (d) If the Work includes a "NOTICE" text file as part of its | ||||
|           distribution, then any Derivative Works that You distribute must | ||||
|           include a readable copy of the attribution notices contained | ||||
|           within such NOTICE file, excluding those notices that do not | ||||
|           pertain to any part of the Derivative Works, in at least one | ||||
|           of the following places: within a NOTICE text file distributed | ||||
|           as part of the Derivative Works; within the Source form or | ||||
|           documentation, if provided along with the Derivative Works; or, | ||||
|           within a display generated by the Derivative Works, if and | ||||
|           wherever such third-party notices normally appear. The contents | ||||
|           of the NOTICE file are for informational purposes only and | ||||
|           do not modify the License. You may add Your own attribution | ||||
|           notices within Derivative Works that You distribute, alongside | ||||
|           or as an addendum to the NOTICE text from the Work, provided | ||||
|           that such additional attribution notices cannot be construed | ||||
|           as modifying the License. | ||||
|  | ||||
|       You may add Your own copyright statement to Your modifications and | ||||
|       may provide additional or different license terms and conditions | ||||
|       for use, reproduction, or distribution of Your modifications, or | ||||
|       for any such Derivative Works as a whole, provided Your use, | ||||
|       reproduction, and distribution of the Work otherwise complies with | ||||
|       the conditions stated in this License. | ||||
|  | ||||
|    5. Submission of Contributions. Unless You explicitly state otherwise, | ||||
|       any Contribution intentionally submitted for inclusion in the Work | ||||
|       by You to the Licensor shall be under the terms and conditions of | ||||
|       this License, without any additional terms or conditions. | ||||
|       Notwithstanding the above, nothing herein shall supersede or modify | ||||
|       the terms of any separate license agreement you may have executed | ||||
|       with Licensor regarding such Contributions. | ||||
|  | ||||
|    6. Trademarks. This License does not grant permission to use the trade | ||||
|       names, trademarks, service marks, or product names of the Licensor, | ||||
|       except as required for reasonable and customary use in describing the | ||||
|       origin of the Work and reproducing the content of the NOTICE file. | ||||
|  | ||||
|    7. Disclaimer of Warranty. Unless required by applicable law or | ||||
|       agreed to in writing, Licensor provides the Work (and each | ||||
|       Contributor provides its Contributions) on an "AS IS" BASIS, | ||||
|       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | ||||
|       implied, including, without limitation, any warranties or conditions | ||||
|       of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A | ||||
|       PARTICULAR PURPOSE. You are solely responsible for determining the | ||||
|       appropriateness of using or redistributing the Work and assume any | ||||
|       risks associated with Your exercise of permissions under this License. | ||||
|  | ||||
|    8. Limitation of Liability. In no event and under no legal theory, | ||||
|       whether in tort (including negligence), contract, or otherwise, | ||||
|       unless required by applicable law (such as deliberate and grossly | ||||
|       negligent acts) or agreed to in writing, shall any Contributor be | ||||
|       liable to You for damages, including any direct, indirect, special, | ||||
|       incidental, or consequential damages of any character arising as a | ||||
|       result of this License or out of the use or inability to use the | ||||
|       Work (including but not limited to damages for loss of goodwill, | ||||
|       work stoppage, computer failure or malfunction, or any and all | ||||
|       other commercial damages or losses), even if such Contributor | ||||
|       has been advised of the possibility of such damages. | ||||
|  | ||||
|    9. Accepting Warranty or Additional Liability. While redistributing | ||||
|       the Work or Derivative Works thereof, You may choose to offer, | ||||
|       and charge a fee for, acceptance of support, warranty, indemnity, | ||||
|       or other liability obligations and/or rights consistent with this | ||||
|       License. However, in accepting such obligations, You may act only | ||||
|       on Your own behalf and on Your sole responsibility, not on behalf | ||||
|       of any other Contributor, and only if You agree to indemnify, | ||||
|       defend, and hold each Contributor harmless for any liability | ||||
|       incurred by, or claims asserted against, such Contributor by reason | ||||
|       of your accepting any such warranty or additional liability. | ||||
|  | ||||
|    END OF TERMS AND CONDITIONS | ||||
|  | ||||
|    APPENDIX: How to apply the Apache License to your work. | ||||
|  | ||||
|       To apply the Apache License to your work, attach the following | ||||
|       boilerplate notice, with the fields enclosed by brackets "[]" | ||||
|       replaced with your own identifying information. (Don't include | ||||
|       the brackets!)  The text should be enclosed in the appropriate | ||||
|       comment syntax for the file format. We also recommend that a | ||||
|       file or class name and description of purpose be included on the | ||||
|       same "printed page" as the copyright notice for easier | ||||
|       identification within third-party archives. | ||||
|  | ||||
|    Copyright [yyyy] [name of copyright owner] | ||||
|  | ||||
|    Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|    you may not use this file except in compliance with the License. | ||||
|    You may obtain a copy of the License at | ||||
|  | ||||
|        http://www.apache.org/licenses/LICENSE-2.0 | ||||
|  | ||||
|    Unless required by applicable law or agreed to in writing, software | ||||
|    distributed under the License is distributed on an "AS IS" BASIS, | ||||
|    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|    See the License for the specific language governing permissions and | ||||
|    limitations under the License. | ||||
							
								
								
									
										91
									
								
								examples/langchain-python-rag-privategpt/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,91 @@ | ||||
| # PrivateGPT with Llama 2 uncensored | ||||
|  | ||||
| https://github.com/jmorganca/ollama/assets/3325447/20cf8ec6-ff25-42c6-bdd8-9be594e3ce1b | ||||
|  | ||||
| > Note: this example is a slightly modified version of PrivateGPT using models such as Llama 2 Uncensored. All credit for PrivateGPT goes to Iván Martínez who is the creator of it, and you can find his GitHub repo [here](https://github.com/imartinez/privateGPT). | ||||
|  | ||||
| ### Setup | ||||
|  | ||||
| Set up a virtual environment (optional): | ||||
|  | ||||
| ``` | ||||
| python3 -m venv .venv | ||||
| source .venv/bin/activate | ||||
| ``` | ||||
|  | ||||
| Install the Python dependencies: | ||||
|  | ||||
| ```shell | ||||
| pip install -r requirements.txt | ||||
| ``` | ||||
|  | ||||
| Pull the model you'd like to use: | ||||
|  | ||||
| ``` | ||||
| ollama pull llama2-uncensored | ||||
| ``` | ||||
|  | ||||
| ### Getting WeWork's latest quarterly earnings report (10-Q) | ||||
|  | ||||
| ``` | ||||
| mkdir source_documents | ||||
| curl https://d18rn0p25nwr6d.cloudfront.net/CIK-0001813756/975b3e9b-268e-4798-a9e4-2a9a7c92dc10.pdf -o source_documents/wework.pdf | ||||
| ``` | ||||
|  | ||||
| ### Ingesting files | ||||
|  | ||||
| ```shell | ||||
| python ingest.py | ||||
| ``` | ||||
|  | ||||
| Output should look like this: | ||||
|  | ||||
| ```shell | ||||
| Creating new vectorstore | ||||
| Loading documents from source_documents | ||||
| Loading new documents: 100%|██████████████████████| 1/1 [00:01<00:00,  1.73s/it] | ||||
| Loaded 1 new documents from source_documents | ||||
| Split into 90 chunks of text (max. 500 tokens each) | ||||
| Creating embeddings. May take some minutes... | ||||
| Using embedded DuckDB with persistence: data will be stored in: db | ||||
| Ingestion complete! You can now run privateGPT.py to query your documents | ||||
| ``` | ||||
|  | ||||
| ### Ask questions | ||||
|  | ||||
| ```shell | ||||
| python privateGPT.py | ||||
|  | ||||
| Enter a query: How many locations does WeWork have? | ||||
|  | ||||
| > Answer (took 17.7 s.): | ||||
| As of June 2023, WeWork has 777 locations worldwide, including 610 Consolidated Locations (as defined in the section entitled Key Performance Indicators). | ||||
| ``` | ||||
|  | ||||
| ### Try a different model: | ||||
|  | ||||
| ``` | ||||
| ollama pull llama2:13b | ||||
| MODEL=llama2:13b python privateGPT.py | ||||
| ``` | ||||
|  | ||||
| ## Adding more files | ||||
|  | ||||
| Put any and all your files into the `source_documents` directory | ||||
|  | ||||
| The supported extensions are: | ||||
|  | ||||
| - `.csv`: CSV, | ||||
| - `.docx`: Word Document, | ||||
| - `.doc`: Word Document, | ||||
| - `.enex`: EverNote, | ||||
| - `.eml`: Email, | ||||
| - `.epub`: EPub, | ||||
| - `.html`: HTML File, | ||||
| - `.md`: Markdown, | ||||
| - `.msg`: Outlook Message, | ||||
| - `.odt`: Open Document Text, | ||||
| - `.pdf`: Portable Document Format (PDF), | ||||
| - `.pptx` : PowerPoint Document, | ||||
| - `.ppt` : PowerPoint Document, | ||||
| - `.txt`: Text file (UTF-8), | ||||
							
								
								
									
										12
									
								
								examples/langchain-python-rag-privategpt/constants.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | ||||
| import os | ||||
| from chromadb.config import Settings | ||||
|  | ||||
| # Define the folder for storing database | ||||
| PERSIST_DIRECTORY = os.environ.get('PERSIST_DIRECTORY', 'db') | ||||
|  | ||||
| # Define the Chroma settings | ||||
| CHROMA_SETTINGS = Settings( | ||||
|         chroma_db_impl='duckdb+parquet', | ||||
|         persist_directory=PERSIST_DIRECTORY, | ||||
|         anonymized_telemetry=False | ||||
| ) | ||||
							
								
								
									
										161
									
								
								examples/langchain-python-rag-privategpt/ingest.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,161 @@ | ||||
| #!/usr/bin/env python3 | ||||
| import os | ||||
| import glob | ||||
| from typing import List | ||||
| from multiprocessing import Pool | ||||
| from tqdm import tqdm | ||||
|  | ||||
| from langchain.document_loaders import ( | ||||
|     CSVLoader, | ||||
|     EverNoteLoader, | ||||
|     PyMuPDFLoader, | ||||
|     TextLoader, | ||||
|     UnstructuredEmailLoader, | ||||
|     UnstructuredEPubLoader, | ||||
|     UnstructuredHTMLLoader, | ||||
|     UnstructuredMarkdownLoader, | ||||
|     UnstructuredODTLoader, | ||||
|     UnstructuredPowerPointLoader, | ||||
|     UnstructuredWordDocumentLoader, | ||||
| ) | ||||
|  | ||||
| from langchain.text_splitter import RecursiveCharacterTextSplitter | ||||
| from langchain.vectorstores import Chroma | ||||
| from langchain.embeddings import HuggingFaceEmbeddings | ||||
| from langchain.docstore.document import Document | ||||
| from constants import CHROMA_SETTINGS | ||||
|  | ||||
|  | ||||
| # Load environment variables | ||||
| persist_directory = os.environ.get('PERSIST_DIRECTORY', 'db') | ||||
| source_directory = os.environ.get('SOURCE_DIRECTORY', 'source_documents') | ||||
| embeddings_model_name = os.environ.get('EMBEDDINGS_MODEL_NAME', 'all-MiniLM-L6-v2') | ||||
| chunk_size = 500 | ||||
| chunk_overlap = 50 | ||||
|  | ||||
| # Custom document loaders | ||||
| class MyElmLoader(UnstructuredEmailLoader): | ||||
|     """Wrapper to fallback to text/plain when default does not work""" | ||||
|  | ||||
|     def load(self) -> List[Document]: | ||||
|         """Wrapper adding fallback for elm without html""" | ||||
|         try: | ||||
|             try: | ||||
|                 doc = UnstructuredEmailLoader.load(self) | ||||
|             except ValueError as e: | ||||
|                 if 'text/html content not found in email' in str(e): | ||||
|                     # Try plain text | ||||
|                     self.unstructured_kwargs["content_source"]="text/plain" | ||||
|                     doc = UnstructuredEmailLoader.load(self) | ||||
|                 else: | ||||
|                     raise | ||||
|         except Exception as e: | ||||
|             # Add file_path to exception message | ||||
|             raise type(e)(f"{self.file_path}: {e}") from e | ||||
|  | ||||
|         return doc | ||||
|  | ||||
|  | ||||
| # Map file extensions to document loaders and their arguments | ||||
| LOADER_MAPPING = { | ||||
|     ".csv": (CSVLoader, {}), | ||||
|     # ".docx": (Docx2txtLoader, {}), | ||||
|     ".doc": (UnstructuredWordDocumentLoader, {}), | ||||
|     ".docx": (UnstructuredWordDocumentLoader, {}), | ||||
|     ".enex": (EverNoteLoader, {}), | ||||
|     ".eml": (MyElmLoader, {}), | ||||
|     ".epub": (UnstructuredEPubLoader, {}), | ||||
|     ".html": (UnstructuredHTMLLoader, {}), | ||||
|     ".md": (UnstructuredMarkdownLoader, {}), | ||||
|     ".odt": (UnstructuredODTLoader, {}), | ||||
|     ".pdf": (PyMuPDFLoader, {}), | ||||
|     ".ppt": (UnstructuredPowerPointLoader, {}), | ||||
|     ".pptx": (UnstructuredPowerPointLoader, {}), | ||||
|     ".txt": (TextLoader, {"encoding": "utf8"}), | ||||
|     # Add more mappings for other file extensions and loaders as needed | ||||
| } | ||||
|  | ||||
|  | ||||
| def load_single_document(file_path: str) -> List[Document]: | ||||
|     ext = "." + file_path.rsplit(".", 1)[-1] | ||||
|     if ext in LOADER_MAPPING: | ||||
|         loader_class, loader_args = LOADER_MAPPING[ext] | ||||
|         loader = loader_class(file_path, **loader_args) | ||||
|         return loader.load() | ||||
|  | ||||
|     raise ValueError(f"Unsupported file extension '{ext}'") | ||||
|  | ||||
| def load_documents(source_dir: str, ignored_files: List[str] = []) -> List[Document]: | ||||
|     """ | ||||
|     Loads all documents from the source documents directory, ignoring specified files | ||||
|     """ | ||||
|     all_files = [] | ||||
|     for ext in LOADER_MAPPING: | ||||
|         all_files.extend( | ||||
|             glob.glob(os.path.join(source_dir, f"**/*{ext}"), recursive=True) | ||||
|         ) | ||||
|     filtered_files = [file_path for file_path in all_files if file_path not in ignored_files] | ||||
|  | ||||
|     with Pool(processes=os.cpu_count()) as pool: | ||||
|         results = [] | ||||
|         with tqdm(total=len(filtered_files), desc='Loading new documents', ncols=80) as pbar: | ||||
|             for i, docs in enumerate(pool.imap_unordered(load_single_document, filtered_files)): | ||||
|                 results.extend(docs) | ||||
|                 pbar.update() | ||||
|  | ||||
|     return results | ||||
|  | ||||
| def process_documents(ignored_files: List[str] = []) -> List[Document]: | ||||
|     """ | ||||
|     Load documents and split in chunks | ||||
|     """ | ||||
|     print(f"Loading documents from {source_directory}") | ||||
|     documents = load_documents(source_directory, ignored_files) | ||||
|     if not documents: | ||||
|         print("No new documents to load") | ||||
|         exit(0) | ||||
|     print(f"Loaded {len(documents)} new documents from {source_directory}") | ||||
|     text_splitter = RecursiveCharacterTextSplitter(chunk_size=chunk_size, chunk_overlap=chunk_overlap) | ||||
|     texts = text_splitter.split_documents(documents) | ||||
|     print(f"Split into {len(texts)} chunks of text (max. {chunk_size} tokens each)") | ||||
|     return texts | ||||
|  | ||||
| def does_vectorstore_exist(persist_directory: str) -> bool: | ||||
|     """ | ||||
|     Checks if vectorstore exists | ||||
|     """ | ||||
|     if os.path.exists(os.path.join(persist_directory, 'index')): | ||||
|         if os.path.exists(os.path.join(persist_directory, 'chroma-collections.parquet')) and os.path.exists(os.path.join(persist_directory, 'chroma-embeddings.parquet')): | ||||
|             list_index_files = glob.glob(os.path.join(persist_directory, 'index/*.bin')) | ||||
|             list_index_files += glob.glob(os.path.join(persist_directory, 'index/*.pkl')) | ||||
|             # At least 3 documents are needed in a working vectorstore | ||||
|             if len(list_index_files) > 3: | ||||
|                 return True | ||||
|     return False | ||||
|  | ||||
| def main(): | ||||
|     # Create embeddings | ||||
|     embeddings = HuggingFaceEmbeddings(model_name=embeddings_model_name) | ||||
|  | ||||
|     if does_vectorstore_exist(persist_directory): | ||||
|         # Update and store locally vectorstore | ||||
|         print(f"Appending to existing vectorstore at {persist_directory}") | ||||
|         db = Chroma(persist_directory=persist_directory, embedding_function=embeddings, client_settings=CHROMA_SETTINGS) | ||||
|         collection = db.get() | ||||
|         texts = process_documents([metadata['source'] for metadata in collection['metadatas']]) | ||||
|         print(f"Creating embeddings. May take some minutes...") | ||||
|         db.add_documents(texts) | ||||
|     else: | ||||
|         # Create and store locally vectorstore | ||||
|         print("Creating new vectorstore") | ||||
|         texts = process_documents() | ||||
|         print(f"Creating embeddings. May take some minutes...") | ||||
|         db = Chroma.from_documents(texts, embeddings, persist_directory=persist_directory, client_settings=CHROMA_SETTINGS) | ||||
|     db.persist() | ||||
|     db = None | ||||
|  | ||||
|     print(f"Ingestion complete! You can now run privateGPT.py to query your documents") | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										3833
									
								
								examples/langchain-python-rag-privategpt/poetry.lock
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										71
									
								
								examples/langchain-python-rag-privategpt/privateGPT.py
									
									
									
									
									
										Executable file
									
								
							
							
						
						| @@ -0,0 +1,71 @@ | ||||
| #!/usr/bin/env python3 | ||||
| from langchain.chains import RetrievalQA | ||||
| from langchain.embeddings import HuggingFaceEmbeddings | ||||
| from langchain.callbacks.streaming_stdout import StreamingStdOutCallbackHandler | ||||
| from langchain.vectorstores import Chroma | ||||
| from langchain.llms import Ollama | ||||
| import os | ||||
| import argparse | ||||
| import time | ||||
|  | ||||
| model = os.environ.get("MODEL", "llama2-uncensored") | ||||
| # For embeddings model, the example uses a sentence-transformers model | ||||
| # https://www.sbert.net/docs/pretrained_models.html  | ||||
| # "The all-mpnet-base-v2 model provides the best quality, while all-MiniLM-L6-v2 is 5 times faster and still offers good quality." | ||||
| embeddings_model_name = os.environ.get("EMBEDDINGS_MODEL_NAME", "all-MiniLM-L6-v2") | ||||
| persist_directory = os.environ.get("PERSIST_DIRECTORY", "db") | ||||
| target_source_chunks = int(os.environ.get('TARGET_SOURCE_CHUNKS',4)) | ||||
|  | ||||
| from constants import CHROMA_SETTINGS | ||||
|  | ||||
| def main(): | ||||
|     # Parse the command line arguments | ||||
|     args = parse_arguments() | ||||
|     embeddings = HuggingFaceEmbeddings(model_name=embeddings_model_name) | ||||
|     db = Chroma(persist_directory=persist_directory, embedding_function=embeddings, client_settings=CHROMA_SETTINGS) | ||||
|     retriever = db.as_retriever(search_kwargs={"k": target_source_chunks}) | ||||
|     # activate/deactivate the streaming StdOut callback for LLMs | ||||
|     callbacks = [] if args.mute_stream else [StreamingStdOutCallbackHandler()] | ||||
|  | ||||
|     llm = Ollama(model=model, callbacks=callbacks) | ||||
|  | ||||
|     qa = RetrievalQA.from_chain_type(llm=llm, chain_type="stuff", retriever=retriever, return_source_documents= not args.hide_source) | ||||
|     # Interactive questions and answers | ||||
|     while True: | ||||
|         query = input("\nEnter a query: ") | ||||
|         if query == "exit": | ||||
|             break | ||||
|         if query.strip() == "": | ||||
|             continue | ||||
|  | ||||
|         # Get the answer from the chain | ||||
|         start = time.time() | ||||
|         res = qa(query) | ||||
|         answer, docs = res['result'], [] if args.hide_source else res['source_documents'] | ||||
|         end = time.time() | ||||
|  | ||||
|         # Print the result | ||||
|         print("\n\n> Question:") | ||||
|         print(query) | ||||
|         print(answer) | ||||
|  | ||||
|         # Print the relevant sources used for the answer | ||||
|         for document in docs: | ||||
|             print("\n> " + document.metadata["source"] + ":") | ||||
|             print(document.page_content) | ||||
|  | ||||
| def parse_arguments(): | ||||
|     parser = argparse.ArgumentParser(description='privateGPT: Ask questions to your documents without an internet connection, ' | ||||
|                                                  'using the power of LLMs.') | ||||
|     parser.add_argument("--hide-source", "-S", action='store_true', | ||||
|                         help='Use this flag to disable printing of source documents used for answers.') | ||||
|  | ||||
|     parser.add_argument("--mute-stream", "-M", | ||||
|                         action='store_true', | ||||
|                         help='Use this flag to disable the streaming StdOut callback for LLMs.') | ||||
|  | ||||
|     return parser.parse_args() | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										26
									
								
								examples/langchain-python-rag-privategpt/pyproject.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | ||||
| [tool.poetry] | ||||
| name = "privategpt" | ||||
| version = "0.1.0" | ||||
| description = "" | ||||
| authors = ["Ivan Martinez <ivanmartit@gmail.com>"] | ||||
| license = "Apache Version 2.0" | ||||
| readme = "README.md" | ||||
|  | ||||
| [tool.poetry.dependencies] | ||||
| python = "^3.10" | ||||
| langchain = "0.0.261" | ||||
| gpt4all = "^1.0.3" | ||||
| chromadb = "^0.3.26" | ||||
| PyMuPDF = "^1.22.5" | ||||
| python-dotenv = "^1.0.0" | ||||
| unstructured = "^0.8.0" | ||||
| extract-msg = "^0.41.5" | ||||
| tabulate = "^0.9.0" | ||||
| pandoc = "^2.3" | ||||
| pypandoc = "^1.11" | ||||
| tqdm = "^4.65.0" | ||||
| sentence-transformers = "^2.2.2" | ||||
|  | ||||
| [build-system] | ||||
| requires = ["poetry-core"] | ||||
| build-backend = "poetry.core.masonry.api" | ||||
							
								
								
									
										2002
									
								
								examples/langchain-python-rag-privategpt/requirements.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
							
								
								
									
										15
									
								
								examples/langchain-python-rag-websummary/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| # LangChain Web Summarization | ||||
|  | ||||
| This example summarizes a website | ||||
|  | ||||
| ## Setup | ||||
|  | ||||
| ``` | ||||
| pip install -r requirements.txt | ||||
| ``` | ||||
|  | ||||
| ## Run | ||||
|  | ||||
| ``` | ||||
| python main.py | ||||
| ``` | ||||
							
								
								
									
										12
									
								
								examples/langchain-python-rag-websummary/main.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | ||||
| from langchain.llms import Ollama | ||||
| from langchain.document_loaders import WebBaseLoader | ||||
| from langchain.chains.summarize import load_summarize_chain | ||||
|  | ||||
| loader = WebBaseLoader("https://ollama.ai/blog/run-llama2-uncensored-locally") | ||||
| docs = loader.load() | ||||
|  | ||||
| llm = Ollama(model="llama2") | ||||
| chain = load_summarize_chain(llm, chain_type="stuff") | ||||
|  | ||||
| result = chain.run(docs) | ||||
| print(result) | ||||
| @@ -0,0 +1,2 @@ | ||||
| langchain==0.0.259 | ||||
| bs4==0.0.1 | ||||
							
								
								
									
										21
									
								
								examples/langchain-python-simple/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| # LangChain | ||||
|  | ||||
| This example is a basic "hello world" of using LangChain with Ollama. | ||||
|  | ||||
| ## Setup | ||||
|  | ||||
| ``` | ||||
| pip install -r requirements.txt | ||||
| ``` | ||||
|  | ||||
| ## Run | ||||
|  | ||||
| ``` | ||||
| python main.py | ||||
| ``` | ||||
|  | ||||
| Running this example will print the response for "hello": | ||||
|  | ||||
| ``` | ||||
| Hello! It's nice to meet you. hopefully you are having a great day! Is there something I can help you with or would you like to chat? | ||||
| ``` | ||||
							
								
								
									
										4
									
								
								examples/langchain-python-simple/main.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,4 @@ | ||||
| from langchain.llms import Ollama | ||||
| llm = Ollama(model="llama2") | ||||
| res = llm.predict("hello") | ||||
| print (res) | ||||
							
								
								
									
										1
									
								
								examples/langchain-python-simple/requirements.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| langchain==0.0.259 | ||||
							
								
								
									
										21
									
								
								examples/langchain-typescript-simple/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| # LangChain | ||||
|  | ||||
| This example is a basic "hello world" of using LangChain with Ollama using Node.js and Typescript. | ||||
|  | ||||
| ## Setup | ||||
|  | ||||
| ```shell | ||||
| npm install | ||||
| ``` | ||||
|  | ||||
| ## Run | ||||
|  | ||||
| ```shell | ||||
| ts-node main.ts | ||||
| ``` | ||||
|  | ||||
| Running this example will print the response for "hello": | ||||
|  | ||||
| ```plaintext | ||||
| Hello! It's nice to meet you. hopefully you are having a great day! Is there something I can help you with or would you like to chat? | ||||
| ``` | ||||
							
								
								
									
										15
									
								
								examples/langchain-typescript-simple/main.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| import { Ollama} from 'langchain/llms/ollama'; | ||||
|  | ||||
| async function main() { | ||||
|   const ollama = new Ollama({ | ||||
|     model: 'mistral'     | ||||
|     // other parameters can be found at https://js.langchain.com/docs/api/llms_ollama/classes/Ollama | ||||
|   }) | ||||
|   const stream = await ollama.stream("Hello"); | ||||
|  | ||||
|   for await (const chunk of stream) { | ||||
|     process.stdout.write(chunk); | ||||
|   } | ||||
| } | ||||
|  | ||||
| main(); | ||||
							
								
								
									
										997
									
								
								examples/langchain-typescript-simple/package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,997 @@ | ||||
| { | ||||
|   "name": "with-langchain-typescript-simplegenerate", | ||||
|   "lockfileVersion": 3, | ||||
|   "requires": true, | ||||
|   "packages": { | ||||
|     "": { | ||||
|       "dependencies": { | ||||
|         "langchain": "^0.0.165" | ||||
|       }, | ||||
|       "devDependencies": { | ||||
|         "typescript": "^5.2.2" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@anthropic-ai/sdk": { | ||||
|       "version": "0.6.2", | ||||
|       "resolved": "https://registry.npmjs.org/@anthropic-ai/sdk/-/sdk-0.6.2.tgz", | ||||
|       "integrity": "sha512-fB9PUj9RFT+XjkL+E9Ol864ZIJi+1P8WnbHspN3N3/GK2uSzjd0cbVIKTGgf4v3N8MwaQu+UWnU7C4BG/fap/g==", | ||||
|       "dependencies": { | ||||
|         "@types/node": "^18.11.18", | ||||
|         "@types/node-fetch": "^2.6.4", | ||||
|         "abort-controller": "^3.0.0", | ||||
|         "agentkeepalive": "^4.2.1", | ||||
|         "digest-fetch": "^1.3.0", | ||||
|         "form-data-encoder": "1.7.2", | ||||
|         "formdata-node": "^4.3.2", | ||||
|         "node-fetch": "^2.6.7" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@types/node": { | ||||
|       "version": "18.18.4", | ||||
|       "resolved": "https://registry.npmjs.org/@types/node/-/node-18.18.4.tgz", | ||||
|       "integrity": "sha512-t3rNFBgJRugIhackit2mVcLfF6IRc0JE4oeizPQL8Zrm8n2WY/0wOdpOPhdtG0V9Q2TlW/axbF1MJ6z+Yj/kKQ==" | ||||
|     }, | ||||
|     "node_modules/@types/node-fetch": { | ||||
|       "version": "2.6.6", | ||||
|       "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.6.6.tgz", | ||||
|       "integrity": "sha512-95X8guJYhfqiuVVhRFxVQcf4hW/2bCuoPwDasMf/531STFoNoWTT7YDnWdXHEZKqAGUigmpG31r2FE70LwnzJw==", | ||||
|       "dependencies": { | ||||
|         "@types/node": "*", | ||||
|         "form-data": "^4.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/@types/retry": { | ||||
|       "version": "0.12.0", | ||||
|       "resolved": "https://registry.npmjs.org/@types/retry/-/retry-0.12.0.tgz", | ||||
|       "integrity": "sha512-wWKOClTTiizcZhXnPY4wikVAwmdYHp8q6DmC+EJUzAMsycb7HB32Kh9RN4+0gExjmPmZSAQjgURXIGATPegAvA==" | ||||
|     }, | ||||
|     "node_modules/@types/uuid": { | ||||
|       "version": "9.0.5", | ||||
|       "resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-9.0.5.tgz", | ||||
|       "integrity": "sha512-xfHdwa1FMJ082prjSJpoEI57GZITiQz10r3vEJCHa2khEFQjKy91aWKz6+zybzssCvXUwE1LQWgWVwZ4nYUvHQ==" | ||||
|     }, | ||||
|     "node_modules/abort-controller": { | ||||
|       "version": "3.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", | ||||
|       "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", | ||||
|       "dependencies": { | ||||
|         "event-target-shim": "^5.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=6.5" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/agentkeepalive": { | ||||
|       "version": "4.5.0", | ||||
|       "resolved": "https://registry.npmjs.org/agentkeepalive/-/agentkeepalive-4.5.0.tgz", | ||||
|       "integrity": "sha512-5GG/5IbQQpC9FpkRGsSvZI5QYeSCzlJHdpBQntCsuTOxhKD8lqKhrleg2Yi7yvMIf82Ycmmqln9U8V9qwEiJew==", | ||||
|       "dependencies": { | ||||
|         "humanize-ms": "^1.2.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">= 8.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/ansi-styles": { | ||||
|       "version": "5.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", | ||||
|       "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/chalk/ansi-styles?sponsor=1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/argparse": { | ||||
|       "version": "2.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", | ||||
|       "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==" | ||||
|     }, | ||||
|     "node_modules/asynckit": { | ||||
|       "version": "0.4.0", | ||||
|       "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", | ||||
|       "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==" | ||||
|     }, | ||||
|     "node_modules/base-64": { | ||||
|       "version": "0.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/base-64/-/base-64-0.1.0.tgz", | ||||
|       "integrity": "sha512-Y5gU45svrR5tI2Vt/X9GPd3L0HNIKzGu202EjxrXMpuc2V2CiKgemAbUUsqYmZJvPtCXoUKjNZwBJzsNScUbXA==" | ||||
|     }, | ||||
|     "node_modules/base64-js": { | ||||
|       "version": "1.5.1", | ||||
|       "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", | ||||
|       "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", | ||||
|       "funding": [ | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://github.com/sponsors/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "patreon", | ||||
|           "url": "https://www.patreon.com/feross" | ||||
|         }, | ||||
|         { | ||||
|           "type": "consulting", | ||||
|           "url": "https://feross.org/support" | ||||
|         } | ||||
|       ] | ||||
|     }, | ||||
|     "node_modules/binary-extensions": { | ||||
|       "version": "2.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", | ||||
|       "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/binary-search": { | ||||
|       "version": "1.3.6", | ||||
|       "resolved": "https://registry.npmjs.org/binary-search/-/binary-search-1.3.6.tgz", | ||||
|       "integrity": "sha512-nbE1WxOTTrUWIfsfZ4aHGYu5DOuNkbxGokjV6Z2kxfJK3uaAb8zNK1muzOeipoLHZjInT4Br88BHpzevc681xA==" | ||||
|     }, | ||||
|     "node_modules/camelcase": { | ||||
|       "version": "6.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", | ||||
|       "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", | ||||
|       "engines": { | ||||
|         "node": ">=10" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/sponsors/sindresorhus" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/charenc": { | ||||
|       "version": "0.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/charenc/-/charenc-0.0.2.tgz", | ||||
|       "integrity": "sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA==", | ||||
|       "engines": { | ||||
|         "node": "*" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/combined-stream": { | ||||
|       "version": "1.0.8", | ||||
|       "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", | ||||
|       "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", | ||||
|       "dependencies": { | ||||
|         "delayed-stream": "~1.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">= 0.8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/commander": { | ||||
|       "version": "10.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/commander/-/commander-10.0.1.tgz", | ||||
|       "integrity": "sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug==", | ||||
|       "engines": { | ||||
|         "node": ">=14" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/crypt": { | ||||
|       "version": "0.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/crypt/-/crypt-0.0.2.tgz", | ||||
|       "integrity": "sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow==", | ||||
|       "engines": { | ||||
|         "node": "*" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/decamelize": { | ||||
|       "version": "1.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", | ||||
|       "integrity": "sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA==", | ||||
|       "engines": { | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/delayed-stream": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", | ||||
|       "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", | ||||
|       "engines": { | ||||
|         "node": ">=0.4.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/digest-fetch": { | ||||
|       "version": "1.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/digest-fetch/-/digest-fetch-1.3.0.tgz", | ||||
|       "integrity": "sha512-CGJuv6iKNM7QyZlM2T3sPAdZWd/p9zQiRNS9G+9COUCwzWFTs0Xp8NF5iePx7wtvhDykReiRRrSeNb4oMmB8lA==", | ||||
|       "dependencies": { | ||||
|         "base-64": "^0.1.0", | ||||
|         "md5": "^2.3.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/event-target-shim": { | ||||
|       "version": "5.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", | ||||
|       "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==", | ||||
|       "engines": { | ||||
|         "node": ">=6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/eventemitter3": { | ||||
|       "version": "4.0.7", | ||||
|       "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz", | ||||
|       "integrity": "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==" | ||||
|     }, | ||||
|     "node_modules/expr-eval": { | ||||
|       "version": "2.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/expr-eval/-/expr-eval-2.0.2.tgz", | ||||
|       "integrity": "sha512-4EMSHGOPSwAfBiibw3ndnP0AvjDWLsMvGOvWEZ2F96IGk0bIVdjQisOHxReSkE13mHcfbuCiXw+G4y0zv6N8Eg==" | ||||
|     }, | ||||
|     "node_modules/flat": { | ||||
|       "version": "5.0.2", | ||||
|       "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz", | ||||
|       "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==", | ||||
|       "bin": { | ||||
|         "flat": "cli.js" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/form-data": { | ||||
|       "version": "4.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", | ||||
|       "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", | ||||
|       "dependencies": { | ||||
|         "asynckit": "^0.4.0", | ||||
|         "combined-stream": "^1.0.8", | ||||
|         "mime-types": "^2.1.12" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">= 6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/form-data-encoder": { | ||||
|       "version": "1.7.2", | ||||
|       "resolved": "https://registry.npmjs.org/form-data-encoder/-/form-data-encoder-1.7.2.tgz", | ||||
|       "integrity": "sha512-qfqtYan3rxrnCk1VYaA4H+Ms9xdpPqvLZa6xmMgFvhO32x7/3J/ExcTd6qpxM0vH2GdMI+poehyBZvqfMTto8A==" | ||||
|     }, | ||||
|     "node_modules/formdata-node": { | ||||
|       "version": "4.4.1", | ||||
|       "resolved": "https://registry.npmjs.org/formdata-node/-/formdata-node-4.4.1.tgz", | ||||
|       "integrity": "sha512-0iirZp3uVDjVGt9p49aTaqjk84TrglENEDuqfdlZQ1roC9CWlPk6Avf8EEnZNcAqPonwkG35x4n3ww/1THYAeQ==", | ||||
|       "dependencies": { | ||||
|         "node-domexception": "1.0.0", | ||||
|         "web-streams-polyfill": "4.0.0-beta.3" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">= 12.20" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/humanize-ms": { | ||||
|       "version": "1.2.1", | ||||
|       "resolved": "https://registry.npmjs.org/humanize-ms/-/humanize-ms-1.2.1.tgz", | ||||
|       "integrity": "sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ==", | ||||
|       "dependencies": { | ||||
|         "ms": "^2.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/is-any-array": { | ||||
|       "version": "2.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/is-any-array/-/is-any-array-2.0.1.tgz", | ||||
|       "integrity": "sha512-UtilS7hLRu++wb/WBAw9bNuP1Eg04Ivn1vERJck8zJthEvXCBEBpGR/33u/xLKWEQf95803oalHrVDptcAvFdQ==" | ||||
|     }, | ||||
|     "node_modules/is-buffer": { | ||||
|       "version": "1.1.6", | ||||
|       "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", | ||||
|       "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==" | ||||
|     }, | ||||
|     "node_modules/js-tiktoken": { | ||||
|       "version": "1.0.7", | ||||
|       "resolved": "https://registry.npmjs.org/js-tiktoken/-/js-tiktoken-1.0.7.tgz", | ||||
|       "integrity": "sha512-biba8u/clw7iesNEWLOLwrNGoBP2lA+hTaBLs/D45pJdUPFXyxD6nhcDVtADChghv4GgyAiMKYMiRx7x6h7Biw==", | ||||
|       "dependencies": { | ||||
|         "base64-js": "^1.5.1" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/js-yaml": { | ||||
|       "version": "4.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", | ||||
|       "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", | ||||
|       "dependencies": { | ||||
|         "argparse": "^2.0.1" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "js-yaml": "bin/js-yaml.js" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/jsonpointer": { | ||||
|       "version": "5.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/jsonpointer/-/jsonpointer-5.0.1.tgz", | ||||
|       "integrity": "sha512-p/nXbhSEcu3pZRdkW1OfJhpsVtW1gd4Wa1fnQc9YLiTfAjn0312eMKimbdIQzuZl9aa9xUGaRlP9T/CJE/ditQ==", | ||||
|       "engines": { | ||||
|         "node": ">=0.10.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/langchain": { | ||||
|       "version": "0.0.165", | ||||
|       "resolved": "https://registry.npmjs.org/langchain/-/langchain-0.0.165.tgz", | ||||
|       "integrity": "sha512-CpbNpjwaE+9lzjdw+pZz0VgnRrFivEgr7CVp9dDaAb5JpaJAA4V2v6uQ9ZPN+TSqupTQ79HFn2sfyZVEl2EG7Q==", | ||||
|       "dependencies": { | ||||
|         "@anthropic-ai/sdk": "^0.6.2", | ||||
|         "ansi-styles": "^5.0.0", | ||||
|         "binary-extensions": "^2.2.0", | ||||
|         "camelcase": "6", | ||||
|         "decamelize": "^1.2.0", | ||||
|         "expr-eval": "^2.0.2", | ||||
|         "flat": "^5.0.2", | ||||
|         "js-tiktoken": "^1.0.7", | ||||
|         "js-yaml": "^4.1.0", | ||||
|         "jsonpointer": "^5.0.1", | ||||
|         "langchainhub": "~0.0.6", | ||||
|         "langsmith": "~0.0.31", | ||||
|         "ml-distance": "^4.0.0", | ||||
|         "object-hash": "^3.0.0", | ||||
|         "openai": "~4.4.0", | ||||
|         "openapi-types": "^12.1.3", | ||||
|         "p-queue": "^6.6.2", | ||||
|         "p-retry": "4", | ||||
|         "uuid": "^9.0.0", | ||||
|         "yaml": "^2.2.1", | ||||
|         "zod": "^3.22.3", | ||||
|         "zod-to-json-schema": "^3.20.4" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=18" | ||||
|       }, | ||||
|       "peerDependencies": { | ||||
|         "@aws-crypto/sha256-js": "^5.0.0", | ||||
|         "@aws-sdk/client-bedrock-runtime": "^3.422.0", | ||||
|         "@aws-sdk/client-dynamodb": "^3.310.0", | ||||
|         "@aws-sdk/client-kendra": "^3.352.0", | ||||
|         "@aws-sdk/client-lambda": "^3.310.0", | ||||
|         "@aws-sdk/client-s3": "^3.310.0", | ||||
|         "@aws-sdk/client-sagemaker-runtime": "^3.310.0", | ||||
|         "@aws-sdk/client-sfn": "^3.310.0", | ||||
|         "@aws-sdk/credential-provider-node": "^3.388.0", | ||||
|         "@azure/storage-blob": "^12.15.0", | ||||
|         "@clickhouse/client": "^0.0.14", | ||||
|         "@cloudflare/ai": "^1.0.12", | ||||
|         "@elastic/elasticsearch": "^8.4.0", | ||||
|         "@getmetal/metal-sdk": "*", | ||||
|         "@getzep/zep-js": "^0.7.0", | ||||
|         "@gomomento/sdk": "^1.23.0", | ||||
|         "@google-ai/generativelanguage": "^0.2.1", | ||||
|         "@google-cloud/storage": "^6.10.1", | ||||
|         "@huggingface/inference": "^1.5.1", | ||||
|         "@mozilla/readability": "*", | ||||
|         "@notionhq/client": "^2.2.10", | ||||
|         "@opensearch-project/opensearch": "*", | ||||
|         "@pinecone-database/pinecone": "^1.1.0", | ||||
|         "@planetscale/database": "^1.8.0", | ||||
|         "@qdrant/js-client-rest": "^1.2.0", | ||||
|         "@raycast/api": "^1.55.2", | ||||
|         "@smithy/eventstream-codec": "^2.0.5", | ||||
|         "@smithy/protocol-http": "^3.0.6", | ||||
|         "@smithy/signature-v4": "^2.0.10", | ||||
|         "@smithy/util-utf8": "^2.0.0", | ||||
|         "@supabase/postgrest-js": "^1.1.1", | ||||
|         "@supabase/supabase-js": "^2.10.0", | ||||
|         "@tensorflow-models/universal-sentence-encoder": "*", | ||||
|         "@tensorflow/tfjs-converter": "*", | ||||
|         "@tensorflow/tfjs-core": "*", | ||||
|         "@upstash/redis": "^1.20.6", | ||||
|         "@vercel/postgres": "^0.5.0", | ||||
|         "@writerai/writer-sdk": "^0.40.2", | ||||
|         "@xata.io/client": "^0.25.1", | ||||
|         "@xenova/transformers": "^2.5.4", | ||||
|         "@zilliz/milvus2-sdk-node": ">=2.2.7", | ||||
|         "apify-client": "^2.7.1", | ||||
|         "axios": "*", | ||||
|         "cassandra-driver": "^4.6.4", | ||||
|         "cheerio": "^1.0.0-rc.12", | ||||
|         "chromadb": "*", | ||||
|         "cohere-ai": ">=6.0.0", | ||||
|         "d3-dsv": "^2.0.0", | ||||
|         "epub2": "^3.0.1", | ||||
|         "faiss-node": "^0.3.0", | ||||
|         "fast-xml-parser": "^4.2.7", | ||||
|         "firebase-admin": "^11.9.0", | ||||
|         "google-auth-library": "^8.9.0", | ||||
|         "googleapis": "^126.0.1", | ||||
|         "hnswlib-node": "^1.4.2", | ||||
|         "html-to-text": "^9.0.5", | ||||
|         "ignore": "^5.2.0", | ||||
|         "ioredis": "^5.3.2", | ||||
|         "jsdom": "*", | ||||
|         "llmonitor": "*", | ||||
|         "lodash": "^4.17.21", | ||||
|         "mammoth": "*", | ||||
|         "mongodb": "^5.2.0", | ||||
|         "mysql2": "^3.3.3", | ||||
|         "neo4j-driver": "*", | ||||
|         "node-llama-cpp": "*", | ||||
|         "notion-to-md": "^3.1.0", | ||||
|         "pdf-parse": "1.1.1", | ||||
|         "peggy": "^3.0.2", | ||||
|         "pg": "^8.11.0", | ||||
|         "pg-copy-streams": "^6.0.5", | ||||
|         "pickleparser": "^0.1.0", | ||||
|         "playwright": "^1.32.1", | ||||
|         "portkey-ai": "^0.1.11", | ||||
|         "puppeteer": "^19.7.2", | ||||
|         "redis": "^4.6.4", | ||||
|         "replicate": "^0.18.0", | ||||
|         "sonix-speech-recognition": "^2.1.1", | ||||
|         "srt-parser-2": "^1.2.2", | ||||
|         "typeorm": "^0.3.12", | ||||
|         "typesense": "^1.5.3", | ||||
|         "usearch": "^1.1.1", | ||||
|         "vectordb": "^0.1.4", | ||||
|         "voy-search": "0.6.2", | ||||
|         "weaviate-ts-client": "^1.4.0", | ||||
|         "web-auth-library": "^1.0.3", | ||||
|         "youtube-transcript": "^1.0.6", | ||||
|         "youtubei.js": "^5.8.0" | ||||
|       }, | ||||
|       "peerDependenciesMeta": { | ||||
|         "@aws-crypto/sha256-js": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@aws-sdk/client-bedrock-runtime": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@aws-sdk/client-dynamodb": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@aws-sdk/client-kendra": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@aws-sdk/client-lambda": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@aws-sdk/client-s3": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@aws-sdk/client-sagemaker-runtime": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@aws-sdk/client-sfn": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@aws-sdk/credential-provider-node": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@azure/storage-blob": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@clickhouse/client": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@cloudflare/ai": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@elastic/elasticsearch": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@getmetal/metal-sdk": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@getzep/zep-js": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@gomomento/sdk": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@google-ai/generativelanguage": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@google-cloud/storage": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@huggingface/inference": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@mozilla/readability": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@notionhq/client": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@opensearch-project/opensearch": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@pinecone-database/pinecone": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@planetscale/database": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@qdrant/js-client-rest": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@raycast/api": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@smithy/eventstream-codec": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@smithy/protocol-http": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@smithy/signature-v4": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@smithy/util-utf8": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@supabase/postgrest-js": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@supabase/supabase-js": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@tensorflow-models/universal-sentence-encoder": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@tensorflow/tfjs-converter": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@tensorflow/tfjs-core": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@upstash/redis": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@vercel/postgres": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@writerai/writer-sdk": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@xata.io/client": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@xenova/transformers": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "@zilliz/milvus2-sdk-node": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "apify-client": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "axios": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "cassandra-driver": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "cheerio": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "chromadb": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "cohere-ai": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "d3-dsv": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "epub2": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "faiss-node": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "fast-xml-parser": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "firebase-admin": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "google-auth-library": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "googleapis": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "hnswlib-node": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "html-to-text": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "ignore": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "ioredis": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "jsdom": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "llmonitor": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "lodash": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "mammoth": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "mongodb": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "mysql2": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "neo4j-driver": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "node-llama-cpp": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "notion-to-md": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "pdf-parse": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "peggy": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "pg": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "pg-copy-streams": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "pickleparser": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "playwright": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "portkey-ai": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "puppeteer": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "redis": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "replicate": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "sonix-speech-recognition": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "srt-parser-2": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "typeorm": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "typesense": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "usearch": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "vectordb": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "voy-search": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "weaviate-ts-client": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "web-auth-library": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "youtube-transcript": { | ||||
|           "optional": true | ||||
|         }, | ||||
|         "youtubei.js": { | ||||
|           "optional": true | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/langchainhub": { | ||||
|       "version": "0.0.6", | ||||
|       "resolved": "https://registry.npmjs.org/langchainhub/-/langchainhub-0.0.6.tgz", | ||||
|       "integrity": "sha512-SW6105T+YP1cTe0yMf//7kyshCgvCTyFBMTgH2H3s9rTAR4e+78DA/BBrUL/Mt4Q5eMWui7iGuAYb3pgGsdQ9w==" | ||||
|     }, | ||||
|     "node_modules/langsmith": { | ||||
|       "version": "0.0.42", | ||||
|       "resolved": "https://registry.npmjs.org/langsmith/-/langsmith-0.0.42.tgz", | ||||
|       "integrity": "sha512-sFuN+e7E+pPBIRaRgFqZh/BRBWNHTZNAwi6uj4kydQawooCZYoJmM5snOkiQrhVSvAhgu6xFhLvmfvkPcKzD7w==", | ||||
|       "dependencies": { | ||||
|         "@types/uuid": "^9.0.1", | ||||
|         "commander": "^10.0.1", | ||||
|         "p-queue": "^6.6.2", | ||||
|         "p-retry": "4", | ||||
|         "uuid": "^9.0.0" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "langsmith": "dist/cli/main.cjs" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/md5": { | ||||
|       "version": "2.3.0", | ||||
|       "resolved": "https://registry.npmjs.org/md5/-/md5-2.3.0.tgz", | ||||
|       "integrity": "sha512-T1GITYmFaKuO91vxyoQMFETst+O71VUPEU3ze5GNzDm0OWdP8v1ziTaAEPUr/3kLsY3Sftgz242A1SetQiDL7g==", | ||||
|       "dependencies": { | ||||
|         "charenc": "0.0.2", | ||||
|         "crypt": "0.0.2", | ||||
|         "is-buffer": "~1.1.6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/mime-db": { | ||||
|       "version": "1.52.0", | ||||
|       "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", | ||||
|       "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", | ||||
|       "engines": { | ||||
|         "node": ">= 0.6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/mime-types": { | ||||
|       "version": "2.1.35", | ||||
|       "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", | ||||
|       "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", | ||||
|       "dependencies": { | ||||
|         "mime-db": "1.52.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">= 0.6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/ml-array-mean": { | ||||
|       "version": "1.1.6", | ||||
|       "resolved": "https://registry.npmjs.org/ml-array-mean/-/ml-array-mean-1.1.6.tgz", | ||||
|       "integrity": "sha512-MIdf7Zc8HznwIisyiJGRH9tRigg3Yf4FldW8DxKxpCCv/g5CafTw0RRu51nojVEOXuCQC7DRVVu5c7XXO/5joQ==", | ||||
|       "dependencies": { | ||||
|         "ml-array-sum": "^1.1.6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/ml-array-sum": { | ||||
|       "version": "1.1.6", | ||||
|       "resolved": "https://registry.npmjs.org/ml-array-sum/-/ml-array-sum-1.1.6.tgz", | ||||
|       "integrity": "sha512-29mAh2GwH7ZmiRnup4UyibQZB9+ZLyMShvt4cH4eTK+cL2oEMIZFnSyB3SS8MlsTh6q/w/yh48KmqLxmovN4Dw==", | ||||
|       "dependencies": { | ||||
|         "is-any-array": "^2.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/ml-distance": { | ||||
|       "version": "4.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/ml-distance/-/ml-distance-4.0.1.tgz", | ||||
|       "integrity": "sha512-feZ5ziXs01zhyFUUUeZV5hwc0f5JW0Sh0ckU1koZe/wdVkJdGxcP06KNQuF0WBTj8FttQUzcvQcpcrOp/XrlEw==", | ||||
|       "dependencies": { | ||||
|         "ml-array-mean": "^1.1.6", | ||||
|         "ml-distance-euclidean": "^2.0.0", | ||||
|         "ml-tree-similarity": "^1.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/ml-distance-euclidean": { | ||||
|       "version": "2.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/ml-distance-euclidean/-/ml-distance-euclidean-2.0.0.tgz", | ||||
|       "integrity": "sha512-yC9/2o8QF0A3m/0IXqCTXCzz2pNEzvmcE/9HFKOZGnTjatvBbsn4lWYJkxENkA4Ug2fnYl7PXQxnPi21sgMy/Q==" | ||||
|     }, | ||||
|     "node_modules/ml-tree-similarity": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/ml-tree-similarity/-/ml-tree-similarity-1.0.0.tgz", | ||||
|       "integrity": "sha512-XJUyYqjSuUQkNQHMscr6tcjldsOoAekxADTplt40QKfwW6nd++1wHWV9AArl0Zvw/TIHgNaZZNvr8QGvE8wLRg==", | ||||
|       "dependencies": { | ||||
|         "binary-search": "^1.3.5", | ||||
|         "num-sort": "^2.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/ms": { | ||||
|       "version": "2.1.3", | ||||
|       "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", | ||||
|       "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" | ||||
|     }, | ||||
|     "node_modules/node-domexception": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz", | ||||
|       "integrity": "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==", | ||||
|       "funding": [ | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://github.com/sponsors/jimmywarting" | ||||
|         }, | ||||
|         { | ||||
|           "type": "github", | ||||
|           "url": "https://paypal.me/jimmywarting" | ||||
|         } | ||||
|       ], | ||||
|       "engines": { | ||||
|         "node": ">=10.5.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/node-fetch": { | ||||
|       "version": "2.7.0", | ||||
|       "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", | ||||
|       "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==", | ||||
|       "dependencies": { | ||||
|         "whatwg-url": "^5.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": "4.x || >=6.0.0" | ||||
|       }, | ||||
|       "peerDependencies": { | ||||
|         "encoding": "^0.1.0" | ||||
|       }, | ||||
|       "peerDependenciesMeta": { | ||||
|         "encoding": { | ||||
|           "optional": true | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/num-sort": { | ||||
|       "version": "2.1.0", | ||||
|       "resolved": "https://registry.npmjs.org/num-sort/-/num-sort-2.1.0.tgz", | ||||
|       "integrity": "sha512-1MQz1Ed8z2yckoBeSfkQHHO9K1yDRxxtotKSJ9yvcTUUxSvfvzEq5GwBrjjHEpMlq/k5gvXdmJ1SbYxWtpNoVg==", | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/sponsors/sindresorhus" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/object-hash": { | ||||
|       "version": "3.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/object-hash/-/object-hash-3.0.0.tgz", | ||||
|       "integrity": "sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==", | ||||
|       "engines": { | ||||
|         "node": ">= 6" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/openai": { | ||||
|       "version": "4.4.0", | ||||
|       "resolved": "https://registry.npmjs.org/openai/-/openai-4.4.0.tgz", | ||||
|       "integrity": "sha512-JN0t628Kh95T0IrXl0HdBqnlJg+4Vq0Bnh55tio+dfCnyzHvMLiWyCM9m726MAJD2YkDU4/8RQB6rNbEq9ct2w==", | ||||
|       "dependencies": { | ||||
|         "@types/node": "^18.11.18", | ||||
|         "@types/node-fetch": "^2.6.4", | ||||
|         "abort-controller": "^3.0.0", | ||||
|         "agentkeepalive": "^4.2.1", | ||||
|         "digest-fetch": "^1.3.0", | ||||
|         "form-data-encoder": "1.7.2", | ||||
|         "formdata-node": "^4.3.2", | ||||
|         "node-fetch": "^2.6.7" | ||||
|       }, | ||||
|       "bin": { | ||||
|         "openai": "bin/cli" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/openapi-types": { | ||||
|       "version": "12.1.3", | ||||
|       "resolved": "https://registry.npmjs.org/openapi-types/-/openapi-types-12.1.3.tgz", | ||||
|       "integrity": "sha512-N4YtSYJqghVu4iek2ZUvcN/0aqH1kRDuNqzcycDxhOUpg7GdvLa2F3DgS6yBNhInhv2r/6I0Flkn7CqL8+nIcw==" | ||||
|     }, | ||||
|     "node_modules/p-finally": { | ||||
|       "version": "1.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", | ||||
|       "integrity": "sha512-LICb2p9CB7FS+0eR1oqWnHhp0FljGLZCWBE9aix0Uye9W8LTQPwMTYVGWQWIw9RdQiDg4+epXQODwIYJtSJaow==", | ||||
|       "engines": { | ||||
|         "node": ">=4" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/p-queue": { | ||||
|       "version": "6.6.2", | ||||
|       "resolved": "https://registry.npmjs.org/p-queue/-/p-queue-6.6.2.tgz", | ||||
|       "integrity": "sha512-RwFpb72c/BhQLEXIZ5K2e+AhgNVmIejGlTgiB9MzZ0e93GRvqZ7uSi0dvRF7/XIXDeNkra2fNHBxTyPDGySpjQ==", | ||||
|       "dependencies": { | ||||
|         "eventemitter3": "^4.0.4", | ||||
|         "p-timeout": "^3.2.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       }, | ||||
|       "funding": { | ||||
|         "url": "https://github.com/sponsors/sindresorhus" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/p-retry": { | ||||
|       "version": "4.6.2", | ||||
|       "resolved": "https://registry.npmjs.org/p-retry/-/p-retry-4.6.2.tgz", | ||||
|       "integrity": "sha512-312Id396EbJdvRONlngUx0NydfrIQ5lsYu0znKVUzVvArzEIt08V1qhtyESbGVd1FGX7UKtiFp5uwKZdM8wIuQ==", | ||||
|       "dependencies": { | ||||
|         "@types/retry": "0.12.0", | ||||
|         "retry": "^0.13.1" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/p-timeout": { | ||||
|       "version": "3.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/p-timeout/-/p-timeout-3.2.0.tgz", | ||||
|       "integrity": "sha512-rhIwUycgwwKcP9yTOOFK/AKsAopjjCakVqLHePO3CC6Mir1Z99xT+R63jZxAT5lFZLa2inS5h+ZS2GvR99/FBg==", | ||||
|       "dependencies": { | ||||
|         "p-finally": "^1.0.0" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=8" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/retry": { | ||||
|       "version": "0.13.1", | ||||
|       "resolved": "https://registry.npmjs.org/retry/-/retry-0.13.1.tgz", | ||||
|       "integrity": "sha512-XQBQ3I8W1Cge0Seh+6gjj03LbmRFWuoszgK9ooCpwYIrhhoO80pfq4cUkU5DkknwfOfFteRwlZ56PYOGYyFWdg==", | ||||
|       "engines": { | ||||
|         "node": ">= 4" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/tr46": { | ||||
|       "version": "0.0.3", | ||||
|       "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", | ||||
|       "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==" | ||||
|     }, | ||||
|     "node_modules/typescript": { | ||||
|       "version": "5.2.2", | ||||
|       "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.2.2.tgz", | ||||
|       "integrity": "sha512-mI4WrpHsbCIcwT9cF4FZvr80QUeKvsUsUvKDoR+X/7XHQH98xYD8YHZg7ANtz2GtZt/CBq2QJ0thkGJMHfqc1w==", | ||||
|       "dev": true, | ||||
|       "bin": { | ||||
|         "tsc": "bin/tsc", | ||||
|         "tsserver": "bin/tsserver" | ||||
|       }, | ||||
|       "engines": { | ||||
|         "node": ">=14.17" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/uuid": { | ||||
|       "version": "9.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/uuid/-/uuid-9.0.1.tgz", | ||||
|       "integrity": "sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==", | ||||
|       "funding": [ | ||||
|         "https://github.com/sponsors/broofa", | ||||
|         "https://github.com/sponsors/ctavan" | ||||
|       ], | ||||
|       "bin": { | ||||
|         "uuid": "dist/bin/uuid" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/web-streams-polyfill": { | ||||
|       "version": "4.0.0-beta.3", | ||||
|       "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-4.0.0-beta.3.tgz", | ||||
|       "integrity": "sha512-QW95TCTaHmsYfHDybGMwO5IJIM93I/6vTRk+daHTWFPhwh+C8Cg7j7XyKrwrj8Ib6vYXe0ocYNrmzY4xAAN6ug==", | ||||
|       "engines": { | ||||
|         "node": ">= 14" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/webidl-conversions": { | ||||
|       "version": "3.0.1", | ||||
|       "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", | ||||
|       "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==" | ||||
|     }, | ||||
|     "node_modules/whatwg-url": { | ||||
|       "version": "5.0.0", | ||||
|       "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", | ||||
|       "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", | ||||
|       "dependencies": { | ||||
|         "tr46": "~0.0.3", | ||||
|         "webidl-conversions": "^3.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/yaml": { | ||||
|       "version": "2.3.2", | ||||
|       "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.3.2.tgz", | ||||
|       "integrity": "sha512-N/lyzTPaJasoDmfV7YTrYCI0G/3ivm/9wdG0aHuheKowWQwGTsK0Eoiw6utmzAnI6pkJa0DUVygvp3spqqEKXg==", | ||||
|       "engines": { | ||||
|         "node": ">= 14" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/zod": { | ||||
|       "version": "3.22.4", | ||||
|       "resolved": "https://registry.npmjs.org/zod/-/zod-3.22.4.tgz", | ||||
|       "integrity": "sha512-iC+8Io04lddc+mVqQ9AZ7OQ2MrUKGN+oIQyq1vemgt46jwCwLfhq7/pwnBnNXXXZb8VTVLKwp9EDkx+ryxIWmg==", | ||||
|       "funding": { | ||||
|         "url": "https://github.com/sponsors/colinhacks" | ||||
|       } | ||||
|     }, | ||||
|     "node_modules/zod-to-json-schema": { | ||||
|       "version": "3.21.4", | ||||
|       "resolved": "https://registry.npmjs.org/zod-to-json-schema/-/zod-to-json-schema-3.21.4.tgz", | ||||
|       "integrity": "sha512-fjUZh4nQ1s6HMccgIeE0VP4QG/YRGPmyjO9sAh890aQKPEk3nqbfUXhMFaC+Dr5KvYBm8BCyvfpZf2jY9aGSsw==", | ||||
|       "peerDependencies": { | ||||
|         "zod": "^3.21.4" | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										8
									
								
								examples/langchain-typescript-simple/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | ||||
| { | ||||
|   "devDependencies": { | ||||
|     "typescript": "^5.2.2" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|     "langchain": "^0.0.165" | ||||
|   } | ||||
| } | ||||
							
								
								
									
										7
									
								
								examples/modelfile-10tweets/Modelfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| # Modelfile for creating a list of ten tweets from a topic | ||||
| # Run `ollama create 10tweets -f ./Modelfile` and then `ollama run 10tweets` and enter a topic | ||||
|  | ||||
| FROM llama2 | ||||
| SYSTEM """ | ||||
| You are a content marketer who needs to come up with 10 short but succinct tweets. The answer should be a list of ten tweets. Each tweet can have a maximum of 280 characters and should include hashtags. Each user input will be a subject and you should expand it in ten creative ways. Never stop after just one tweet. Always include ten.  | ||||
| """ | ||||
							
								
								
									
										23
									
								
								examples/modelfile-10tweets/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | ||||
| # Ten Tweets Modelfile | ||||
|  | ||||
| This is a simple modelfile that generates ten tweets based off any topic. | ||||
|  | ||||
| ```bash | ||||
| ollama create tentweets | ||||
|  | ||||
| ollama run tentweets | ||||
| >>> underwater basketweaving | ||||
|  Great! Here are ten creative tweets about underwater basketweaving: | ||||
|  | ||||
| 1. "Just discovered the ultimate stress-reliever: Underwater basketweaving! 🌊🧵 #UnderwaterBasketweaving #StressRelief" | ||||
| 2. "Who needs meditation when you can do underwater basketweaving? 😴👀 #PeacefulDistraction #UnderwaterBasketweaving" | ||||
| 3. "Just spent an hour in the pool and still managed to knot my basket. Goal: untangle it before next session. 💪🏽 #ChallengeAccepted #UnderwaterBasketweaving" | ||||
| 4. "When life gives you lemons, make underwater basketweaving! 🍋🧵 #LemonadeLife #UnderwaterBasketweaving" | ||||
| 5. "Just realized my underwater basketweaving skills could come in handy during a zombie apocalypse. 😂🧡 #SurvivalTips #UnderwaterBasketweaving" | ||||
| 6. "I'm not lazy, I'm just conserving energy for my next underwater basketweaving session. 😴💤 #LazyDay #UnderwaterBasketweaving" | ||||
| 7. "Just found my inner peace while doing underwater basketweaving. It's like meditation, but with knots! 🙏🧵 #Mindfulness #UnderwaterBasketweaving" | ||||
| 8. "Why study for exams when you can do underwater basketweaving and forget all your worries? 😜🧵 #ProcrastinationStation #UnderwaterBasketweaving" | ||||
| 9. "Just had to cut my underwater basketweaving session short due to a sudden urge to breathe. 🤯🌊 #AquaticAdventures #UnderwaterBasketweaving" | ||||
| 10. "I'm not sure what's more impressive: my underwater basketweaving skills or the fact that I didn't drown trying to make this tweet. 😅🧵 #Accomplishment  | ||||
| #UnderwaterBasketweaving" | ||||
| ``` | ||||
							
								
								
									
										8
									
								
								examples/modelfile-devopsengineer/Modelfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | ||||
| # Modelfile for creating a devops engineer assistant | ||||
| # Run `ollama create devops-engineer -f ./Modelfile` and then `ollama run devops-engineer` and enter a topic | ||||
|  | ||||
| FROM mistral | ||||
| PARAMETER temperature 1 | ||||
| SYSTEM """ | ||||
| You are a senior devops engineer, acting as an assistant. You offer help with cloud technologies like: Terraform, AWS, kubernetes, python. You answer with code examples when possible | ||||
| """ | ||||
							
								
								
									
										40
									
								
								examples/modelfile-devopsengineer/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,40 @@ | ||||
| # DevOps Engineer Modelfile | ||||
|  | ||||
| This is a simple modelfile that takes will respond to any questions that a senior DevOps engineer should know how to solve. | ||||
|  | ||||
| ````text | ||||
| ollama create devopsengineer | ||||
|  | ||||
| ollama run devopsengineer | ||||
| >>> how can i proxy all connections from inside the office through a single server that caches large files. Provide a config file for nginx to achieve this? | ||||
|  | ||||
| Here is an example configuration file for Nginx that can be used to proxy all connections from inside the office network to a centralized cache server: | ||||
| ```perl | ||||
| http { | ||||
|     upstream cache_server { | ||||
|         server <cache_server_ip>; | ||||
|         server <cache_server_ip>; | ||||
|         server <cache_server_ip>; | ||||
|     } | ||||
|      | ||||
|     server { | ||||
|         listen 80; | ||||
|         server_name localhost; | ||||
|          | ||||
|         location / { | ||||
|             proxy_pass http://cache_server; | ||||
|             proxy_set_header Host $host; | ||||
|             proxy_set_header X-Real-IP $remote_addr; | ||||
|             proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| ``` | ||||
| In this configuration, we define an `upstream` block that lists the IP addresses of the cache servers. We then define a server block that listens on port 80 and routes all  | ||||
| traffic to the `/` location to the cache servers using the `proxy_pass` directive. The `proxy_set_header` directives are used to preserve the source IP address of the client | ||||
| request when forwarding it to the cache server. | ||||
|  | ||||
| To use this configuration, you would need to replace the placeholder `<cache_server_ip>` with the actual IP addresses of your cache servers. You would also need to make sure | ||||
| that the cache servers are configured to accept incoming connections from the Nginx server and handle requests for files. | ||||
|  | ||||
| ```` | ||||
							
								
								
									
										5
									
								
								examples/modelfile-mario/Modelfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | ||||
| FROM llama2 | ||||
| PARAMETER temperature 1 | ||||
| SYSTEM """ | ||||
| You are Mario from super mario bros, acting as an assistant. | ||||
| """ | ||||
							
								
								
									
										
											BIN
										
									
								
								examples/modelfile-mario/logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 446 KiB | 
							
								
								
									
										43
									
								
								examples/modelfile-mario/readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,43 @@ | ||||
| <img src="logo.png" alt="image of Italian plumber" height="200"/> | ||||
|  | ||||
| # Example character: Mario | ||||
|  | ||||
| This example shows how to create a basic character using Llama2 as the base model. | ||||
|  | ||||
| To run this example: | ||||
|  | ||||
| 1. Download the Modelfile | ||||
| 2. `ollama pull llama2` to get the base model used in the model file. | ||||
| 3. `ollama create NAME -f ./Modelfile` | ||||
| 4. `ollama run NAME` | ||||
|  | ||||
| Ask it some questions like "Who are you?" or "Is Peach in trouble again?" | ||||
|  | ||||
| ## Editing this file | ||||
|  | ||||
| What the model file looks like: | ||||
|  | ||||
| ``` | ||||
| FROM llama2 | ||||
| PARAMETER temperature 1 | ||||
| SYSTEM """ | ||||
| You are Mario from Super Mario Bros, acting as an assistant. | ||||
| """ | ||||
| ``` | ||||
|  | ||||
| What if you want to change its behaviour? | ||||
|  | ||||
| - Try changing the prompt | ||||
| - Try changing the parameters [Docs](https://github.com/jmorganca/ollama/blob/main/docs/modelfile.md) | ||||
| - Try changing the model (e.g. An uncensored model by `FROM wizard-vicuna` this is the wizard-vicuna uncensored model ) | ||||
|  | ||||
| Once the changes are made, | ||||
|  | ||||
| 1. `ollama create NAME -f ./Modelfile` | ||||
| 2. `ollama run NAME` | ||||
| 3. Iterate until you are happy with the results. | ||||
|  | ||||
| Notes: | ||||
|  | ||||
| - This example is for research purposes only. There is no affiliation with any entity. | ||||
| - When using an uncensored model, please be aware that it may generate offensive content. | ||||
							
								
								
									
										11
									
								
								examples/modelfile-midjourney/Modelfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | ||||
| # Modelfile for creating a Midjourney prompts from a topic | ||||
| # This prompt was adapted from the original at https://www.greataiprompts.com/guide/midjourney/best-chatgpt-prompt-for-midjourney/ | ||||
| # Run `ollama create mj -f ./Modelfile` and then `ollama run mj` and enter a topic | ||||
|  | ||||
| FROM zephyr | ||||
| PARAMETER temperature 0.8 | ||||
| PARAMETER top_k 500 | ||||
| PARAMETER top_p 0.9 | ||||
| SYSTEM """ | ||||
| Embrace your role as a creative illustrator. Based on a concept provided, you must produce a single paragraph with a multifaceted description of an image, ensuring significant details of the concept and more is represented in your instructions. You do not need to write complete sentences but rather short concepts with the following information: the level of detail that should be represented, an artistic style and maybe a specific name of a painter or illustrator, the ideal color pallete, lighting, mood, perspective, the setting, time of day, weather, the season, the time period, location, materials, the textures, patterns, lines, brushstrokes, techniques, the medium, the genre, the rendering style. Don't include everything and keep the description length under 250 words.  | ||||
| """ | ||||
							
								
								
									
										11
									
								
								examples/modelfile-midjourney/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,11 @@ | ||||
| # Midjourney Prompt Generator Modelfile | ||||
|  | ||||
| This simple modelfile will help create a prompt to feed to Midjourney. | ||||
|  | ||||
| ```text | ||||
| ollama create midjourney | ||||
|  | ||||
| ollama run midjourney | ||||
| >>> a sports car in the mountains.  | ||||
| A sleek, high-performance automobile cuts through a serpentine mountain landscape. The concept is a classic illustration of speed and power, depicted in the style of pop art by Andy Warhol. The color palette is dominated by bold, primary hues of red, blue, and yellow, with striking accent colors of white, black, and metallic shades. The lighting is bright and focused, casting sharp shadows on the rugged terrain. A sense of excitement and anticipation permeates throughout the scene, as the car navigates a treacherous course through the winding road. The perspective is low, allowing for a full view of the vehicle's sleek lines and intricate details. The setting takes place in the afternoon during a sunny day in autumn, as evidenced by the vibrant foliage on the mountainside. The time period is modern, with nods to classic car design. The materials are primarily digital, allowing for smooth curves and sharp contrasts. The textures are sleek and polished, with meticulously detailed lines and brushstrokes that accentuate the car's aerodynamic design. The patterns consist of geometric shapes and bold stripes, adding to the car's dynamic appeal. The genre is modern realism, with a focus on precision and detail. The rendering style is highly technical, capturing the nuances and subtleties of the vehicle and its surroundings in breathtaking detail. | ||||
| ``` | ||||
							
								
								
									
										6
									
								
								examples/modelfile-recipemaker/Modelfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,6 @@ | ||||
| # Modelfile for creating a recipe from a list of ingredients | ||||
| # Run `ollama create recipemaker -f ./Modelfile` and then `ollama run recipemaker` and feed it lists of ingredients to create recipes around. | ||||
| FROM nous-hermes | ||||
| SYSTEM """ | ||||
| The instruction will be a list of ingredients. You should generate a recipe that can be made in less than an hour. You can also include ingredients that most people will find in their pantry every day. The recipe should be 4 people and you should include a description of what the meal will taste like | ||||
| """ | ||||
							
								
								
									
										20
									
								
								examples/modelfile-recipemaker/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | ||||
| # Recipe Maker Modelfile  | ||||
|  | ||||
| Simple modelfile to generate a recipe from a short list of ingredients. | ||||
|  | ||||
| ``` | ||||
| ollama create recipemaker | ||||
|  | ||||
| ollama run recipemaker | ||||
| >>> chilli pepper, white chocolate, kale | ||||
|  Ingredients: | ||||
| - 1 small chili pepper | ||||
| - 4 squares of white chocolate | ||||
| - handful of kale leaves | ||||
|  | ||||
| Instructions: | ||||
| 1. In a blender or food processor, puree the chilies and white chocolate until smooth. | ||||
| 2. Add the chopped kale leaves to the blender and pulse until well combined. | ||||
| 3. Serve immediately as a dip for crackers or use it as an ingredient in your favorite recipe. The mixture of spicy chili pepper with sweet white chocolate and nutritious  | ||||
| kale will make your taste buds dance with delight! | ||||
| ``` | ||||
							
								
								
									
										28
									
								
								examples/modelfile-sentiments/Modelfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,28 @@ | ||||
| # Modelfile for creating a sentiment analyzer.  | ||||
| # Run `ollama create sentiments -f pathtofile` and then `ollama run sentiments` and enter a topic | ||||
|  | ||||
| FROM orca | ||||
| TEMPLATE """ | ||||
| {{- if .First }} | ||||
| ### System: | ||||
| {{ .System }} | ||||
| {{- end }} | ||||
| ### User:  | ||||
| I hate it when my phone dies | ||||
| ### Response:  | ||||
| NEGATIVE | ||||
| ### User:  | ||||
| He is awesome | ||||
| ### Response:  | ||||
| POSITIVE | ||||
| ### User:  | ||||
| This is the link to the article | ||||
| ### Response:  | ||||
| NEUTRAL | ||||
| ### User: | ||||
| {{ .Prompt }} | ||||
|  | ||||
| ### Response: | ||||
| """ | ||||
|  | ||||
| SYSTEM """You are a sentiment analyzer. You will receive text and output only one word, either POSITIVE or NEGATIVE or NEUTRAL, depending on the sentiment of the text.""" | ||||
							
								
								
									
										25
									
								
								examples/modelfile-sentiments/Readme.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | ||||
| # Sentiments Modelfile | ||||
|  | ||||
| This is a simple sentiments analyzer using the Orca model. When you pull Orca from the registry, it has a Template already defined that looks like this: | ||||
|  | ||||
| ```Modelfile | ||||
| {{- if .First }} | ||||
| ### System: | ||||
| {{ .System }} | ||||
| {{- end }} | ||||
|  | ||||
| ### User: | ||||
| {{ .Prompt }} | ||||
|  | ||||
| ### Response: | ||||
| ``` | ||||
|  | ||||
| If we just wanted to have the text: | ||||
|  | ||||
| ```Plaintext | ||||
| You are a sentiment analyzer. You will receive text and output only one word, either POSITIVE or NEGATIVE or NEUTRAL, depending on the sentiment of the text. | ||||
| ``` | ||||
|  | ||||
| then we could have put this in a SYSTEM block. But we want to provide examples which require updating the full Template. Any Modelfile you create will inherit all the settings from the source model. But in this example, we are overriding the Template. | ||||
|  | ||||
| When providing examples for the input and output, you should include the way the model usually provides information. Since the Orca model expects a user prompt to appear after ### User: and the response is after ### Response, we should format our examples like that as well. If we were using the Llama 2 model, the format would be a bit different. | ||||
							
								
								
									
										7
									
								
								examples/modelfile-tweetwriter/Modelfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| # Modelfile for creating a tweet from a topic | ||||
| # Run `ollama create tweetwriter -f ./Modelfile` and then `ollama run tweetwriter` and enter a topic | ||||
|  | ||||
| FROM nous-hermes | ||||
| SYSTEM """ | ||||
| You are a content marketer who needs to come up with a short but succinct tweet. Make sure to include the appropriate hashtags and links. Sometimes when appropriate, describe a meme that can be included as well. All answers should be in the form of a tweet which has a max size of 280 characters. Every instruction will be the topic to create a tweet about. | ||||
| """ | ||||
							
								
								
									
										20
									
								
								examples/python-dockerit/Modelfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | ||||
| FROM mistral | ||||
| SYSTEM """ | ||||
| You are an experienced Devops engineer focused on docker. When given specifications for a particular need or application you know the best way to host that within a docker container. For instance if someone tells you they want an nginx server to host files located at /web you will answer as follows | ||||
|  | ||||
| ---start | ||||
| FROM nginx:alpine | ||||
| COPY /myweb /usr/share/nginx/html | ||||
| EXPOSE 80 | ||||
| ---end | ||||
|  | ||||
| Notice that the answer you should give is just the contents of the dockerfile with no explanation and there are three dashes and the word start at the beginning and 3 dashes and the word end. The full output can be piped into a file and run as is. Here is another example. The user will ask to launch a Postgres server with a password of abc123. And the response should be | ||||
|  | ||||
| ---start | ||||
| FROM postgres:latest | ||||
| ENV POSTGRES_PASSWORD=abc123 | ||||
| EXPOSE 5432 | ||||
| ---end | ||||
|  | ||||
| Again it's just the contents of the dockerfile and nothing else. | ||||
| """ | ||||
							
								
								
									
										15
									
								
								examples/python-dockerit/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| # DockerIt | ||||
|  | ||||
| DockerIt is a tool to help you build and run your application in a Docker container. It consists of a model that defines the system prompt and model weights to use, along with a python script to then build the container and run the image automatically.  | ||||
|  | ||||
| ## Caveats | ||||
|  | ||||
| This is an simple example. It's assuming the Dockerfile content generated is going to work. In many cases, even with simple web servers, it fails when trying to copy files that don't exist. It's simply an example of what you could possibly do. | ||||
|  | ||||
| ## Example Usage | ||||
|  | ||||
| ```bash | ||||
| > python3 ./dockerit.py "simple postgres server with admin password set to 123" | ||||
| Enter the name of the image: matttest | ||||
| Container named happy_keller  started with id:  7c201bb6c30f02b356ddbc8e2a5af9d7d7d7b8c228519c9a501d15c0bd9d6b3e | ||||
| ``` | ||||
							
								
								
									
										17
									
								
								examples/python-dockerit/dockerit.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,17 @@ | ||||
| import requests, json, docker, io, sys | ||||
| inputDescription = " ".join(sys.argv[1:]) | ||||
| imageName = input("Enter the name of the image: ") | ||||
| client = docker.from_env() | ||||
| s = requests.Session() | ||||
| output="" | ||||
| with s.post('http://localhost:11434/api/generate', json={'model': 'dockerit', 'prompt': inputDescription}, stream=True) as r: | ||||
|   for line in r.iter_lines(): | ||||
|     if line: | ||||
|       j = json.loads(line) | ||||
|       if "response" in j: | ||||
|         output = output +j["response"] | ||||
| output = output[output.find("---start")+9:output.find("---end")-1] | ||||
| f = io.BytesIO(bytes(output, 'utf-8')) | ||||
| client.images.build(fileobj=f, tag=imageName) | ||||
| container = client.containers.run(imageName, detach=True) | ||||
| print("Container named", container.name, " started with id: ",container.id) | ||||
							
								
								
									
										1
									
								
								examples/python-dockerit/requirements.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| docker | ||||
							
								
								
									
										38
									
								
								examples/python-simplegenerate/client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,38 @@ | ||||
| import json | ||||
| import requests | ||||
|  | ||||
| # NOTE: ollama must be running for this to work, start the ollama app or run `ollama serve` | ||||
| model = 'llama2' # TODO: update this for whatever model you wish to use | ||||
|  | ||||
| def generate(prompt, context): | ||||
|     r = requests.post('http://localhost:11434/api/generate', | ||||
|                       json={ | ||||
|                           'model': model, | ||||
|                           'prompt': prompt, | ||||
|                           'context': context, | ||||
|                       }, | ||||
|                       stream=True) | ||||
|     r.raise_for_status() | ||||
|  | ||||
|     for line in r.iter_lines(): | ||||
|         body = json.loads(line) | ||||
|         response_part = body.get('response', '') | ||||
|         # the response streams one token at a time, print that as we recieve it | ||||
|         print(response_part, end='', flush=True) | ||||
|  | ||||
|         if 'error' in body: | ||||
|             raise Exception(body['error']) | ||||
|  | ||||
|         if body.get('done', False): | ||||
|             return body['context'] | ||||
|  | ||||
| def main(): | ||||
|     context = [] # the context stores a conversation history, you can use this to make the model more context aware | ||||
|     while True: | ||||
|         user_input = input("Enter a prompt: ") | ||||
|         print() | ||||
|         context = generate(user_input, context) | ||||
|         print() | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
| @@ -1,15 +0,0 @@ | ||||
| # Python | ||||
|  | ||||
| This is a simple example of calling the Ollama api from a python app. | ||||
|  | ||||
| First, download a model: | ||||
|  | ||||
| ``` | ||||
| curl -L https://huggingface.co/TheBloke/orca_mini_3B-GGML/resolve/main/orca-mini-3b.ggmlv3.q4_1.bin -o orca.bin | ||||
| ``` | ||||
|  | ||||
| Then run it using the example script. You'll need to have Ollama running on your machine. | ||||
|  | ||||
| ``` | ||||
| python3 main.py orca.bin | ||||
| ``` | ||||
| @@ -1,32 +0,0 @@ | ||||
| import http.client | ||||
| import json | ||||
| import os | ||||
| import sys | ||||
|  | ||||
| if len(sys.argv) < 2: | ||||
|     print("Usage: python main.py <model file>") | ||||
|     sys.exit(1) | ||||
|  | ||||
| conn = http.client.HTTPConnection('localhost', 11434) | ||||
|  | ||||
| headers = { 'Content-Type': 'application/json' } | ||||
|  | ||||
| # generate text from the model | ||||
| conn.request("POST", "/api/generate", json.dumps({ | ||||
|     'model': os.path.join(os.getcwd(), sys.argv[1]), | ||||
|     'prompt': 'write me a short story', | ||||
|     'stream': True | ||||
| }), headers) | ||||
|  | ||||
| response = conn.getresponse() | ||||
|  | ||||
| def parse_generate(data): | ||||
|     for event in data.decode('utf-8').split("\n"): | ||||
|         if not event: | ||||
|             continue | ||||
|         yield event | ||||
|  | ||||
| if response.status == 200: | ||||
|     for chunk in response: | ||||
|         for event in parse_generate(chunk): | ||||
|             print(json.loads(event)['response'], end="", flush=True) | ||||
							
								
								
									
										2
									
								
								examples/typescript-mentors/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,2 @@ | ||||
| node_modules | ||||
| package-lock.json | ||||
							
								
								
									
										21
									
								
								examples/typescript-mentors/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| # Ask the Mentors | ||||
|  | ||||
| This example demonstrates how one would create a set of 'mentors' you can have a conversation with. The mentors are generated using the `character-generator.ts` file. This will use **Stable Beluga 70b** to create a bio and list of verbal ticks and common phrases used by each person. Then `mentors.ts` will take a question, and choose three of the 'mentors' and start a conversation with them. Occasionally, they will talk to each other, and other times they will just deliver a set of monologues. It's fun to see what they do and say. | ||||
|  | ||||
| ## Usage | ||||
|  | ||||
| ```bash | ||||
| ts-node ./character-generator.ts "Lorne Greene" | ||||
| ``` | ||||
|  | ||||
| This will create `lornegreene/Modelfile`. Now you can create a model with this command: | ||||
|  | ||||
| ```bash | ||||
| ollama create lornegreene -f lornegreene/Modelfile | ||||
| ``` | ||||
|  | ||||
| If you want to add your own mentors, you will have to update the code to look at your namespace instead of **mattw**. Also set the list of mentors to include yours. | ||||
|  | ||||
| ```bash | ||||
| ts-node ./mentors.ts "What is a Jackalope?" | ||||
| ``` | ||||
							
								
								
									
										26
									
								
								examples/typescript-mentors/character-generator.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | ||||
| import { Ollama } from 'ollama-node' | ||||
| import fs from 'fs'; | ||||
| import path from 'path'; | ||||
|  | ||||
| async function characterGenerator() { | ||||
|   const character = process.argv[2]; | ||||
|   console.log(`You are creating a character for ${character}.`); | ||||
|   const foldername = character.replace(/\s/g, '').toLowerCase(); | ||||
|   const directory = path.join(__dirname, foldername); | ||||
|   if (!fs.existsSync(directory)) { | ||||
|     fs.mkdirSync(directory, { recursive: true }); | ||||
|   } | ||||
|  | ||||
|   const ollama = new Ollama(); | ||||
|   ollama.setModel("stablebeluga2:70b-q4_K_M"); | ||||
|   const bio = await ollama.generate(`create a bio of ${character} in a single long paragraph. Instead of saying '${character} is...' or '${character} was...' use language like 'You are...' or 'You were...'. Then create a paragraph describing the speaking mannerisms and style of ${character}. Don't include anything about how ${character} looked or what they sounded like, just focus on the words they said. Instead of saying '${character} would say...' use language like 'You should say...'. If you use quotes, always use single quotes instead of double quotes. If there are any specific words or phrases you used a lot, show how you used them. `); | ||||
|  | ||||
|   const thecontents = `FROM llama2\nSYSTEM """\n${bio.response.replace(/(\r\n|\n|\r)/gm, " ").replace('would', 'should')} All answers to questions should be related back to what you are most known for.\n"""`; | ||||
|  | ||||
|   fs.writeFile(path.join(directory, 'Modelfile'), thecontents, (err: any) => { | ||||
|     if (err) throw err; | ||||
|     console.log('The file has been saved!'); | ||||
|   }); | ||||
| } | ||||
|  | ||||
| characterGenerator(); | ||||
							
								
								
									
										59
									
								
								examples/typescript-mentors/mentors.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,59 @@ | ||||
| import { Ollama } from 'ollama-node'; | ||||
|  | ||||
| const mentorCount = 3; | ||||
| const ollama = new Ollama(); | ||||
|  | ||||
| function getMentors(): string[] { | ||||
|   const mentors = ['Gary Vaynerchuk', 'Kanye West', 'Martha Stewart', 'Neil deGrasse Tyson', 'Owen Wilson', 'Ronald Reagan', 'Donald Trump', 'Barack Obama', 'Jeff Bezos']; | ||||
|   const chosenMentors: string[] = []; | ||||
|   for (let i = 0; i < mentorCount; i++) { | ||||
|     const mentor = mentors[Math.floor(Math.random() * mentors.length)]; | ||||
|     chosenMentors.push(mentor); | ||||
|     mentors.splice(mentors.indexOf(mentor), 1); | ||||
|   } | ||||
|   return chosenMentors; | ||||
| } | ||||
|  | ||||
| function getMentorFileName(mentor: string): string { | ||||
|   const model = mentor.toLowerCase().replace(/\s/g, ''); | ||||
|   return `mattw/${model}`; | ||||
| } | ||||
|  | ||||
| async function getSystemPrompt(mentor: string, isLast: boolean, question: string): Promise<string> { | ||||
|   ollama.setModel(getMentorFileName(mentor)); | ||||
|   const info = await ollama.showModelInfo() | ||||
|   let SystemPrompt = info.system || ''; | ||||
|   SystemPrompt += ` You should continue the conversation as if you were ${mentor} and acknowledge the people before you in the conversation. You should adopt their mannerisms and tone, but also not use language they wouldn't use. If they are not known to know about the concept in the question, don't offer an answer. Your answer should be no longer than 1 paragraph. And definitely try not to sound like anyone else. Don't repeat any slang or phrases already used. And if it is a question the original ${mentor} wouldn't have know the answer to, just say that you don't know, in the style of ${mentor}. And think about the time the person lived. Don't use terminology that they wouldn't have used.` | ||||
|  | ||||
|   if (isLast) { | ||||
|     SystemPrompt += ` End your answer with something like I hope our answers help you out`; | ||||
|   } else { | ||||
|     SystemPrompt += ` Remember, this is a conversation, so you don't need a conclusion, but end your answer with a question related to the first question: "${question}".`; | ||||
|   } | ||||
|   return SystemPrompt; | ||||
| } | ||||
|  | ||||
| async function main() { | ||||
|   const mentors = getMentors(); | ||||
|   const question = process.argv[2]; | ||||
|   let theConversation = `Here is the conversation so far.\nYou: ${question}\n` | ||||
|  | ||||
|   for await (const mentor of mentors) { | ||||
|     const SystemPrompt = await getSystemPrompt(mentor, mentor === mentors[mentorCount - 1], question); | ||||
|     ollama.setModel(getMentorFileName(mentor)); | ||||
|     ollama.setSystemPrompt(SystemPrompt); | ||||
|     let output = ''; | ||||
|     process.stdout.write(`\n${mentor}: `); | ||||
|     for await (const chunk of ollama.streamingGenerate(theConversation + `Continue the conversation as if you were ${mentor} on the question "${question}".`)) { | ||||
|       if (chunk.response) { | ||||
|         output += chunk.response; | ||||
|         process.stdout.write(chunk.response); | ||||
|       } else { | ||||
|         process.stdout.write('\n'); | ||||
|       } | ||||
|     } | ||||
|     theConversation += `${mentor}: ${output}\n\n` | ||||
|   } | ||||
| } | ||||
|  | ||||
| main(); | ||||
							
								
								
									
										7
									
								
								examples/typescript-mentors/package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,7 @@ | ||||
| { | ||||
|   "dependencies": { | ||||
|     "fs": "^0.0.1-security", | ||||
|     "ollama-node": "^0.0.3", | ||||
|     "path": "^0.12.7" | ||||
|   } | ||||
| } | ||||
							
								
								
									
										16
									
								
								format/bytes.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,16 @@ | ||||
| package format | ||||
|  | ||||
| import "fmt" | ||||
|  | ||||
| func HumanBytes(b int64) string { | ||||
| 	switch { | ||||
| 	case b > 1000*1000*1000: | ||||
| 		return fmt.Sprintf("%d GB", b/1000/1000/1000) | ||||
| 	case b > 1000*1000: | ||||
| 		return fmt.Sprintf("%d MB", b/1000/1000) | ||||
| 	case b > 1000: | ||||
| 		return fmt.Sprintf("%d KB", b/1000) | ||||
| 	default: | ||||
| 		return fmt.Sprintf("%d B", b) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										102
									
								
								format/openssh.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,102 @@ | ||||
| // Copyright 2012 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // Code originally from https://go-review.googlesource.com/c/crypto/+/218620 | ||||
|  | ||||
| // TODO: replace with upstream once the above change is merged and released. | ||||
|  | ||||
| package format | ||||
|  | ||||
| import ( | ||||
| 	"crypto" | ||||
| 	"crypto/ed25519" | ||||
| 	"crypto/rand" | ||||
| 	"encoding/binary" | ||||
| 	"encoding/pem" | ||||
| 	"fmt" | ||||
|  | ||||
| 	"golang.org/x/crypto/ssh" | ||||
| ) | ||||
|  | ||||
| const privateKeyAuthMagic = "openssh-key-v1\x00" | ||||
|  | ||||
| type openSSHEncryptedPrivateKey struct { | ||||
| 	CipherName string | ||||
| 	KDFName    string | ||||
| 	KDFOptions string | ||||
| 	KeysCount  uint32 | ||||
| 	PubKey     []byte | ||||
| 	KeyBlocks  []byte | ||||
| } | ||||
|  | ||||
| type openSSHPrivateKey struct { | ||||
| 	Check1  uint32 | ||||
| 	Check2  uint32 | ||||
| 	Keytype string | ||||
| 	Rest    []byte `ssh:"rest"` | ||||
| } | ||||
|  | ||||
| type openSSHEd25519PrivateKey struct { | ||||
| 	Pub     []byte | ||||
| 	Priv    []byte | ||||
| 	Comment string | ||||
| 	Pad     []byte `ssh:"rest"` | ||||
| } | ||||
|  | ||||
| func OpenSSHPrivateKey(key crypto.PrivateKey, comment string) (*pem.Block, error) { | ||||
| 	var check uint32 | ||||
| 	if err := binary.Read(rand.Reader, binary.BigEndian, &check); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var pk1 openSSHPrivateKey | ||||
| 	pk1.Check1 = check | ||||
| 	pk1.Check2 = check | ||||
|  | ||||
| 	var w openSSHEncryptedPrivateKey | ||||
| 	w.KeysCount = 1 | ||||
|  | ||||
| 	if k, ok := key.(*ed25519.PrivateKey); ok { | ||||
| 		key = *k | ||||
| 	} | ||||
|  | ||||
| 	switch k := key.(type) { | ||||
| 	case ed25519.PrivateKey: | ||||
| 		pub, priv := k[32:], k | ||||
| 		key := openSSHEd25519PrivateKey{ | ||||
| 			Pub:     pub, | ||||
| 			Priv:    priv, | ||||
| 			Comment: comment, | ||||
| 		} | ||||
|  | ||||
| 		pk1.Keytype = ssh.KeyAlgoED25519 | ||||
| 		pk1.Rest = ssh.Marshal(key) | ||||
|  | ||||
| 		w.PubKey = ssh.Marshal(struct { | ||||
| 			KeyType string | ||||
| 			Pub     []byte | ||||
| 		}{ | ||||
| 			ssh.KeyAlgoED25519, pub, | ||||
| 		}) | ||||
| 	default: | ||||
| 		return nil, fmt.Errorf("ssh: unknown key type %T", k) | ||||
| 	} | ||||
|  | ||||
| 	w.KeyBlocks = openSSHPadding(ssh.Marshal(pk1), 8) | ||||
|  | ||||
| 	w.CipherName, w.KDFName, w.KDFOptions = "none", "none", "" | ||||
|  | ||||
| 	return &pem.Block{ | ||||
| 		Type:  "OPENSSH PRIVATE KEY", | ||||
| 		Bytes: append([]byte(privateKeyAuthMagic), ssh.Marshal(w)...), | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| func openSSHPadding(block []byte, blocksize int) []byte { | ||||
| 	for i, j := 0, len(block); (j+i)%blocksize != 0; i++ { | ||||
| 		block = append(block, byte(i+1)) | ||||
| 	} | ||||
|  | ||||
| 	return block | ||||
| } | ||||
							
								
								
									
										68
									
								
								format/time.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,68 @@ | ||||
| package format | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"math" | ||||
| 	"strings" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| // humanDuration returns a human-readable approximation of a | ||||
| // duration (eg. "About a minute", "4 hours ago", etc.). | ||||
| func humanDuration(d time.Duration) string { | ||||
| 	seconds := int(d.Seconds()) | ||||
|  | ||||
| 	switch { | ||||
| 	case seconds < 1: | ||||
| 		return "Less than a second" | ||||
| 	case seconds == 1: | ||||
| 		return "1 second" | ||||
| 	case seconds < 60: | ||||
| 		return fmt.Sprintf("%d seconds", seconds) | ||||
| 	} | ||||
|  | ||||
| 	minutes := int(d.Minutes()) | ||||
| 	switch { | ||||
| 	case minutes == 1: | ||||
| 		return "About a minute" | ||||
| 	case minutes < 60: | ||||
| 		return fmt.Sprintf("%d minutes", minutes) | ||||
| 	} | ||||
|  | ||||
| 	hours := int(math.Round(d.Hours())) | ||||
| 	switch { | ||||
| 	case hours == 1: | ||||
| 		return "About an hour" | ||||
| 	case hours < 48: | ||||
| 		return fmt.Sprintf("%d hours", hours) | ||||
| 	case hours < 24*7*2: | ||||
| 		return fmt.Sprintf("%d days", hours/24) | ||||
| 	case hours < 24*30*2: | ||||
| 		return fmt.Sprintf("%d weeks", hours/24/7) | ||||
| 	case hours < 24*365*2: | ||||
| 		return fmt.Sprintf("%d months", hours/24/30) | ||||
| 	} | ||||
|  | ||||
| 	return fmt.Sprintf("%d years", int(d.Hours())/24/365) | ||||
| } | ||||
|  | ||||
| func HumanTime(t time.Time, zeroValue string) string { | ||||
| 	return humanTime(t, zeroValue) | ||||
| } | ||||
|  | ||||
| func HumanTimeLower(t time.Time, zeroValue string) string { | ||||
| 	return strings.ToLower(humanTime(t, zeroValue)) | ||||
| } | ||||
|  | ||||
| func humanTime(t time.Time, zeroValue string) string { | ||||
| 	if t.IsZero() { | ||||
| 		return zeroValue | ||||
| 	} | ||||
|  | ||||
| 	delta := time.Since(t) | ||||
| 	if delta < 0 { | ||||
| 		return humanDuration(-delta) + " from now" | ||||
| 	} | ||||
|  | ||||
| 	return humanDuration(delta) + " ago" | ||||
| } | ||||
							
								
								
									
										35
									
								
								format/time_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,35 @@ | ||||
| package format | ||||
|  | ||||
| import ( | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func assertEqual(t *testing.T, a interface{}, b interface{}) { | ||||
| 	if a != b { | ||||
| 		t.Errorf("Assert failed, expected %v, got %v", b, a) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestHumanTime(t *testing.T) { | ||||
| 	now := time.Now() | ||||
|  | ||||
| 	t.Run("zero value", func(t *testing.T) { | ||||
| 		assertEqual(t, HumanTime(time.Time{}, "never"), "never") | ||||
| 	}) | ||||
|  | ||||
| 	t.Run("time in the future", func(t *testing.T) { | ||||
| 		v := now.Add(48 * time.Hour) | ||||
| 		assertEqual(t, HumanTime(v, ""), "2 days from now") | ||||
| 	}) | ||||
|  | ||||
| 	t.Run("time in the past", func(t *testing.T) { | ||||
| 		v := now.Add(-48 * time.Hour) | ||||
| 		assertEqual(t, HumanTime(v, ""), "2 days ago") | ||||
| 	}) | ||||
|  | ||||
| 	t.Run("soon", func(t *testing.T) { | ||||
| 		v := now.Add(800*time.Millisecond) | ||||
| 		assertEqual(t, HumanTime(v, ""), "Less than a second from now") | ||||
| 	}) | ||||
| } | ||||