Spaces:
Running
Running
File size: 13,640 Bytes
e43f92e 31d62a0 0f705a4 e43f92e 0f705a4 7a1a853 3c48c3b 0f705a4 e43f92e 85c6280 e43f92e 85c6280 e43f92e 85c6280 ec2ae5c e43f92e 7a1a853 85c6280 ec2ae5c 85c6280 0793441 85c6280 0793441 a987701 0793441 a987701 0793441 85c6280 0f705a4 7a1a853 e8d1ff0 4f6a9ca 2dd89c2 4f6a9ca 7a1a853 2b9607d 1db6fbd 0793441 0f705a4 da50add 0f705a4 003b42f 0f705a4 0e8504c 7886e62 0793441 05b3af6 0f705a4 3c48c3b 85c6280 ec2ae5c 85c6280 0793441 a987701 7a1a853 0793441 7a1a853 e43f92e 0793441 4883d8e e3e7c89 0793441 7a1a853 0793441 7a1a853 0793441 7a1a853 0793441 7a1a853 e3e7c89 7a1a853 4883d8e 0793441 a987701 0793441 a987701 86292eb a987701 0793441 a3b7031 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 |
from time import sleep
import streamlit as st
import openai
import pinecone
from postgres_db import query_postgresql_realvest
PINECONE_API_KEY = st.secrets["PINECONE_API_KEY"]
OPENAI_API_KEY = st.secrets["OPENAI_API_KEY"]
INDEX_NAME = 'realvest-data-v2'
EMBEDDING_MODEL = "text-embedding-ada-002" # OpenAI's best embeddings as of Apr 2023
MAX_LENGTH_DESC = 200
MATCH_SCORE_THR = 0.0
TOP_K = 20
def query_pinecone(xq, top_k: int=3, include_metadata: bool=True, sleep_time: int=10):
MAX_TRIALS = 5
trial = 0
out = None
while (out is None) and (trial < MAX_TRIALS):
try:
out = st.session_state['index'].query(xq, top_k=top_k, include_metadata=include_metadata)
return out
except pinecone.core.exceptions.PineconeProtocolError as err:
print(f"Error, sleep! {err}")
sleep(sleep_time)
trial = trial + 1
return out
def sort_dict_by_value(d: dict, ascending: bool=True):
"""
Sort dictionary {k1: v1, k2: v2} by its value. The output is
a sorted list of tuples [(k1, v1), (k2, v2)]
"""
return sorted(d.items(), key=lambda x: x[1], reverse=not ascending)
# initialize connection to pinecone (get API key at app.pinecone.io)
from tenacity import retry, stop_after_attempt, wait_fixed
pinecone.init(
api_key=PINECONE_API_KEY,
environment="us-central1-gcp" # may be different, check at app.pinecone.io
)
@retry(stop=stop_after_attempt(5), wait=wait_fixed(15))
def setup_pinecone_index():
try:
print("Attempting to set up Pinecone index...") # add this line
if "index" not in st.session_state:
st.session_state['index'] = pinecone.Index(INDEX_NAME)
return st.session_state['index']
except AttributeError as e:
print("Caught an AttributeError:", e)
raise # Re-raise the exception so that tenacity can catch it and retry
except Exception as e: # add this block
print("Caught an unexpected exception:", e)
raise
def init_session_state():
try:
st.session_state['index'] = setup_pinecone_index()
# stats = test_pinecone()
except Exception as e:
print("Failed to set up Pinecone index after several attempts. Error:", e)
if 'display_results' not in st.session_state:
st.session_state['display_results'] = False
if 'count_checked' not in st.session_state:
st.session_state['count_checked'] = 0
def callback_count_checked():
st.session_state['count_checked'] = 0
st.session_state['checked_boxes'] = []
for key in list(st.session_state.keys()):
if (key.split('__')[0] == 'cb_compare') and (st.session_state[key] == True):
st.session_state['count_checked'] += 1
st.session_state['checked_boxes'].append(key)
def summarize_products(products: list) -> str:
"""
Input:
products = [
{text information of product#1},
{text information of product#2},
{text information of product#3},
]
Output:
summary = "{summary of all products}"
"""
NEW_LINE = '\n'
prompt = f"""
Based on the product information below, please read and try to understand it.
{ f"{NEW_LINE*2}---{NEW_LINE*2}".join(products) }
Please write a concise and insightful summary table (display as HTML) to compare the products for investors, which should inlcude but not limited to:
- description
- category
- asking price
- location
- potential profit margin
"""
print(f"prompt: {prompt}")
openai.api_key = OPENAI_API_KEY
completion = openai.ChatCompletion.create(
model="gpt-4",
messages=[
{"role": "system", "content": "You are a helpful assistant."},
{"role": "user", "content": prompt}
]
)
summary = completion.choices[0].message
return summary
### Main
# st.set_page_config(layout="centered")
css='''
<style>
section.main > div {max-width:75rem}
input[type="text"] {
background-color: #F2FEEF !important;
}
</style>
'''
st.markdown(css, unsafe_allow_html=True)
# remove the hamburger in the upper right hand corner and the Made with Streamlit footer
hide_menu_style = """
<style>
#MainMenu {visibility: hidden;}
footer {visibility: hidden;}
</style>
"""
st.markdown(hide_menu_style, unsafe_allow_html=True)
# initialize state
init_session_state()
# Create a text input field
query = st.text_input("Storages, Offices, Car Washes, Laundromats... we have it all")
# Create a button
if st.button('Search'):
# # initialize
# st.session_state.clear()
# init_session_state()
st.session_state['count_checked'] = 0
st.session_state['checked_boxes'] = []
# ### call OpenAI text-embedding
res = openai.Embedding.create(model=EMBEDDING_MODEL, input=[query], api_key=OPENAI_API_KEY)
xq = res['data'][0]['embedding']
out = query_pinecone(xq, top_k=TOP_K, include_metadata=True)
if (out is not None) and ('matches' in out):
metadata = {match['metadata']['product_id']: match['metadata'] for match in out['matches'] if 'metadata' in match and match['metadata'] is not None}
### candidates
metadata = {match['metadata']['product_id']: match['metadata'] for match in out['matches']}
match_score = {match['metadata']['product_id']: match['score'] for match in out['matches']}
above_thr_sorted = [
item
for item in sort_dict_by_value(match_score, ascending=False)
if item[1] > MATCH_SCORE_THR
]
pids = metadata.keys()
### query pids
pids_str = [f"'{pid}'" for pid, _ in above_thr_sorted]
query = f"""
SELECT productid, name, category, alternatename, url, logo, description
FROM main_products
WHERE productid in ({', '.join(pids_str)});
"""
results = query_postgresql_realvest(query)
results = {
result['productid']: result
for result in results
}
### For test
# print(f"above_thr_sorted: {above_thr_sorted}")
# print(f"results: {results}")
# print(f"metadata: {metadata}")
# # TEST ONLY
# above_thr_sorted = [('2086773', 0.800059378), ('1951083', 0.797319531), ('1998714', 0.795623)]
# results = {'1951083': {'productid': '1951083', 'name': '2 for 1 Turn-key Business Opportunity in Lynnwood, Washington - BizBuySell', 'category': 'Other', 'alternatename': None, 'url': 'https://www.bizbuysell.com/Business-Opportunity/2-for-1-Turn-key-Business-Opportunity/1951083/', 'logo': 'https://images.bizbuysell.com/shared/listings/195/1951083/87198e08-a191-4d97-a33b-9e9f40fa02f4-W768.jpg', 'description': 'Your chance to own a successful Korean traditional KBBQ grill restaurant and Korean dive bar. Owner is retiring after 19 years of business. This Korean BBQ restaurant utilizes a traditional grill called "Sot Ttu Kkeong" widely found in Korea. There are 10 separate grilling tables with a unique hood system to eliminate odors immediately. The bar next door may be able to extend hours into the summer. With one shared full kitchen, the new owner will be able to maximize business and potentially earn double income.'}, '1998714': {'productid': '1998714', 'name': 'Portland CPA Firm in Portland, Oregon - BizBuySell', 'category': 'Accounting and Tax Practices', 'alternatename': None, 'url': 'https://www.bizbuysell.com/Business-Opportunity/Portland-CPA-Firm/1998714/', 'logo': 'https://images.bizbuysell.com/shared/listings/199/1998714/cd02bdb9-32c9-409d-b82e-d0531c12eb39-W768.jpg', 'description': 'OR1002: UPDATED :The seller of this Portland CPA firm is approaching retirement and ready to sell the firm. The firm has a great reputation, has good systems in place, is paperless, and has a great staff. The mix of services offers a consistent stream of cash flow to the owner. The seller is seeking a CPA buyer. The office space is available for continued lease after the sale. Revenues for sale include:7% Accounting, bookkeeping and payroll services26% Income tax preparation services for individual clients35% Income tax preparation services for business and other clients28% Audits and reviews4% Consulting services'}, '2086773': {'productid': '2086773', 'name': 'Asian Grocery Supermarket, 1 owner for 29 years in Salem, Oregon - BizBuySell', 'category': 'Grocery Stores and Supermarkets', 'alternatename': None, 'url': 'https://www.bizbuysell.com/Business-Real-Estate-For-Sale/Asian-Grocery-Supermarket-1-owner-for-29-years/2086773/', 'logo': 'https://images.bizbuysell.com/shared/listings/208/2086773/861f6ba6-a994-4e90-9c62-0a593dae2a31-W768.jpg', 'description': 'Great location, well established and profitable supermarket.We have been the sole owner for almost 29 years, so business boasts of a great reputation.'}}
# metadata = {'2086773': {'asking_price': 1000000.0, 'asking_price_currency': 'USD', 'building_status': 'Established', 'category': 'Grocery Stores and Supermarkets', 'chunk_type': 'profile', 'city': 'Salem', 'document': '# Listing Profile\n \nAsking Price (USD): 1000000 \n\nReason for Selling: Retire ', 'listing_type': 'Retail', 'location': 'Salem, OR', 'main_category': 'Grocery Stores and Supermarkets', 'offer_type': 'Offer', 'offers__available_from__address__locality': 'Salem', 'offers__available_from__address__region': 'Oregon', 'offers__available_from__address__type': 'PostalAddress', 'offers__available_from__type': 'Place', 'product_id': '2086773', 'similar_pids': ['2074401', '2087795', '2068650'], 'state_code': 'OR'}, '1951083': {'asking_price': 200000.0, 'asking_price_currency': 'USD', 'category': 'Other', 'chunk_type': 'profile', 'city': 'Lynnwood', 'document': '# Listing Profile\n \nAsking Price (USD): 200000 \n\nReason for Selling: Retiring ', 'location': 'Lynnwood, WA', 'main_category': 'Other', 'offer_type': 'Offer', 'offers__available_from__address__locality': 'Lynnwood', 'offers__available_from__address__region': 'Washington', 'offers__available_from__address__type': 'PostalAddress', 'offers__available_from__type': 'Place', 'product_id': '1951083', 'similar_pids': ['2113741', '2033980', '2034855'], 'state_code': 'WA'}, '1998714': {'asking_price': 900000.0, 'asking_price_currency': 'USD', 'category': 'Accounting and Tax Practices', 'chunk_type': 'profile', 'city': 'Portland', 'document': '# Listing Profile\n \nAsking Price (USD): 900000 \n\nReason for Selling: Approaching retirement ', 'fin__gross_revenue': 958000.0, 'location': 'Portland, OR', 'main_category': 'Accounting and Tax Practices', 'offer_type': 'Offer', 'offers__available_from__address__locality': 'Portland', 'offers__available_from__address__region': 'Oregon', 'offers__available_from__address__type': 'PostalAddress', 'offers__available_from__type': 'Place', 'product_id': '1998714', 'similar_pids': ['2026155', '2066311'], 'state_code': 'OR'}}
# update
st.session_state['above_thr_sorted'] = above_thr_sorted
st.session_state['results'] = results
st.session_state['metadata'] = metadata
st.session_state['display_results'] = True
else:
print("No matches found.")
metadata = {}
if st.session_state['display_results']:
summary_container = st.empty()
st.header("Results")
st.divider()
# display matched results
for pid, match_score in st.session_state['above_thr_sorted']:
if pid not in st.session_state['results']:
continue
metadata_pid = st.session_state['metadata'][pid]
result = st.session_state['results'][pid]
col_icon, col_info, col_compare = st.columns([2, 6, 1])
with col_icon:
st.image(result["logo"])
with col_info:
# TODO: make asking price display $xxx,xxx
st.markdown(f"""match score: { round(100 * match_score, 2) }
<br>
**{result['name']}**
<br>
_Asking Price:_ {metadata_pid.get('asking_price', 'N/A')}
<br>
_Category:_ {metadata_pid.get('category', 'N/A')}
<br>
_Location:_ {metadata_pid.get('location', 'N/A')}
""", unsafe_allow_html=True)
st.markdown(f"""**_Description:_** {result['description'][:MAX_LENGTH_DESC]}...[more]({result['url']})
""")
with col_compare:
st.checkbox('compare', key=f"cb_compare__{pid}", on_change=callback_count_checked)
# display summary tab
if st.session_state['count_checked'] > 0:
with summary_container.container():
st.header('Summary')
if st.button('Compare Products'):
products = []
for key in st.session_state['checked_boxes']:
# TODO: Need to pull all the document
# TODO: Need to dedup the pid too
pid = key.split('__')[-1]
products.append(
st.session_state['metadata'][pid].get('document')
)
with st.spinner('Summarizing...'):
summary = summarize_products(products)
st.markdown(summary.get("content"), unsafe_allow_html=True)
else:
try:
summary_container.empty()
except NameError:
pass
with st.expander("developer tool"):
st.json(st.session_state) |