Skip to content

Commit a9a8f6d

Browse files
committed
improve incremental sync
support updates of schema and rows
1 parent 6823d8e commit a9a8f6d

File tree

2 files changed

+35
-26
lines changed

2 files changed

+35
-26
lines changed

notion_docs_sync/__init__.py

+28-7
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import logging
22
import os
33
from random import choice
4+
from collections import OrderedDict
45
from argparse import ArgumentParser
56

67
from notion.client import NotionClient
@@ -94,6 +95,19 @@ def block_matches_markdown_block(block, markdown_block_type, **markdown_block):
9495
return True
9596

9697

98+
def sync_collection_schema(collection, expected_schema):
99+
existing_schema = collection.get('schema')
100+
101+
# The schemas must match!
102+
if existing_schema == expected_schema:
103+
return
104+
105+
logger.info(f"Updating schema of {collection.id}")
106+
107+
# If they don't, try to make them match.
108+
collection.set('schema', expected_schema)
109+
110+
97111
def sync_collection_rows(block, collection_schema, collection_rows):
98112
if block.collection is None:
99113
logger.info(f"Creating a new collection for {block.id}")
@@ -102,12 +116,17 @@ def sync_collection_rows(block, collection_schema, collection_rows):
102116
block.collection = client.get_collection(
103117
# Low-level use of the API
104118
# TODO: Update when notion-py provides a better interface for this
105-
client.create_record("collection", parent=block, schema=collection_schema)
119+
client.create_record("collection", parent=block, schema={"title": {"text": "_", "type": "text"}})
106120
)
107121

108122
block.views.add_new(view_type="table")
109123

110-
# TODO: Compare collection schema and update the collection if it's not matching.
124+
collection_schema_ids = ['title']
125+
126+
for i in range(len(collection_schema) - 1):
127+
collection_schema_ids.append('x' + format(i, '0>4x'))
128+
129+
sync_collection_schema(block.collection, dict(zip(collection_schema_ids, collection_schema)))
111130

112131
existing_rows = block.collection.get_rows()
113132

@@ -122,12 +141,14 @@ def sync_collection_rows(block, collection_schema, collection_rows):
122141
except StopIteration:
123142
row_block = block.collection.add_row()
124143

125-
for idx, prop_name in enumerate(prop["name"] for prop in collection_schema.values()):
126-
prop_name = prop_name.lower() # The actual prop name in notion-py is lowercase
127-
prop_val = row[idx]
144+
if len(row) > len(collection_schema_ids):
145+
row = row[:len(collection_schema_ids)]
146+
147+
row = zip(collection_schema_ids, row)
128148

129-
if getattr(row_block, prop_name) != prop_val:
130-
setattr(row_block, prop_name, prop_val)
149+
for schema_id, prop_value in row:
150+
if row_block.get_property(schema_id) != prop_value:
151+
row_block.set_property(schema_id, prop_value)
131152

132153

133154
def sync_markdown_blocks_to_block(markdown_blocks, block):

notion_docs_sync/markdown.py

+7-19
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,11 @@
11
import logging
2-
import random
32
import re
43
import collections
54
from notion.block import CodeBlock, DividerBlock, HeaderBlock, SubheaderBlock, \
65
SubsubheaderBlock, QuoteBlock, TextBlock, NumberedListBlock, \
76
BulletedListBlock, ImageBlock, CollectionViewBlock
87
import mistletoe
9-
import string
108
from mistletoe.base_renderer import BaseRenderer
11-
from mistletoe.span_token import Image, Link
129

1310

1411
logger = logging.getLogger(__name__)
@@ -231,22 +228,13 @@ def render_list_item(self, token):
231228
}
232229

233230
def render_table(self, token):
234-
header_row = self.render(token.header) #Header is a single row
235-
rows = [self.render(r) for r in token.children] #don't use renderMultiple because it flattens
236-
237-
def random_column_id():
238-
return ''.join(random.choices(string.ascii_letters + string.digits, k=4))
239-
240-
# The schema is basically special identifiers + the type of property to put into Notion.
241-
schema = { random_column_id() : { 'name' : header_row[r], 'type': 'text' }for r in range(len(header_row) - 1) }
242-
243-
# However, The last one needs to be named 'Title' and is type title
244-
schema.update({
245-
'title' : {
246-
'name': header_row[-1],
247-
'type': 'title'
248-
}
249-
})
231+
header_row = self.render(token.header)
232+
rows = [self.render(r) for r in token.children]
233+
234+
schema = []
235+
236+
for row in header_row:
237+
schema.append({"name": row, "type": "text"})
250238

251239
return {
252240
'type': CollectionViewBlock,

0 commit comments

Comments
 (0)