-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathPhotoDeckAPI.lua
1341 lines (1206 loc) · 57 KB
/
PhotoDeckAPI.lua
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
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
local LrDate = import 'LrDate'
local LrDigest = import 'LrDigest'
local LrFileUtils = import 'LrFileUtils'
local LrHttp = import 'LrHttp'
local LrStringUtils = import 'LrStringUtils'
local LrXml = import 'LrXml'
local LrTasks = import 'LrTasks'
local PhotoDeckUtils = require 'PhotoDeckUtils'
local PhotoDeckAPIXSLT = require 'PhotoDeckAPIXSLT'
local logger = import 'LrLogger'( 'PhotoDeckPublishLightroomPlugin' )
logger:enable('logfile')
local PhotoDeckAPI_BASEURL = 'https://api.photodeck.com'
local PhotoDeckMY_BASEURL = 'https://my.photodeck.com'
local PhotoDeckAPI_SESSIONCOOKIE = '_ficelle_session'
local PhotoDeckAPI_KEY = ''
local PhotoDeckAPI_SECRET = ''
local isTable = PhotoDeckUtils.isTable
local printTable = PhotoDeckUtils.printTable
local PhotoDeckAPI = {
hasDistributionKeys = PhotoDeckAPI_KEY and PhotoDeckAPI_KEY ~= '',
key = '',
secret = '',
password = '',
loggedin = false,
sessionCookie = nil,
canSynchronize = true
}
local PhotoDeckAPICache = {}
local canRequestUploadLocation = true
-- sign API request according to docs at
-- http://www.photodeck.com/developers/get-started/
local function sign(method, uri, querystring)
querystring = querystring or ''
local cocoatime = LrDate.currentTime()
local timestamp = LrDate.timeToW3CDate(cocoatime)
local request = string.format('%s\n%s\n%s\n%s\n%s\n', method, uri,
querystring, PhotoDeckAPI.secret, timestamp)
local signature = PhotoDeckAPI.key .. ':' .. LrDigest.SHA1.digest(request)
-- logger:trace(timestamp)
-- logger:trace(signature)
return {
{ field = 'X-PhotoDeck-TimeStamp', value=timestamp },
{ field = 'X-PhotoDeck-Authorization', value=signature },
}
end
local function auth_headers(method, uri, querystring)
-- sign request
local headers = sign(method, uri, querystring)
-- set login cookies
if PhotoDeckAPI.username and PhotoDeckAPI.password and not PhotoDeckAPI.loggedin then
-- not logged in, send HTTP Basic credentials
local authorization = 'Basic ' .. LrStringUtils.encodeBase64(PhotoDeckAPI.username ..
':' .. PhotoDeckAPI.password)
table.insert(headers, { field = 'Authorization', value = authorization })
elseif PhotoDeckAPI.sessionCookie then
-- already logged in, inject last known session cookie.
-- NOTE: Lightroom usually does this by itself, but some installations seems to loose cookies between calls. So we are doing this manually.
local cookie = PhotoDeckAPI_SESSIONCOOKIE .. '=' .. PhotoDeckAPI.sessionCookie
table.insert(headers, { field = 'Cookie', value = cookie })
end
return headers
end
-- extra chars from http://tools.ietf.org/html/rfc3986#section-2.2
local function urlencode(s)
s = string.gsub(s, "([][:/?#@!#'()*,;&=+%%%c])", function (c)
return string.format("%%%02X", string.byte(c))
end)
s = string.gsub(s, " ", "+")
return s
end
-- convert lua table to url encoded data
-- from http://www.lua.org/pil/20.3.html
local function table_to_querystring(data)
assert(PhotoDeckUtils.isTable(data))
local s = ""
for k,v in pairs(data) do
s = s .. "&" .. urlencode(k) .. "=" .. urlencode(v)
end
return string.sub(s, 2) -- remove first `&'
end
-- Makes sure that we don't call the API more than once every second starting from the 5th request in a row.
-- This is done to avoid hitting rate limits on the PhotoeDeck API and throwing errors
local resume_requests_at = 0
local last_request_at = 0
local consecutive_requests = 0
local function throttle_request()
local now = LrDate.currentTime()
local sleep_for = resume_requests_at - now
if sleep_for > 0 then
logger:trace(string.format(' ** Sleeping for %.2f seconds as we have previously hit a rate limit', sleep_for))
LrTasks.sleep(sleep_for)
now = LrDate.currentTime()
end
resume_requests_at = 0
local elapsed = now - last_request_at
if elapsed < 30 then
consecutive_requests = consecutive_requests + 1
if consecutive_requests > 4 and elapsed < 1 then
sleep_for = 1 - elapsed
logger:trace(string.format(' ** Sleeping for %.2f seconds to keep request rate at 1/sec max', sleep_for))
LrTasks.sleep(sleep_for)
now = LrDate.currentTime()
end
else
consecutive_requests = 1
end
last_request_at = now
end
local function handle_response(seq, response, resp_headers, onerror)
local status = PhotoDeckUtils.filter(resp_headers, function(v) return isTable(v) and v.field == 'Status' end)[1]
local request_id = PhotoDeckUtils.filter(resp_headers, function(v) return isTable(v) and v.field == 'X-Request-Id' end)[1]
local error_msg = nil
local status_code = "999"
if request_id then
request_id = request_id.value
else
request_id = "No request ID"
end
if resp_headers.status then
-- Get HTTP response code
status_code = tostring(resp_headers.status)
end
if status then
-- Get status from Status header, if any
status_code = string.sub(status.value, 1, 3)
end
if status_code >= "400" then
if status then
-- Get error from Status header
error_msg = status.value
if status_code == "429" then
-- Too Many Requests. Wait 30 seconds until next request.
-- Note: this HTTP error seems to be filtered out on LR/Windows at a lower level (the error will get catched in the status_code = "999" case)
resume_requests_at = LrDate.currentTime() + 30
end
else
-- Generic HTTP error
if status_code == "999" then
error_msg = LOC("$$$/PhotoDeck/API/UnknownError=Unknwon error")
else
error_msg = LOC("$$$/PhotoDeck/API/HTTPError=HTTP error ^1", status_code)
end
end
if not response and status_code == "999" then
error_msg = LOC("$$$/PhotoDeck/API/NoResponse=No response from network")
-- No network connection, or we are blocked. Wait 30 seconds until next request.
resume_requests_at = LrDate.currentTime() + 30
end
local error_msg_from_xml = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.error)
if error_msg_from_xml and error_msg_from_xml ~= "" then
-- We got an error from the API, use that error message instead
error_msg = error_msg_from_xml
end
if onerror and onerror[status_code] then
logger:trace(string.format(' %s <- %s [%s] (handled by onerror)', seq, status_code, request_id))
return onerror[status_code]()
end
--logger:error("Bad response: " .. error_msg .. " => " .. (response or "(no response)"))
--if resp_headers then
-- logger:error(PhotoDeckUtils.printLrTable(resp_headers))
--end
if status_code == "401" or status_code == "999" then
PhotoDeckAPI.loggedin = false
PhotoDeckAPI.sessionCookie = nil
end
if status_code == "999" then
logger:error(string.format(' %s <- %s [%s]: %s %s', seq, status_code, request_id, error_msg, printTable(resp_headers)))
else
logger:error(string.format(' %s <- %s [%s]: %s', seq, status_code, request_id, error_msg))
end
else
PhotoDeckAPI.loggedin = true
logger:trace(string.format(' %s <- %s [%s]', seq, status_code, request_id))
-- Try to extract session cookie. We will reinject it later, as it seems that some Lightroom installations loose cookies between calls.
for _, set_cookie in ipairs(PhotoDeckUtils.filter(resp_headers, function(v) return isTable(v) and v.field == 'Set-Cookie' end)) do
local parsed_cookies = LrHttp.parseCookie(set_cookie.value, false)
if parsed_cookies[PhotoDeckAPI_SESSIONCOOKIE] then
PhotoDeckAPI.sessionCookie = parsed_cookies[PhotoDeckAPI_SESSIONCOOKIE]
end
end
end
return response, error_msg
end
-- make HTTP GET request to PhotoDeck API
-- must be called within an LrTask
function PhotoDeckAPI.request(method, uri, data, onerror)
local querystring = ''
local body = ''
local error_msg
if data then
if method == 'GET' then
querystring = table_to_querystring(data)
else
body = table_to_querystring(data)
end
end
-- set up authorisation headers
local headers = auth_headers(method, uri, querystring)
-- build full url
local fullurl = PhotoDeckAPI_BASEURL .. uri
if querystring and querystring ~= '' then
fullurl = fullurl .. '?' .. querystring
end
-- call API
throttle_request()
local result, resp_headers
local seq = string.format("%5i", math.random(99999))
if method == 'GET' then
logger:trace(string.format(' %s -> %s %s', seq, method, fullurl))
result, resp_headers = LrHttp.get(fullurl, headers)
else
-- override default Content-Type!
logger:trace(string.format(' %s -> %s %s\n%s', seq, method, fullurl, body))
table.insert(headers, { field = 'Content-Type', value = 'application/x-www-form-urlencoded'})
result, resp_headers = LrHttp.post(fullurl, body, headers, method)
end
result, error_msg = handle_response(seq, result, resp_headers, onerror)
return result, error_msg
end
function PhotoDeckAPI.requestMultiPart(method, uri, content, onerror)
local error_msg
local seq = string.format("%5i", math.random(99999))
logger:trace(string.format(' %s -> %s[multipart] %s', seq, method, uri))
if method ~= "POST" then
-- LrHttp doesn't implement non-POSTs multipart requests:
-- POST content but pass the correct method to the PhotoDeck API as a field
table.insert(content, { name = "_method", value = method })
method = "POST"
end
-- set up authorisation headers
local headers = auth_headers(method, uri)
-- build full url
local fullurl = PhotoDeckAPI_BASEURL .. uri
-- call API
throttle_request()
local result, resp_headers
result, resp_headers = LrHttp.postMultipart(fullurl, content, headers)
result, error_msg = handle_response(seq, result, resp_headers, onerror)
return result, error_msg
end
function PhotoDeckAPI.connect(key, secret, username, password)
if PhotoDeckAPI.hasDistributionKeys then
-- use builtin keys
PhotoDeckAPI.key = PhotoDeckAPI_KEY
PhotoDeckAPI.secret = PhotoDeckAPI_SECRET
else
-- use the one supplied by the user
PhotoDeckAPI.key = key
PhotoDeckAPI.secret = secret
end
if PhotoDeckAPI.loggedin and PhotoDeckAPI.username ~= username then
PhotoDeckAPI.logout()
end
PhotoDeckAPI.username = username
PhotoDeckAPI.password = password
PhotoDeckAPI.loggedin = false
PhotoDeckAPI.sessionCookie = nil
end
function PhotoDeckAPI.ping(text)
logger:trace('PhotoDeckAPI.ping()')
local t = {}
if text then
t = { text = text }
end
local response, error_msg = PhotoDeckAPI.request('GET', '/ping.xml', t)
local result = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.ping)
return result, error_msg
end
function PhotoDeckAPI.logout()
logger:trace('PhotoDeckAPI.logout()')
local response, error_msg = PhotoDeckAPI.request('GET', '/logout.xml')
return response, error_msg
end
function PhotoDeckAPI.whoami()
logger:trace('PhotoDeckAPI.whoami()')
local response, error_msg = PhotoDeckAPI.request('GET', '/whoami.xml')
local result = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.user)
if not result or not result.email or result.email == '' then
PhotoDeckAPI.loggedin = false
PhotoDeckAPI.sessionCookie = nil
end
-- logger:trace(printTable(result))
return result, error_msg
end
function PhotoDeckAPI.websites()
logger:trace('PhotoDeckAPI.websites()')
local cacheKey = 'websites/' .. PhotoDeckAPI.username
local result = PhotoDeckAPICache[cacheKey]
local response, error_msg = nil
if not result then
response, error_msg = PhotoDeckAPI.request('GET', '/websites.xml', { view = 'details' })
result = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.websites)
if not error_msg then
local websites_count = 0
if result then
for _ in pairs(result) do websites_count = websites_count + 1 end
end
if websites_count == 0 then
error_msg = LOC("$$$/PhotoDeck/API/Websites/NotFound=No websites found")
end
end
if error_msg then
PhotoDeckAPI.loggedin = false
PhotoDeckAPI.sessionCookie = nil
else
PhotoDeckAPICache[cacheKey] = result
end
-- logger:trace(printTable(result))
end
return result, error_msg
end
function PhotoDeckAPI.website(urlname)
local websites, error_msg = PhotoDeckAPI.websites()
local website = nil
if not error_msg then
website = websites[urlname]
if not website then
error_msg = LOC("$$$/PhotoDeck/API/Website/NotFound=Website not found")
end
end
return website, error_msg
end
function PhotoDeckAPI.galleries(urlname)
logger:trace(string.format('PhotoDeckAPI.galleries("%s")', urlname))
local galleries
local allgalleries = {}
local response
local error_msg = nil
local page = 0
local totalPages = 1
local newTotalPages
while not error_msg and page < totalPages do
page = page + 1
response, error_msg = PhotoDeckAPI.request('GET', '/websites/' .. urlname .. '/galleries.xml', { view = 'details', page = page, per_page = 500 })
galleries = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.galleries)
newTotalPages = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.totalPages)
if newTotalPages and newTotalPages ~= "" then
totalPages = tonumber(newTotalPages)
end
--logger:trace("PhotoDeckAPI.galleries " .. tostring(page) .. "/" .. tostring(totalPages) .. ": " .. printTable(galleries))
if not galleries and not error_msg then
error_msg = LOC("$$$/PhotoDeck/API/Gallery/ErrorGettingGalleries=Couldn't get galleries")
end
if not error_msg then
local added_count = 0
for uuid, gallery in pairs(galleries) do
added_count = added_count + 1
allgalleries[uuid] = gallery
end
if added_count == 0 then break end
end
end
if not error_msg then
--logger:trace("PhotoDeckAPI.galleries: " .. printTable(allgalleries))
return allgalleries
else
return nil, error_msg
end
end
function PhotoDeckAPI.gallery(urlname, galleryId, ignore_not_found)
logger:trace(string.format('PhotoDeckAPI.gallery("%s", "%s")', urlname, galleryId))
local onerror = {}
if ignore_not_found then
onerror["404"] = function() return nil end
end
local response, error_msg = PhotoDeckAPI.request('GET', '/websites/' .. urlname .. '/galleries/' .. galleryId .. '.xml', { view = 'details' }, onerror)
local result = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.gallery)
-- logger:trace(printTable(result))
return result, error_msg
end
function PhotoDeckAPI.openGalleryInBackend(galleryId)
logger:trace(string.format('PhotoDeckAPI.openGalleryInBackend("%s")', galleryId))
LrHttp.openUrlInBrowser(PhotoDeckMY_BASEURL .. '/medias/manage?gallery_id=' .. galleryId)
end
local function buildGalleryInfoFromLrCollectionInfo(collectionInfo)
local galleryInfo = {}
galleryInfo['gallery[name]'] = collectionInfo.name
local collectionSettings = collectionInfo.collectionSettings
if collectionSettings then
galleryInfo['gallery[description]'] = collectionSettings['description']
galleryInfo['gallery[display_style]'] = collectionSettings['display_style']
end
return galleryInfo
end
function PhotoDeckAPI.createGallery(urlname, parentId, collectionInfo)
logger:trace(string.format('PhotoDeckAPI.createGallery("%s", "%s", <collectionInfo>)', urlname, parentId))
local galleryInfo = buildGalleryInfoFromLrCollectionInfo(collectionInfo)
galleryInfo['gallery[content_order]'] = 'manual-last'
galleryInfo['gallery[parent]'] = parentId
local response, error_msg = PhotoDeckAPI.request('POST', '/websites/' .. urlname .. '/galleries.xml', galleryInfo)
local gallery = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.gallery)
return gallery, error_msg
end
function PhotoDeckAPI.updateGallery(urlname, galleryId, parentId, collectionInfo)
logger:trace(string.format('PhotoDeckAPI.updateGallery("%s", "%s", "%s", <collectionInfo>)', urlname, galleryId, parentId))
local galleryInfo = buildGalleryInfoFromLrCollectionInfo(collectionInfo)
galleryInfo['gallery[parent]'] = parentId
local response, error_msg = PhotoDeckAPI.request('PUT', '/websites/' .. urlname .. '/galleries/' .. galleryId .. '.xml', galleryInfo)
local gallery = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.gallery)
return gallery, error_msg
end
function PhotoDeckAPI.createOrUpdateGallery(urlname, collectionInfo, updateSettings)
logger:trace(string.format('PhotoDeckAPI.createOrUpdateGallery("%s", <collectionInfo>)', urlname))
local website, error_msg = PhotoDeckAPI.website(urlname)
if error_msg then
return nil, error_msg
end
local collection = collectionInfo.publishedCollection
local parentGalleryId = nil
local parentJustCreated = false
-- Find PhotoDeck gallery and parent gallery
local gallery = nil
local galleryId = collection:getRemoteId()
if galleryId then
-- find by remote ID if known
gallery = PhotoDeckAPI.gallery(urlname, galleryId, true)
if gallery then
local parentsCount = 0
-- check LR parent and see if the PD gallery is still properly connected
for _, parent in pairs(collectionInfo.parents) do
parentsCount = parentsCount + 1
if parent.remoteCollectionId == gallery.parentuuid then
-- ok, found, no need to go back to all parents one by one to reconnect everything
parentGalleryId = gallery.parentuuid
break
end
end
if parentsCount == 0 then
-- top level gallery
parentGalleryId = website.rootgalleryuuid
end
end
end
if not gallery or not parentGalleryId then
-- Find PhotoDeck parent galleries, create if missing and connect them to Lightroom if not already done
-- Start from the root gallery
parentGalleryId = website.rootgalleryuuid
if not parentGalleryId or parentGalleryId == "" then
return nil, LOC("$$$/PhotoDeck/API/Galleries/RootNotFound=Couldn't find PhotoDeck root gallery")
end
-- Now iterate over each parent, starting from the top level
for _, parent in pairs(collectionInfo.parents) do
local parentGallery = nil
local parentId = parent.remoteCollectionId
if parentId then
-- find by remote ID if known
parentGallery = PhotoDeckAPI.gallery(urlname, parentId, true)
end
if not parentGallery and not parentJustCreated then
-- not found, search by name within subgalleries present in our parent
-- (unless we have just created this gallery, in which case we assume that it's empty)
local subgalleries, error_msg = PhotoDeckAPI.subGalleriesInGallery(urlname, parentGalleryId)
if error_msg then
return nil, error_msg
end
for uuid, subgallery in pairs(subgalleries) do
if subgallery.name == parent.name then
parentGallery, error_msg = PhotoDeckAPI.gallery(urlname, uuid)
if error_msg or not parentGallery then
return nil, error_msg or LOC("$$$/PhotoDeck/API/Gallery/SubGalleryNotFound=Couldn't get subgallery")
end
break
end
end
end
if not parentGallery then
-- not found, create
parentGallery, error_msg = PhotoDeckAPI.createGallery(urlname, parentGalleryId, parent:getCollectionInfoSummary())
if error_msg then
return nil, error_msg
end
parentJustCreated = true
else
parentJustCreated = false
end
local parentCollection = collection.catalog:getPublishedCollectionByLocalIdentifier(parent.localCollectionId)
parentGallery.fullurl = website.homeurl .. "/-/" .. parentGallery.fullurlpath
if parentCollection and (not parent.remoteCollectionId or parentCollection:getRemoteId() ~= parent.remoteCollectionId or parentCollection:getRemoteUrl() ~= parentGallery.fullurl) then
--logger:trace('Updating parent remote Id and Url')
parentCollection.catalog:withWriteAccessDo('Set Parent Remote Id and Url', function()
parentCollection:setRemoteId(parentGallery.uuid)
parentCollection:setRemoteUrl(parentGallery.fullurl)
end)
end
parentGalleryId = parentGallery.uuid -- our parent gallery is now this one
end
-- now search by name within subgalleries present in our parent
-- (unless we have just created the parent gallery, in which case we assume that it's empty)
if not parentJustCreated then
local subgalleries, error_msg = PhotoDeckAPI.subGalleriesInGallery(urlname, parentGalleryId)
if error_msg then
return nil, error_msg
end
for uuid, subgallery in pairs(subgalleries) do
if subgallery.name == collectionInfo.name then
gallery, error_msg = PhotoDeckAPI.gallery(urlname, uuid)
if error_msg or not gallery then
return nil, error_msg or LOC("$$$/PhotoDeck/API/Gallery/SubGalleryNotFound=Couldn't get subgallery")
end
break
end
end
end
end
if gallery then
-- PhotoDeck gallery found, update if necessary
local changed = gallery.parentuuid ~= parentGalleryId or gallery.name ~= collectionInfo.name
local collectionSettings
local settingsChanged = false
if updateSettings then
-- User has edited the gallery settings (ie, description and/or display style), so update gallery if changed
collectionSettings = collectionInfo.collectionSettings
if collectionSettings then
if not settingsChanged and collectionSettings['description'] and collectionSettings['description'] ~= '' then
settingsChanged = collectionSettings['description'] ~= gallery.description
end
if not settingsChanged and collectionSettings['display_style'] and collectionSettings['display_style'] ~= '' then
settingsChanged = collectionSettings['display_style'] ~= gallery.displaystyle
end
changed = changed or settingsChanged
end
end
if changed then
gallery, error_msg = PhotoDeckAPI.updateGallery(urlname, gallery.uuid, parentGalleryId, collectionInfo)
if not error_msg and settingsChanged then
-- resynchronize LR published collection settings with the actual data in PhotoDeck
collectionSettings['description'] = gallery.description
collectionSettings['display_style'] = gallery.displaystyle
collection.catalog:withWriteAccessDo('Resynchronize LR collection settings', function()
if collection:type() == 'LrPublishedCollection' then
collection:setCollectionSettings(collectionSettings)
elseif collection:type() == 'LrPublishedCollectionSet' then
collection:setCollectionSetSettings(collectionSettings)
end
end)
end
end
else
-- PhotoDeck gallery not found, create
gallery, error_msg = PhotoDeckAPI.createGallery(urlname, parentGalleryId, collectionInfo)
end
if error_msg then
return gallery, error_msg
end
gallery.fullurl = website.homeurl .. "/-/" .. gallery.fullurlpath
if collection:getRemoteId() == nil or collection:getRemoteId() ~= gallery.uuid or
collection:getRemoteUrl() ~= gallery.fullurl then
--logger:trace('Updating collection remote Id and Url')
collection.catalog:withWriteAccessDo('Set Remote Id and Url', function()
collection:setRemoteId(gallery.uuid)
collection:setRemoteUrl(gallery.fullurl)
end)
end
return gallery
end
function PhotoDeckAPI.synchronizeGalleries(urlname, publishService, progressScope)
logger:trace(string.format('PhotoDeckAPI.synchronizeGalleries("%s", <publishService>, <progressScope>)', urlname))
local catalog = publishService.catalog
if not PhotoDeckAPI.canSynchronize then
return nil, LOC("$$$/PhotoDeck/SynchronizeStatus/AlreadyInProgress=Task already in progress")
end
PhotoDeckAPI.canSynchronize = false
local createCount = 0
local deleteCount = 0
local updateCount = 0
local errorsCount = 0
progressScope:setCaption(LOC("$$$/PhotoDeck/SynchronizeStatus/Connecting=Connecting to PhotoDeck website"))
local website = PhotoDeckAPI.website(urlname)
local rootGalleryId = nil
if website then
rootGalleryId = website.rootgalleryuuid
end
if not rootGalleryId then
PhotoDeckAPI.canSynchronize = true
return nil, LOC("$$$/PhotoDeck/API/Galleries/RootNotFound=Couldn't find PhotoDeck root gallery")
end
progressScope:setCaption(LOC("$$$/PhotoDeck/SynchronizeStatus/ReadingStructure=Reading PhotoDeck gallery structure"))
local photodeckGalleries, error_msg = PhotoDeckAPI.galleries(urlname)
if not photodeckGalleries or error_msg then
PhotoDeckAPI.canSynchronize = true
return nil, error_msg or LOC("$$$/PhotoDeck/SynchronizeStatus/ErrorReadingStructure=Couldn't get PhotoDeck gallery structure")
end
local photodeckGalleriesByParent = {}
for uuid, gallery in pairs(photodeckGalleries) do
if gallery.parentuuid == '' or not gallery.parentuuid then
gallery.parentuuid = 'NONE'
end
if not photodeckGalleriesByParent[gallery.parentuuid] then
photodeckGalleriesByParent[gallery.parentuuid] = {}
end
photodeckGalleriesByParent[gallery.parentuuid][uuid] = gallery
end
local synchronizeGallery
synchronizeGallery = function(depth, parentPDGalleryUUID, parentLRCollectionSet)
if progressScope:isCanceled() then return end
local parentPDGallery = photodeckGalleries[parentPDGalleryUUID]
progressScope:setCaption(LOC("$$$/PhotoDeck/SynchronizeStatus/Synchronizing=Synchronizing ^1", parentPDGallery.name))
logger:trace(string.format("SYNC: Exploring PhotoDeck galleries under %s '%s' at depth %i", parentPDGalleryUUID, parentPDGallery.name, depth))
local pdGalleries = photodeckGalleriesByParent[parentPDGalleryUUID] or {}
local lrCollectionSets = parentLRCollectionSet:getChildCollectionSets()
local lrCollections = parentLRCollectionSet:getChildCollections()
for uuid, gallery in pairs(pdGalleries) do
gallery.fullurl = website.homeurl .. '/-/' .. gallery.fullurlpath
end
-- Scan Lightroom published collections, and connect them to PhotoDeck galleries
local lrCollectionsByRemoteId = {}
for _, lrCollection in pairs(lrCollections) do
local rid = lrCollection:getRemoteId()
if not rid or rid == '' then
-- unconnected published collection, try to connect by name
local lrCollectionName = lrCollection:getName()
for uuid, gallery in pairs(pdGalleries) do
if lrCollectionName == gallery.name then
-- found matching gallery
local collectionSettings = lrCollection:getCollectionInfoSummary().collectionSettings or {}
collectionSettings.description = gallery.description
collectionSettings.display_style = gallery.displaystyle
catalog:withWriteAccessDo('Resynchronize LR collection settings', function()
lrCollection:setRemoteId(uuid)
lrCollection:setRemoteUrl(gallery.fullurl)
lrCollection:setCollectionSettings(collectionSettings)
end)
updateCount = updateCount + 1
rid = uuid
break
end
end
end
local gallery = pdGalleries[rid]
if not gallery or gallery.parentuuid ~= parentPDGalleryUUID then
logger:trace(string.format("SYNC: Lightroom Published Collection %i '%s' is connected to PhotoDeck gallery %s, but it doesn't exist anymore. Deleting Published Collection.", lrCollection.localIdentifier, lrCollection:getName(), rid or '(none)'))
catalog:withWriteAccessDo('Deleting Published Collection', function()
lrCollection:delete()
end)
deleteCount = deleteCount + 1
elseif lrCollectionsByRemoteId[rid] then
-- duplicate LR collections!
local lrCollectionDup = lrCollectionsByRemoteId[rid]
if gallery.name == lrCollectionDup:getName() then
logger:trace(string.format("SYNC: Lightroom Published Collection %i '%s' is connected to PhotoDeck gallery %s '%s', but we already have Published Collection %i '%s' connected to it. Deleting the former.", lrCollection.localIdentifier, lrCollection:getName(), rid, gallery.name, lrCollectionDup.localIdentifier, lrCollectionDup:getName()))
catalog:withWriteAccessDo('Deleting Published Collection', function()
lrCollection:delete()
end)
deleteCount = deleteCount + 1
else
logger:trace(string.format("SYNC: Lightroom Published Collection %i '%s' is connected to PhotoDeck gallery %s '%s', but we already have Published Collection %i '%s' connected to it. Deleting the later.", lrCollection.localIdentifier, lrCollection:getName(), rid, gallery.name, lrCollectionDup.localIdentifier, lrCollectionDup:getName()))
catalog:withWriteAccessDo('Deleting Published Collection', function()
lrCollectionDup:delete()
end)
deleteCount = deleteCount + 1
end
else
lrCollectionsByRemoteId[rid] = lrCollection
end
end
-- Scan Lightroom published collections sets, and connect them to PhotoDeck galleries
local lrCollectionSetsByRemoteId = {}
for _, lrCollectionSet in pairs(lrCollectionSets) do
local rid = lrCollectionSet:getRemoteId()
if not rid or rid == '' then
-- unconnected published collection, try to connect by name
local lrCollectionSetName = lrCollectionSet:getName()
for uuid, gallery in pairs(pdGalleries) do
if lrCollectionSetName == gallery.name then
-- found matching gallery
local collectionSettings = lrCollectionSet:getCollectionSetInfoSummary().collectionSettings or {}
collectionSettings.description = gallery.description
collectionSettings.display_style = gallery.displaystyle
catalog:withWriteAccessDo('Resynchronize LR collection settings', function()
lrCollectionSet:setRemoteId(uuid)
lrCollectionSet:setRemoteUrl(gallery.fullurl)
lrCollectionSet:setCollectionSetSettings(collectionSettings)
end)
updateCount = updateCount + 1
rid = uuid
break
end
end
end
local gallery = pdGalleries[rid]
if not gallery or gallery.parentuuid ~= parentPDGalleryUUID then
logger:trace(string.format("SYNC: Lightroom Published Collection Set %i '%s' is connected to PhotoDeck gallery %s, but it doesn't exist anymore. Deleting Published Collection Set.", lrCollectionSet.localIdentifier, lrCollectionSet:getName(), rid or '(none)'))
catalog:withWriteAccessDo('Deleting Published Collection Set', function()
lrCollectionSet:delete()
end)
deleteCount = deleteCount + 1
elseif lrCollectionSetsByRemoteId[rid] then
-- duplicate LR collections sets!
local lrCollectionSetd = lrCollectionSetsByRemoteId[rid]
if gallery.name == lrCollectionSetd:getName() then
logger:trace(string.format("SYNC: Lightroom Published Collection Set %i '%s' is connected to PhotoDeck gallery %s '%s', but we already have Published Collection Set %i '%s' connected to it. Deleting the former.", lrCollectionSet.localIdentifier, lrCollectionSet:getName(), rid, gallery.name, lrCollectionSetd.localIdentifier, lrCollectionSetd:getName()))
catalog:withWriteAccessDo('Deleting Published Collection Set', function()
lrCollectionSet:delete()
end)
deleteCount = deleteCount + 1
else
logger:trace(string.format("SYNC: Lightroom Published Collection Set %i '%s' is connected to PhotoDeck gallery %s '%s', but we already have Published Collection Set %i '%s' connected to it. Deleting the later.", lrCollectionSet.localIdentifier, lrCollectionSet:getName(), rid, gallery.name, lrCollectionSetd.localIdentifier, lrCollectionSetd:getName()))
catalog:withWriteAccessDo('Deleting Published Collection Set', function()
lrCollectionSetd:delete()
end)
deleteCount = deleteCount + 1
end
else
lrCollectionSetsByRemoteId[rid] = lrCollectionSet
end
end
-- Find missing Lightroom published collections / collection sets
for uuid, gallery in pairs(pdGalleries) do
if progressScope:isCanceled() then return end
local lrCollectionSet = lrCollectionSetsByRemoteId[uuid]
local lrCollection = lrCollectionsByRemoteId[uuid]
local shouldBeACollectionSet = photodeckGalleriesByParent[uuid]
local shouldBeACollection = not shouldBeACollectionSet and gallery.mediascount and gallery.mediascount ~= '' and tonumber(gallery.mediascount) > 0
if lrCollection and shouldBeACollectionSet then
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' is connected to Lightroom Published Collection %i, but it should be Publish Collection Set. Deleting Published Collection.", uuid, gallery.name, lrCollection.localIdentifier))
catalog:withWriteAccessDo('Deleting Published Collection', function()
lrCollection:delete()
end)
deleteCount = deleteCount + 1
lrCollection = nil
end
if lrCollectionSet and shouldBeACollection then
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' is connected to Lightroom Published Collection Set %i, but it should be Publish Collection. Deleting Published Collection Set.", uuid, gallery.name, lrCollectionSet.localIdentifier))
catalog:withWriteAccessDo('Deleting Published Collection Set', function()
lrCollectionSet:delete()
end)
deleteCount = deleteCount + 1
lrCollectionSet = nil
end
if lrCollection and lrCollectionSet then
-- exists has both a Lightroom Published Collection and Published Collection Set. Choose the right type and delete the other.
if shouldBeACollectionSet then
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' is connected to Lightroom Published Collection Set %i AND to Lightroom Published Collection %i, but it should be Publish Collection Set. Deleting Published Collection.", uuid, gallery.name, lrCollectionSet.localIdentifier, lrCollection.localIdentifier))
catalog:withWriteAccessDo('Deleting Published Collection', function()
lrCollection:delete()
end)
deleteCount = deleteCount + 1
lrCollection = nil
elseif shouldBeACollection then
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' is connected to Lightroom Published Collection Set %i AND to Lightroom Published Collection %i, but it should be Publish Collection. Deleting Published Collection Set.", uuid, gallery.name, lrCollectionSet.localIdentifier, lrCollection.localIdentifier))
catalog:withWriteAccessDo('Deleting Published Collection Set', function()
lrCollectionSet:delete()
end)
deleteCount = deleteCount + 1
lrCollectionSet = nil
else
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' is connected to Lightroom Published Collection Set %i AND to Lightroom Published Collection %i, and we don't know yet what it should be. Assuming Published Collection, and deleting Published Collection Set.", uuid, gallery.name, lrCollectionSet.localIdentifier, lrCollection.localIdentifier))
catalog:withWriteAccessDo('Deleting Published Collection', function()
lrCollection:delete()
end)
deleteCount = deleteCount + 1
lrCollectionSet = nil
end
end
if lrCollectionSet then
-- Already properly connected, good
local collectionSettings = lrCollectionSet:getCollectionSetInfoSummary().collectionSettings or {}
if lrCollectionSet:getRemoteUrl() ~= gallery.fullurl
or collectionSettings.description ~= gallery.description
or collectionSettings.display_style ~= gallery.displaystyle then
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' already connected to Lightroom Published Collection Set %i. Updating.", uuid, gallery.name, lrCollectionSet.localIdentifier))
collectionSettings.description = gallery.description
collectionSettings.display_style = gallery.displaystyle
catalog:withWriteAccessDo('Resynchronize LR collection settings', function()
lrCollectionSet:setRemoteUrl(gallery.fullurl)
lrCollectionSet:setCollectionSetSettings(collectionSettings)
end)
updateCount = updateCount + 1
else
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' already connected to Lightroom Published Collection Set %i. Doing nothing.", uuid, gallery.name, lrCollectionSet.localIdentifier))
end
elseif lrCollection then
-- Already properly connected, good
local collectionSettings = lrCollection:getCollectionInfoSummary().collectionSettings or {}
if lrCollection:getRemoteUrl() ~= gallery.fullurl
or collectionSettings.description ~= gallery.description
or collectionSettings.display_style ~= gallery.displaystyle then
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' already connected to Lightroom Published Collection %i. Updating.", uuid, gallery.name, lrCollection.localIdentifier))
collectionSettings.description = gallery.description
collectionSettings.display_style = gallery.displaystyle
catalog:withWriteAccessDo('Resynchronize LR collection settings', function()
lrCollection:setRemoteUrl(gallery.fullurl)
lrCollection:setCollectionSettings(collectionSettings)
end)
updateCount = updateCount + 1
else
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' already connected to Lightroom Published Collection %i. Doing nothing.", uuid, gallery.name, lrCollection.localIdentifier))
end
else
-- Missing in Lightroom: create
local collectionName = gallery.name
-- Check for duplicate gallery names in this parent gallery: Lightroom does indeed require name uniqueness, but PhotoDeck doesn't
local copyCount = 1
for uuidN, galleryN in pairs(pdGalleries) do
if galleryN.name == gallery.name then
if uuid == uuidN then
break
else
copyCount = copyCount + 1
end
end
end
if copyCount > 1 then
collectionName = collectionName .. ' (' .. tostring(copyCount) .. ')'
end
if shouldBeACollectionSet then
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' NOT found in Lightroom. Creating Published Collection Set.", uuid, collectionName))
catalog:withWriteAccessDo('Creating Published Collection Set', function()
lrCollectionSet = publishService:createPublishedCollectionSet(collectionName, parentLRCollectionSet)
end)
if lrCollectionSet then
local collectionSettings = {}
collectionSettings.description = gallery.description
collectionSettings.display_style = gallery.displaystyle
catalog:withWriteAccessDo('Set LR collection settings', function()
lrCollectionSet:setRemoteId(uuid)
lrCollectionSet:setRemoteUrl(gallery.fullurl)
lrCollectionSet:setCollectionSetSettings(collectionSettings)
end)
createCount = createCount + 1
else
logger:trace(string.format("SYNC ERROR: PhotoDeck gallery %s '%s' NOT found in Lightroom, and failed to create Published Collection Set.", uuid, collectionName))
errorsCount = errorsCount + 1
end
elseif shouldBeACollection then
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' NOT found in Lightroom. Creating Published Collection.", uuid, collectionName))
catalog:withWriteAccessDo('Creating Published Collection', function()
lrCollection = publishService:createPublishedCollection(collectionName, parentLRCollectionSet)
end)
if lrCollection then
local collectionSettings = {}
collectionSettings.description = gallery.description
collectionSettings.display_style = gallery.displaystyle
catalog:withWriteAccessDo('Set LR collection settings', function()
lrCollection:setRemoteId(uuid)
lrCollection:setRemoteUrl(gallery.fullurl)
lrCollection:setCollectionSettings(collectionSettings)
end)
createCount = createCount + 1
else
logger:trace(string.format("SYNC ERROR: PhotoDeck gallery %s '%s' NOT found in Lightroom, and failed to create Published Collection.", uuid, collectionName))
errorsCount = errorsCount + 1
end
else
logger:trace(string.format("SYNC: PhotoDeck gallery %s '%s' NOT found in Lightroom. Creating Published Collection by default.", uuid, collectionName))
catalog:withWriteAccessDo('Creating Published Collection', function()
lrCollection = publishService:createPublishedCollection(collectionName, parentLRCollectionSet)
end)
if lrCollection then
local collectionSettings = {}
collectionSettings.description = gallery.description
collectionSettings.display_style = gallery.displaystyle
catalog:withWriteAccessDo('Set LR collection settings', function()
lrCollection:setRemoteId(uuid)
lrCollection:setRemoteUrl(gallery.fullurl)
lrCollection:setCollectionSettings(collectionSettings)
end)
createCount = createCount + 1
else
logger:trace(string.format("SYNC ERROR: PhotoDeck gallery %s '%s' NOT found in Lightroom, and failed to create Published Collection.", uuid, collectionName))
errorsCount = errorsCount + 1
end
end
end
-- Recurse in sub galleries
if photodeckGalleriesByParent[uuid] and lrCollectionSet then
synchronizeGallery(depth + 1, uuid, lrCollectionSet)
end
end
end
synchronizeGallery(1, rootGalleryId, publishService)
if progressScope:isCanceled() then
logger:trace("SYNC: Canceled")
end
logger:trace(string.format("SYNC: Done, created: %i, deleted: %i, updated: %i, errors: %i", createCount, deleteCount, updateCount, errorsCount))
PhotoDeckAPI.canSynchronize = true
return { created = createCount, deleted = deleteCount, updated = updateCount, errors = errorsCount }
end
-- getPhoto returns a photo with remote ID uuid, or nil if it does not exist
function PhotoDeckAPI.getPhoto(photoId)
logger:trace(string.format('PhotoDeckAPI.getPhoto("%s")', photoId))
local url = '/medias/' .. photoId .. '.xml'
local onerror = {}
onerror["404"] = function() return nil end
local response, error_msg = PhotoDeckAPI.request('GET', url, nil, onerror)
local result = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.media)
--logger:trace('PhotoDeckAPI.getPhoto: ' .. printTable(result))
return result, error_msg
end
function PhotoDeckAPI.photosInGallery(urlname, galleryId)
logger:trace(string.format('PhotoDeckAPI.photosInGallery("%s", "%s")', urlname, galleryId))
local url = '/websites/' .. urlname .. '/galleries/' .. galleryId .. '.xml'
local response, error_msg = PhotoDeckAPI.request('GET', url, { view = 'details_with_medias' })
local medias = PhotoDeckAPIXSLT.transform(response, PhotoDeckAPIXSLT.mediasInGallery)