-
Notifications
You must be signed in to change notification settings - Fork 2
/
02-parallel-ocr-with-ray.py
310 lines (216 loc) · 11.4 KB
/
02-parallel-ocr-with-ray.py
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
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
# Databricks notebook source
# MAGIC %md
# MAGIC # 02 - Parallel OCR with Ray on Databricks
# MAGIC This notebook converts your databricks cluster into a [Ray cluster](https://docs.databricks.com/en/machine-learning/ray-integration.html) for distributed processing. This allows us to use every core available on the cluster's worker node to perform OCR and chunking of PDF documents at the same time. In testing, you can observe a nearly linear scaling for every core added to the cluster. For example:
# MAGIC * _Traditional processing:_ OCR on average takes `3 minutes` per document, so `100 documents` would take approximately **`5 hours.`**
# MAGIC * _Ray on Databricks:_ OCR on average take `3 minutes` per document per core, so `100 documents` running on a cluster of `3 worker nodes` with `8 cores per node` takes approximately **`15 minutes.`**
# MAGIC
# MAGIC Note: in this example, there are some complex outlier documents that can take upwards of 30 minutes for OCR, so this makes the total processing time for 100 documents longer. But those numbers hold true for functions having more consistent processing speeds. These outliers would even more greatly affect processing times if running in series.
# COMMAND ----------
# MAGIC %md
# MAGIC ## init scripts
# MAGIC We utilize the [`unstructured`](https://unstructured.io/) library to perform OCR and chunking of pdf documents. Unstructured utilizes [tesseract](https://github.com/tesseract-ocr/tesseract) for OCR, which requires tesseract to be installed on every node of the cluster. In order to do that, we have to take advantage of databricks [init scripts](https://docs.databricks.com/en/init-scripts/index.html). Init scripts are run at cluster startup and are the best way to install os-level libraries before spark is started on each cluster.
# MAGIC
# MAGIC In this case, it is a very simple, two line file:
# MAGIC ```bash
# MAGIC apt update -y
# MAGIC apt-get install -y poppler-utils libmagic-dev tesseract-ocr
# MAGIC ```
# COMMAND ----------
# MAGIC %sql
# MAGIC -- CREATE VOLUME if not exists '/Volumes/yyang/ner/init'
# MAGIC CREATE VOLUME IF NOT EXISTS yyang.ner.init;
# COMMAND ----------
# DBTITLE 1,write into init script
#: python way
# with open("init_script.sh", "w") as file:
# file.write("apt update -y\n")
# file.write("apt-get install -y poppler-utils libmagic-dev tesseract-ocr\n")
# or
#: bash cat << EOF way
# %sh
# cat <<EOF > /Volumes/yyang/ner/init/init_script.sh
# #!/bin/bash
# apt update -y
# apt-get install -y poppler-utils libmagic-dev tesseract-ocr
# EOF
# COMMAND ----------
# DBTITLE 1,write into init script
# MAGIC %sh
# MAGIC echo -e '''#!/bin/bash
# MAGIC apt update -y
# MAGIC apt-get install -y poppler-utils libmagic-dev tesseract-ocr
# MAGIC ''' > /Volumes/yyang/ner/init/init_script.sh
# COMMAND ----------
# MAGIC %sh
# MAGIC cat /Volumes/yyang/ner/init/init_script.sh
# COMMAND ----------
# MAGIC %md
# MAGIC ### Now go to your Cluster UI and set "Advanced" -> "Init scripts",
# MAGIC 1. under it locate your `init_script.sh` under the correct path, save your setting and restart your cluster.
# MAGIC 2. come back to this notebook and start from the next cell (which means you only run the above steps once).
# COMMAND ----------
# DBTITLE 1,if you have single-node, you can uncomment below
# %sh
# apt update -y && apt-get install -y poppler-utils libmagic-dev tesseract-ocr
# # if you're working on a single node cluster, you can just run this line in the notebook instead of using an init script
# COMMAND ----------
# MAGIC %md
# MAGIC ## Update pip packges
# MAGIC ref: https://fabric.guru/to-pip-or-pip-install-python-libraries-in-a-spark-cluster
# COMMAND ----------
# MAGIC %md
# MAGIC Ray and Spark can coexist on a Databricks cluster, but careful resource management is necessary to prevent conflicts. Here are some key points to consider:
# MAGIC
# MAGIC Resource Allocation: Ensure that both Ray and Spark have sufficient resources (memory, CPU, and/or GPU) allocated to avoid contention. You can configure the number of worker nodes and CPUs allocated to Ray to manage this.
# MAGIC
# MAGIC Cluster Configuration: Use the setup_ray_cluster function to configure the Ray cluster. This allows you to specify the minimum and maximum number of worker nodes for Ray, ensuring that resources are appropriately divided between Ray and Spark.
# MAGIC
# MAGIC Avoid Over-subscribing: Avoid over-subscribing Ray cluster resources in third-party applications, as this can lead to instability.
# MAGIC
# MAGIC __Installation Timing: Avoid running %pip to install packages on a running Ray cluster, as it will shut down the cluster. Install necessary libraries before initializing the Ray cluster.__
# COMMAND ----------
# MAGIC %pip install 'databricks-sdk' unstructured[pdf] ray[default]>=2.3.0 nltk --upgrade
# MAGIC dbutils.library.restartPython()
# COMMAND ----------
# DBTITLE 1,setup correct Catalog.Schema path
# MAGIC %sql
# MAGIC USE CATALOG 'yyang';
# MAGIC USE SCHEMA ner;
# COMMAND ----------
# MAGIC %sql
# MAGIC SHOW TABLES;
# COMMAND ----------
# VOLUME = "/Volumes/yyang/ocr/data/"
VOLUME = "/Volumes/yyang/ner/data/"
# COMMAND ----------
import os
ray_logs_dir = os.path.join(VOLUME, "ray_logs")
os.makedirs(ray_logs_dir, exist_ok=True)
# COMMAND ----------
# MAGIC %md
# MAGIC Pull filenames from `papers` table into a list. We order by filesize descending to make sure we process the larger files first so we don't have the longest files processed at the very end of the parallel process.
# COMMAND ----------
filenames = spark.sql('SELECT filename FROM papers ORDER BY filesize DESC').collect()
filenames = [row.filename for row in filenames]
# COMMAND ----------
# MAGIC %md
# MAGIC To prepare for Ray cluster declaration, we pull the current number of worker nodes and cores using the [`databricks-sdk`](https://databricks-sdk-py.readthedocs.io/en/latest/)
# COMMAND ----------
from databricks.sdk import WorkspaceClient
cluster_id = spark.conf.get("spark.databricks.clusterUsageTags.clusterId")
w = WorkspaceClient()
cluster_info = w.clusters.get(cluster_id=cluster_id)
nodes = w.clusters.list_node_types()
min_worker_nodes = cluster_info.autoscale.min_workers
max_worker_nodes = cluster_info.autoscale.max_workers
num_cpus_worker_node = int([x.num_cores for x in nodes.node_types if x.node_type_id==cluster_info.node_type_id][0])
print(f"min_worker_nodes={min_worker_nodes}")
print(f"max_worker_nodes={max_worker_nodes}")
print(f"num_cpus_worker_node={num_cpus_worker_node}")
# COMMAND ----------
# from databricks.sdk import WorkspaceClient
# cluster_id = spark.conf.get("spark.databricks.clusterUsageTags.clusterId")
# w = WorkspaceClient()
# cluster_info = w.clusters.get(cluster_id=cluster_id)
# nodes = w.clusters.list_node_types()
# max_worker_nodes = len(cluster_info.executors)
# num_cpus_worker_node = int([x.num_cores for x in nodes.node_types if x.node_type_id==cluster_info.node_type_id][0])
# print(f"max_worker_nodes={max_worker_nodes}")
# print(f"num_cpus_worker_node={num_cpus_worker_node}")
# COMMAND ----------
# MAGIC %md
# MAGIC Start a Ray cluster on your databricks cluster. This allows you to broadcast a `ray.remote` function across the cluster to take advantage of processing on every core in the cluster. This temporarily overrides the spark functionality of the cluster. However, when you shutdown your Ray cluster, all of your native spark functionality comes back.
# MAGIC
# MAGIC I recommend clicking on the `Open Ray Cluster Dashboard in a new tab` link at the bottom of this cell's output to track the utilization/parallelization of the Ray cluster.
# COMMAND ----------
import ray
from ray.util.spark import setup_ray_cluster, shutdown_ray_cluster
try:
shutdown_ray_cluster()
print("Successfully shut down the currently running Ray cluster. Setting up a new one.")
except:
print("No current Ray cluster. Setting up a new one.")
setup_ray_cluster(
min_worker_nodes=min_worker_nodes,
max_worker_nodes=max_worker_nodes,
num_cpus_per_node=num_cpus_worker_node,
num_gpus_worker_node=0,
collect_log_to_path=ray_logs_dir
)
# setup_ray_cluster(
# max_worker_nodes=max_worker_nodes,
# num_cpus_worker_node=num_cpus_worker_node,
# num_gpus_worker_node=0
# )
# COMMAND ----------
#: (DONT RUN in notebook) Instead copy to terminal and run.
# %sh
# ray metrics launch-prometheus
# COMMAND ----------
# MAGIC %md
# MAGIC Declare a remote ray function to run [`unstructured`](https://unstructured.io/) paritioning and chunking functions in parallel across the cluster.
# COMMAND ----------
import pandas
import datetime as dt
from unstructured.partition.pdf import partition_pdf
from unstructured.chunking.title import chunk_by_title
from pyspark.sql.types import ArrayType, StringType
@ray.remote
def parse_pdf_ray(location: str) -> list:
stime = dt.datetime.now()
elements = partition_pdf(location, strategy="ocr_only")
# elements = partition_pdf(location, strategy="hi_res")
chunks = chunk_by_title(elements)
print(str(dt.datetime.now() - stime) + f" for {location}")
return [str(x) for x in chunks if len(str(x)) > 50]
# COMMAND ----------
stime = dt.datetime.now()
chunked_pdfs = [parse_pdf_ray.remote(loc) for loc in filenames]
chunks = ray.get(chunked_pdfs)
print("\n\n" + str(dt.datetime.now() - stime) + f" for all")
# COMMAND ----------
# MAGIC %md
# MAGIC Shutdown your Ray cluster after processing to return to expected spark functionality
# MAGIC
# MAGIC The spark operation example below will hang forever because all CPUs has been assigned to Ray cluster during `setup_ray_cluster` phase and guaranteed exclusive ownership.
# COMMAND ----------
# #: (Dont RUN) this will hang forever if you subscribed all the CPUs previously to Ray cluster.
# # the spark operation will hang forever because all CPUs has been assigned to Ray cluster during `setup_ray_cluster` phase
# spark.table("yyang.ner.papers").display()
# COMMAND ----------
# shutdown ray cluster
try:
shutdown_ray_cluster()
print("Successfully shut down the currently running Ray cluster.")
except:
print("No Ray cluster to shut down.")
# COMMAND ----------
# #: now spark functionality has been restored!
spark.table("yyang.ner.papers").display()
# COMMAND ----------
# MAGIC %md
# MAGIC Take the chunking results from your `ray.remote` function (stored in a list of lists) and join it with your `papers` dataframe.
# COMMAND ----------
from pyspark.sql.functions import monotonically_increasing_id, row_number, explode
from pyspark.sql import Window
papers = spark.sql("SELECT * FROM papers ORDER BY filesize DESC")
add_chunks_udf = udf(lambda i: chunks[i-1], ArrayType(StringType()))
papers = papers.withColumn("chunk_id", row_number().over(Window.orderBy(monotonically_increasing_id())))
papers = papers.withColumn("chunks", add_chunks_udf("chunk_id")).drop("chunk_id")
# COMMAND ----------
# MAGIC %md
# MAGIC Use [`explode`](https://spark.apache.org/docs/3.1.3/api/python/reference/api/pyspark.sql.functions.explode.html) to create a row for each chunk. This will allow you to apply your model to each chunk indivually and efficiently. Then save the table for use in subsequent notebooks.
# COMMAND ----------
chunks_df = (papers.withColumn("chunk", explode("chunks"))
.select(["entry_id","title","chunk"])
.write
.mode("overwrite")
.saveAsTable("chunks")
)
# COMMAND ----------
c = spark.sql("SELECT * FROM chunks")
c.count()
# COMMAND ----------
display(c)
# COMMAND ----------