tiktokautouploader 2.7__py2.py3-none-any.whl → 2.9__py2.py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- tiktokautouploader/Js_assets/login.js +7 -10
- tiktokautouploader/function.py +207 -111
- tiktokautouploader-2.9.dist-info/METADATA +138 -0
- tiktokautouploader-2.9.dist-info/RECORD +7 -0
- tiktokautouploader-2.7.dist-info/METADATA +0 -19
- tiktokautouploader-2.7.dist-info/RECORD +0 -7
- {tiktokautouploader-2.7.dist-info → tiktokautouploader-2.9.dist-info}/WHEEL +0 -0
- {tiktokautouploader-2.7.dist-info → tiktokautouploader-2.9.dist-info}/licenses/LICENSE.md +0 -0
@@ -1,18 +1,16 @@
|
|
1
1
|
const { chromium } = require('playwright-extra')
|
2
2
|
|
3
|
-
|
4
|
-
// Note: playwright-extra is compatible with most puppeteer-extra plugins
|
3
|
+
|
5
4
|
const stealth = require('puppeteer-extra-plugin-stealth')()
|
6
5
|
|
7
|
-
// Add the plugin to playwright (any number of plugins can be added)
|
8
6
|
chromium.use(stealth)
|
9
7
|
|
10
|
-
|
11
|
-
function
|
8
|
+
|
9
|
+
function sleep(time) {
|
12
10
|
return new Promise(resolve => setTimeout(resolve, time));
|
13
11
|
}
|
14
12
|
|
15
|
-
|
13
|
+
|
16
14
|
async function checkForRedirect(page) {
|
17
15
|
const currentUrl = page.url();
|
18
16
|
const pattern = /^https:\/\/www\.tiktok\.com\/foryou/;
|
@@ -25,19 +23,18 @@ async function checkForRedirect(page) {
|
|
25
23
|
const page = await browser.newPage();
|
26
24
|
await page.goto('https://www.tiktok.com/login');
|
27
25
|
|
28
|
-
// Loop until redirect is detected
|
29
26
|
while (!redirected) {
|
30
27
|
redirected = await checkForRedirect(page);
|
31
28
|
if (!redirected) {
|
32
|
-
await
|
29
|
+
await sleep(1000);
|
33
30
|
}
|
34
31
|
}
|
35
32
|
|
36
|
-
|
33
|
+
sleep(2000)
|
37
34
|
const cookies = await page.context().cookies();
|
38
35
|
const fs = require('fs');
|
39
36
|
fs.writeFileSync('TK_cookies.json', JSON.stringify(cookies, null, 2));
|
40
37
|
|
41
|
-
|
38
|
+
|
42
39
|
await browser.close();
|
43
40
|
})();
|
tiktokautouploader/function.py
CHANGED
@@ -11,9 +11,6 @@ import os
|
|
11
11
|
import warnings
|
12
12
|
warnings.filterwarnings("ignore")
|
13
13
|
|
14
|
-
def no_draft_warning():
|
15
|
-
print("SAVE AS DRAFT BUTTON NOT FOUND; CODE WILL CONTINUE RUNNING BUT VIDEO IS UNLIKELY TO UPLOAD; BEING ABLE TO SAVE AS DRAFT IS ESSENTIAL TO BEING ABLE TO EDIT VIDEOS AND POST, PLEASE BUILD UP ENOUGH ACCOUNT HISTORY TO BE ABLE TO SAVE DRAFTS ")
|
16
|
-
|
17
14
|
def login_warning():
|
18
15
|
print("NO COOKIES FILE FOUND, PLEASE LOG-IN WHEN PROMPTED")
|
19
16
|
|
@@ -21,10 +18,26 @@ def save_cookies(cookies):
|
|
21
18
|
with open('TK_cookies.json', 'w') as file:
|
22
19
|
json.dump(cookies, file, indent=4)
|
23
20
|
|
21
|
+
def check_expiry():
|
22
|
+
with open('TK_cookies.json', 'r') as file:
|
23
|
+
cookies = json.load(file)
|
24
|
+
|
25
|
+
current_time = int(time.time())
|
26
|
+
cookies_expire = []
|
27
|
+
expired = False
|
28
|
+
for cookie in cookies:
|
29
|
+
if cookie['name'] in ['sessionid', 'sid_tt', 'sessionid_ss', 'passport_auth_status']:
|
30
|
+
expiry = cookie.get('expires')
|
31
|
+
cookies_expire.append(expiry < current_time)
|
32
|
+
|
33
|
+
if all(cookies_expire):
|
34
|
+
expired = True
|
35
|
+
|
36
|
+
return expired
|
37
|
+
|
24
38
|
def run_javascript():
|
25
39
|
js_file_path = pkg_resources.resource_filename(__name__, 'Js_assets/login.js')
|
26
40
|
result = subprocess.run(['node', js_file_path], capture_output=True, text=True)
|
27
|
-
return result.stdout, result.stderr
|
28
41
|
|
29
42
|
def install_js_dependencies():
|
30
43
|
js_dir = pkg_resources.resource_filename(__name__, 'Js_assets')
|
@@ -34,7 +47,6 @@ def install_js_dependencies():
|
|
34
47
|
print("JavaScript dependencies not found. Installing...")
|
35
48
|
subprocess.run(['npm', 'install', 'playwright', 'playwright-extra', 'puppeteer-extra-plugin-stealth', '--silent'], cwd=js_dir, check=True)
|
36
49
|
else:
|
37
|
-
print("found dependencies")
|
38
50
|
time.sleep(0.1)
|
39
51
|
|
40
52
|
|
@@ -197,7 +209,6 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
197
209
|
video (str) -> path to video to upload
|
198
210
|
description (str) -> description for video
|
199
211
|
hashtags (str)(array) -> hashtags for video
|
200
|
-
cookies_path (str) -> path to tik tok cookies .json file
|
201
212
|
sound_name (str) -> name of tik tok sound to use for video
|
202
213
|
sound_aud_vol (str) -> volume of tik tok sound, 'main', 'mix' or 'background', check documentation for more info -> https://github.com/haziq-exe/TikTokAutoUploader
|
203
214
|
schedule (str) -> format HH:MM, your local time to upload video
|
@@ -209,12 +220,17 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
209
220
|
|
210
221
|
retries = 0
|
211
222
|
cookie_read = False
|
223
|
+
install_js_dependencies()
|
212
224
|
|
213
225
|
if os.path.exists('TK_cookies.json'):
|
214
226
|
cookies, cookie_read = read_cookies(cookies_path='TK_cookies.json')
|
227
|
+
expired = check_expiry()
|
228
|
+
if expired == True:
|
229
|
+
os.remove('TK_cookies.json')
|
230
|
+
print("COOKIES EXPIRED, PLEASE LOG-IN AGAIN")
|
231
|
+
cookie_read = False
|
215
232
|
|
216
233
|
if cookie_read == False:
|
217
|
-
install_js_dependencies()
|
218
234
|
login_warning()
|
219
235
|
run_javascript()
|
220
236
|
|
@@ -224,11 +240,14 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
224
240
|
|
225
241
|
|
226
242
|
with sync_playwright() as p:
|
227
|
-
|
243
|
+
|
244
|
+
browser = p.firefox.launch(headless=True)
|
245
|
+
|
228
246
|
context = browser.new_context()
|
229
247
|
context.add_cookies(cookies)
|
230
248
|
page = context.new_page()
|
231
249
|
url = 'https://www.tiktok.com/tiktokstudio/upload?from=upload&lang=en'
|
250
|
+
url2 = 'https://www.tiktok.com/tiktokstudio/content?tab=draft'
|
232
251
|
|
233
252
|
while retries < 2:
|
234
253
|
try:
|
@@ -254,7 +273,6 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
254
273
|
time.sleep(0.1)
|
255
274
|
|
256
275
|
if captcha == True:
|
257
|
-
print("CAPTCHA FOUND")
|
258
276
|
image = get_image_src(page)
|
259
277
|
if image:
|
260
278
|
if suppressprint == False:
|
@@ -291,7 +309,7 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
291
309
|
if attempts > 5:
|
292
310
|
sys.exit("FAILED TO SOLVE CAPTCHA")
|
293
311
|
try:
|
294
|
-
page.wait_for_selector('.upload-text-container', timeout=
|
312
|
+
page.wait_for_selector('.upload-text-container', timeout=5000)
|
295
313
|
os.remove('captcha_image.jpg')
|
296
314
|
if suppressprint == False:
|
297
315
|
print("Captcha Solved")
|
@@ -308,7 +326,6 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
308
326
|
while objectclick == 'N.A':
|
309
327
|
page.click('span.secsdk_captcha_refresh--text')
|
310
328
|
page.wait_for_selector('div.VerifyBar___StyledDiv-sc-12zaxoy-0.hRJhHT')
|
311
|
-
time.sleep(2)
|
312
329
|
question = page.locator('div.VerifyBar___StyledDiv-sc-12zaxoy-0.hRJhHT').text_content()
|
313
330
|
objectclick = understood_Qs(question)
|
314
331
|
image = get_image_src(page)
|
@@ -333,10 +350,10 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
333
350
|
click_on_objects(page, webpage_coords)
|
334
351
|
time.sleep(0.5)
|
335
352
|
page.click("div.verify-captcha-submit-button")
|
336
|
-
if attempts >
|
353
|
+
if attempts > 10:
|
337
354
|
sys.exit("FAILED TO SOLVE CAPTCHA")
|
338
355
|
try:
|
339
|
-
page.wait_for_selector('.upload-text-container', timeout=
|
356
|
+
page.wait_for_selector('.upload-text-container', timeout=5000)
|
340
357
|
solved = True
|
341
358
|
os.remove('captcha_image.jpg')
|
342
359
|
if suppressprint == False:
|
@@ -344,13 +361,11 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
344
361
|
except:
|
345
362
|
continue
|
346
363
|
|
347
|
-
|
348
364
|
|
349
365
|
try:
|
350
366
|
page.set_input_files('input[type="file"][accept="video/*"]', f'{video}')
|
351
367
|
except:
|
352
368
|
sys.exit("ERROR: FAILED TO INPUT FILE. Possible Issues: Wifi too slow, file directory wrong, or check documentation to see if captcha is solvable")
|
353
|
-
|
354
369
|
page.wait_for_selector('div[data-contents="true"]')
|
355
370
|
page.click('div[data-contents="true"]')
|
356
371
|
if suppressprint == False:
|
@@ -400,75 +415,10 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
400
415
|
except:
|
401
416
|
sys.exit("ERROR: TIK TOK TOOK TOO LONG TO UPLOAD YOUR FILE (>20min). Try again, if issue persists then try a lower file size or different wifi connection")
|
402
417
|
|
403
|
-
time.sleep(0.
|
418
|
+
time.sleep(0.2)
|
404
419
|
if suppressprint == False:
|
405
420
|
print("Tik tok done loading file onto servers")
|
406
|
-
|
407
|
-
page.click('button.TUXButton.TUXButton--default.TUXButton--large.TUXButton--secondary:has-text("Save draft")', timeout=10000)
|
408
|
-
page.wait_for_selector("path[d='M37.37 4.85a4.01 4.01 0 0 0-.99-.79 3 3 0 0 0-2.72 0c-.45.23-.81.6-1 .79a9 9 0 0 1-.04.05l-19.3 19.3c-1.64 1.63-2.53 2.52-3.35 3.47a36 36 0 0 0-4.32 6.16c-.6 1.1-1.14 2.24-2.11 4.33l-.3.6c-.4.75-.84 1.61-.8 2.43a2.5 2.5 0 0 0 2.37 2.36c.82.05 1.68-.4 2.44-.79l.59-.3c2.09-.97 3.23-1.5 4.33-2.11a36 36 0 0 0 6.16-4.32c.95-.82 1.84-1.71 3.47-3.34l19.3-19.3.05-.06a3 3 0 0 0 .78-3.71c-.22-.45-.6-.81-.78-1l-.02-.02-.03-.03-3.67-3.67a8.7 8.7 0 0 1-.06-.05ZM16.2 26.97 35.02 8.15l2.83 2.83L19.03 29.8c-1.7 1.7-2.5 2.5-3.33 3.21a32 32 0 0 1-7.65 4.93 32 32 0 0 1 4.93-7.65c.73-.82 1.51-1.61 3.22-3.32Z']")
|
409
|
-
page.click("path[d='M37.37 4.85a4.01 4.01 0 0 0-.99-.79 3 3 0 0 0-2.72 0c-.45.23-.81.6-1 .79a9 9 0 0 1-.04.05l-19.3 19.3c-1.64 1.63-2.53 2.52-3.35 3.47a36 36 0 0 0-4.32 6.16c-.6 1.1-1.14 2.24-2.11 4.33l-.3.6c-.4.75-.84 1.61-.8 2.43a2.5 2.5 0 0 0 2.37 2.36c.82.05 1.68-.4 2.44-.79l.59-.3c2.09-.97 3.23-1.5 4.33-2.11a36 36 0 0 0 6.16-4.32c.95-.82 1.84-1.71 3.47-3.34l19.3-19.3.05-.06a3 3 0 0 0 .78-3.71c-.22-.45-.6-.81-.78-1l-.02-.02-.03-.03-3.67-3.67a8.7 8.7 0 0 1-.06-.05ZM16.2 26.97 35.02 8.15l2.83 2.83L19.03 29.8c-1.7 1.7-2.5 2.5-3.33 3.21a32 32 0 0 1-7.65 4.93 32 32 0 0 1 4.93-7.65c.73-.82 1.51-1.61 3.22-3.32Z']")
|
410
|
-
page.wait_for_selector('div[data-contents="true"]')
|
411
|
-
page.wait_for_function("document.querySelector('.info-progress-num').textContent.trim() === '100%'", timeout=3000000)
|
412
|
-
time.sleep(0.5)
|
413
|
-
except:
|
414
|
-
no_draft_warning()
|
415
|
-
|
416
|
-
if sound_name != None:
|
417
|
-
page.click("div.TUXButton-label:has-text('Edit video')")
|
418
|
-
page.wait_for_selector("input.search-bar-input")
|
419
|
-
page.fill(f"input.search-bar-input", f"{sound_name}")
|
420
|
-
time.sleep(0.5)
|
421
|
-
page.click("div.TUXButton-label:has-text('Search')")
|
422
|
-
try:
|
423
|
-
page.wait_for_selector('div.music-card-container')
|
424
|
-
page.click("div.music-card-container")
|
425
|
-
page.wait_for_selector("div.TUXButton-label:has-text('Use')")
|
426
|
-
page.click("div.TUXButton-label:has-text('Use')")
|
427
|
-
except:
|
428
|
-
sys.exit("ERROR: SOUND NOT FOUND, VIDEO SAVED AS DRAFT")
|
429
|
-
try:
|
430
|
-
page.wait_for_selector('img[src=""]')
|
431
|
-
page.click('img[src=""]')
|
432
|
-
time.sleep(0.5)
|
433
|
-
sliders = page.locator("input.scaleInput")
|
434
|
-
|
435
|
-
if sound_aud_vol == 'background':
|
436
|
-
slider1 = sliders.nth(0)
|
437
|
-
bounding_box1 = slider1.bounding_box()
|
438
|
-
if bounding_box1:
|
439
|
-
x1 = bounding_box1["x"] + (bounding_box1["width"] * 0.95)
|
440
|
-
y1 = bounding_box1["y"] + bounding_box1["height"] / 2
|
441
|
-
page.mouse.click(x1, y1)
|
442
|
-
|
443
|
-
slider2 = sliders.nth(1)
|
444
|
-
bounding_box2 = slider2.bounding_box()
|
445
|
-
if bounding_box2:
|
446
|
-
x2 = bounding_box2["x"] + (bounding_box2["width"] * 0.097)
|
447
|
-
y2 = bounding_box2["y"] + bounding_box2["height"] / 2
|
448
|
-
page.mouse.click(x2, y2)
|
449
|
-
|
450
|
-
if sound_aud_vol == 'main':
|
451
|
-
slider1 = sliders.nth(0)
|
452
|
-
bounding_box1 = slider1.bounding_box()
|
453
|
-
if bounding_box1:
|
454
|
-
x1 = bounding_box1["x"] + (bounding_box1["width"] * 0.097)
|
455
|
-
y1 = bounding_box1["y"] + bounding_box1["height"] / 2
|
456
|
-
page.mouse.click(x1, y1)
|
457
|
-
slider2 = sliders.nth(1)
|
458
|
-
bounding_box2 = slider2.bounding_box()
|
459
|
-
if bounding_box2:
|
460
|
-
x2 = bounding_box2["x"] + (bounding_box2["width"] * 0.95)
|
461
|
-
y2 = bounding_box2["y"] + bounding_box2["height"] / 2
|
462
|
-
page.mouse.click(x2, y2)
|
463
|
-
except:
|
464
|
-
sys.exit("ERROR ADJUSTING SOUND VOLUME, VIDEO SAVED AS DRAFT")
|
465
|
-
|
466
|
-
page.wait_for_selector("div.TUXButton-label:has-text('Save edit')")
|
467
|
-
page.click("div.TUXButton-label:has-text('Save edit')")
|
468
|
-
if suppressprint == False:
|
469
|
-
print("Added sound")
|
470
|
-
|
471
|
-
page.wait_for_selector('div[data-contents="true"]')
|
421
|
+
|
472
422
|
if schedule != None:
|
473
423
|
try:
|
474
424
|
hour = schedule[0:2]
|
@@ -480,25 +430,31 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
480
430
|
sys.exit("SCHEDULE TIME ERROR: PLEASE MAKE SURE YOUR SCHEDULE TIME IS A STRING THAT FOLLOWS THE 24H FORMAT 'HH:MM', VIDEO SAVED AS DRAFT")
|
481
431
|
|
482
432
|
page.locator('div.TUXRadioStandalone.TUXRadioStandalone--medium').nth(1).click()
|
483
|
-
time.sleep(
|
433
|
+
time.sleep(1)
|
434
|
+
if page.locator('button.TUXButton.TUXButton--default.TUXButton--medium.TUXButton--primary:has-text("Allow")').nth(0).is_visible():
|
435
|
+
page.locator('button.TUXButton.TUXButton--default.TUXButton--medium.TUXButton--primary:has-text("Allow")').nth(0).click()
|
436
|
+
time.sleep(0.2)
|
437
|
+
else:
|
438
|
+
if page.locator('div.TUXTextInputCore-trailingIconWrapper').nth(1).is_visible():
|
439
|
+
time.sleep(0.2)
|
484
440
|
if day != None:
|
485
441
|
page.locator('div.TUXTextInputCore-trailingIconWrapper').nth(1).click()
|
486
|
-
time.sleep(0.
|
442
|
+
time.sleep(0.2)
|
487
443
|
try:
|
488
444
|
page.locator(f'span.day.valid:has-text("{day}")').click()
|
489
445
|
except:
|
490
446
|
sys.exit("SCHEDULE DAY ERROR: ERROR WITH SCHEDULED DAY, read documentation for more information on format of day")
|
491
447
|
try:
|
448
|
+
time.sleep(1)
|
492
449
|
page.locator('div.TUXTextInputCore-trailingIconWrapper').nth(0).click()
|
493
|
-
time.sleep(0.
|
494
|
-
page.locator(f'.tiktok-timepicker-option-text:has-text("{hour}")').nth(0).scroll_into_view_if_needed()
|
495
|
-
page.locator(f'.tiktok-timepicker-option-text:has-text("{hour}")').nth(0).click()
|
496
|
-
time.sleep(0.5)
|
497
|
-
page.locator('div.TUXTextInputCore-trailingIconWrapper').nth(0).click()
|
498
|
-
time.sleep(0.5)
|
450
|
+
time.sleep(0.2)
|
499
451
|
page.locator(f'.tiktok-timepicker-option-text.tiktok-timepicker-right:has-text("{minute}")').nth(0).scroll_into_view_if_needed()
|
500
|
-
time.sleep(0.
|
452
|
+
time.sleep(0.2)
|
501
453
|
page.locator(f'.tiktok-timepicker-option-text.tiktok-timepicker-right:has-text("{minute}")').nth(0).click()
|
454
|
+
time.sleep(0.2)
|
455
|
+
page.locator(f'.tiktok-timepicker-option-text:has-text("{hour}")').nth(0).scroll_into_view_if_needed()
|
456
|
+
page.locator(f'.tiktok-timepicker-option-text:has-text("{hour}")').nth(0).click()
|
457
|
+
time.sleep(1)
|
502
458
|
|
503
459
|
if suppressprint == False:
|
504
460
|
print("Done scheduling video")
|
@@ -506,28 +462,29 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
506
462
|
except:
|
507
463
|
sys.exit("SCHEDULING ERROR: VIDEO SAVED AS DRAFT")
|
508
464
|
|
465
|
+
if (schedule == None) and (day != None):
|
466
|
+
sys.exit("ERROR: CANT SCHEDULE FOR ANOTHER DAY USING 'day' WITHOUT ALSO INCLUDING TIME OF UPLOAD WITH 'schedule'; PLEASE ALSO INCLUDE TIME WITH 'schedule' PARAMETER")
|
509
467
|
|
510
|
-
if copyrightcheck == True:
|
511
|
-
page.locator(".TUXSwitch-input").nth(0).click()
|
512
|
-
while copyrightcheck == True:
|
513
|
-
time.sleep(0.5)
|
514
|
-
if page.locator("span", has_text="No issues detected.").is_visible():
|
515
|
-
if suppressprint == False:
|
516
|
-
print("Copyright check complete")
|
517
|
-
break
|
518
|
-
if page.locator("span", has_text="Copyright issues detected.").is_visible():
|
519
|
-
sys.exit("COPYRIGHT CHECK FAILED: VIDEO SAVED AS DRAFT, COPYRIGHT AUDIO DETECTED FROM TIKTOK")
|
520
|
-
|
521
468
|
|
522
|
-
|
469
|
+
if(sound_name == None):
|
470
|
+
if copyrightcheck == True:
|
471
|
+
page.locator(".TUXSwitch-input").nth(0).click()
|
472
|
+
while copyrightcheck == True:
|
473
|
+
time.sleep(0.2)
|
474
|
+
if page.locator("span", has_text="No issues detected.").is_visible():
|
475
|
+
if suppressprint == False:
|
476
|
+
print("Copyright check complete")
|
477
|
+
break
|
478
|
+
if page.locator("span", has_text="Copyright issues detected.").is_visible():
|
479
|
+
sys.exit("COPYRIGHT CHECK FAILED: COPYRIGHT AUDIO DETECTED FROM TIKTOK")
|
480
|
+
|
523
481
|
if schedule == None:
|
524
482
|
page.click('button.TUXButton.TUXButton--default.TUXButton--large.TUXButton--primary:has-text("Post")', timeout=10000)
|
525
483
|
uploaded = False
|
526
484
|
checks = 0
|
527
485
|
while uploaded == False:
|
528
486
|
if page.locator(':has-text("Leaving the page does not interrupt")').nth(0).is_visible():
|
529
|
-
|
530
|
-
time.sleep(1)
|
487
|
+
time.sleep(0.2)
|
531
488
|
break
|
532
489
|
time.sleep(0.2)
|
533
490
|
checks += 1
|
@@ -541,7 +498,7 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
541
498
|
checks = 0
|
542
499
|
while uploaded == False:
|
543
500
|
if page.locator(':has-text("Leaving the page does not interrupt")').nth(0).is_visible():
|
544
|
-
time.sleep(
|
501
|
+
time.sleep(0.2)
|
545
502
|
break
|
546
503
|
time.sleep(0.2)
|
547
504
|
checks += 1
|
@@ -551,9 +508,148 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
551
508
|
break
|
552
509
|
if suppressprint == False:
|
553
510
|
print("Done uploading video, NOTE: it may take a minute or two to show on TikTok")
|
554
|
-
|
555
|
-
|
556
|
-
|
557
|
-
|
511
|
+
|
512
|
+
page.close()
|
513
|
+
|
514
|
+
else:
|
515
|
+
try:
|
516
|
+
page.click('button.TUXButton.TUXButton--default.TUXButton--large.TUXButton--secondary:has-text("Save draft")', timeout=10000)
|
517
|
+
except:
|
518
|
+
sys.exit("SAVE AS DRAFT BUTTON NOT FOUND; CANNOT ADD SOUND WITHOUT ABILITY TO SAVE DRAFTS")
|
519
|
+
|
520
|
+
time.sleep(0.5)
|
521
|
+
page.close()
|
522
|
+
|
523
|
+
browser = p.chromium.launch(headless=True)
|
524
|
+
|
525
|
+
context = browser.new_context()
|
526
|
+
context.add_cookies(cookies)
|
527
|
+
page = context.new_page()
|
528
|
+
url2 = 'https://www.tiktok.com/tiktokstudio/content?tab=draft'
|
529
|
+
|
530
|
+
while retries < 2:
|
531
|
+
try:
|
532
|
+
page.goto(url2, timeout=30000)
|
533
|
+
except:
|
534
|
+
retries +=1
|
535
|
+
time.sleep(5)
|
536
|
+
if retries == 2:
|
537
|
+
sys.exit("ERROR: TIK TOK PAGE FAILED TO LOAD, try again.")
|
538
|
+
else:
|
539
|
+
break
|
540
|
+
|
541
|
+
try:
|
542
|
+
page.wait_for_selector("path[d='M37.37 4.85a4.01 4.01 0 0 0-.99-.79 3 3 0 0 0-2.72 0c-.45.23-.81.6-1 .79a9 9 0 0 1-.04.05l-19.3 19.3c-1.64 1.63-2.53 2.52-3.35 3.47a36 36 0 0 0-4.32 6.16c-.6 1.1-1.14 2.24-2.11 4.33l-.3.6c-.4.75-.84 1.61-.8 2.43a2.5 2.5 0 0 0 2.37 2.36c.82.05 1.68-.4 2.44-.79l.59-.3c2.09-.97 3.23-1.5 4.33-2.11a36 36 0 0 0 6.16-4.32c.95-.82 1.84-1.71 3.47-3.34l19.3-19.3.05-.06a3 3 0 0 0 .78-3.71c-.22-.45-.6-.81-.78-1l-.02-.02-.03-.03-3.67-3.67a8.7 8.7 0 0 1-.06-.05ZM16.2 26.97 35.02 8.15l2.83 2.83L19.03 29.8c-1.7 1.7-2.5 2.5-3.33 3.21a32 32 0 0 1-7.65 4.93 32 32 0 0 1 4.93-7.65c.73-.82 1.51-1.61 3.22-3.32Z']")
|
543
|
+
page.click("path[d='M37.37 4.85a4.01 4.01 0 0 0-.99-.79 3 3 0 0 0-2.72 0c-.45.23-.81.6-1 .79a9 9 0 0 1-.04.05l-19.3 19.3c-1.64 1.63-2.53 2.52-3.35 3.47a36 36 0 0 0-4.32 6.16c-.6 1.1-1.14 2.24-2.11 4.33l-.3.6c-.4.75-.84 1.61-.8 2.43a2.5 2.5 0 0 0 2.37 2.36c.82.05 1.68-.4 2.44-.79l.59-.3c2.09-.97 3.23-1.5 4.33-2.11a36 36 0 0 0 6.16-4.32c.95-.82 1.84-1.71 3.47-3.34l19.3-19.3.05-.06a3 3 0 0 0 .78-3.71c-.22-.45-.6-.81-.78-1l-.02-.02-.03-.03-3.67-3.67a8.7 8.7 0 0 1-.06-.05ZM16.2 26.97 35.02 8.15l2.83 2.83L19.03 29.8c-1.7 1.7-2.5 2.5-3.33 3.21a32 32 0 0 1-7.65 4.93 32 32 0 0 1 4.93-7.65c.73-.82 1.51-1.61 3.22-3.32Z']")
|
544
|
+
page.wait_for_selector('div[data-contents="true"]')
|
545
|
+
page.wait_for_function("document.querySelector('.info-progress-num').textContent.trim() === '100%'", timeout=3000000)
|
546
|
+
time.sleep(0.2)
|
547
|
+
except:
|
548
|
+
sys.exit("ERROR ADDING SOUND: Video saved as draft")
|
549
|
+
|
550
|
+
if sound_name != None:
|
551
|
+
page.click("div.TUXButton-label:has-text('Edit video')")
|
552
|
+
page.wait_for_selector("input.search-bar-input")
|
553
|
+
page.fill(f"input.search-bar-input", f"{sound_name}")
|
554
|
+
time.sleep(0.2)
|
555
|
+
page.click("div.TUXButton-label:has-text('Search')")
|
556
|
+
try:
|
557
|
+
page.wait_for_selector('div.music-card-container')
|
558
|
+
page.click("div.music-card-container")
|
559
|
+
page.wait_for_selector("div.TUXButton-label:has-text('Use')")
|
560
|
+
page.click("div.TUXButton-label:has-text('Use')")
|
561
|
+
except:
|
562
|
+
sys.exit(f"ERROR: SOUND '{sound_name}' NOT FOUND")
|
563
|
+
try:
|
564
|
+
page.wait_for_selector('img[src=""]')
|
565
|
+
page.click('img[src=""]')
|
566
|
+
time.sleep(0.5)
|
567
|
+
sliders = page.locator("input.scaleInput")
|
568
|
+
|
569
|
+
if sound_aud_vol == 'background':
|
570
|
+
slider1 = sliders.nth(0)
|
571
|
+
bounding_box1 = slider1.bounding_box()
|
572
|
+
if bounding_box1:
|
573
|
+
x1 = bounding_box1["x"] + (bounding_box1["width"] * 0.92)
|
574
|
+
y1 = bounding_box1["y"] + bounding_box1["height"] / 2
|
575
|
+
page.mouse.click(x1, y1)
|
576
|
+
|
577
|
+
slider2 = sliders.nth(1)
|
578
|
+
bounding_box2 = slider2.bounding_box()
|
579
|
+
if bounding_box2:
|
580
|
+
x2 = bounding_box2["x"] + (bounding_box2["width"] * 0.097)
|
581
|
+
y2 = bounding_box2["y"] + bounding_box2["height"] / 2
|
582
|
+
page.mouse.click(x2, y2)
|
583
|
+
|
584
|
+
if sound_aud_vol == 'main':
|
585
|
+
slider1 = sliders.nth(0)
|
586
|
+
bounding_box1 = slider1.bounding_box()
|
587
|
+
if bounding_box1:
|
588
|
+
x1 = bounding_box1["x"] + (bounding_box1["width"] * 0.092)
|
589
|
+
y1 = bounding_box1["y"] + bounding_box1["height"] / 2
|
590
|
+
page.mouse.click(x1, y1)
|
591
|
+
slider2 = sliders.nth(1)
|
592
|
+
bounding_box2 = slider2.bounding_box()
|
593
|
+
if bounding_box2:
|
594
|
+
x2 = bounding_box2["x"] + (bounding_box2["width"] * 0.92)
|
595
|
+
y2 = bounding_box2["y"] + bounding_box2["height"] / 2
|
596
|
+
page.mouse.click(x2, y2)
|
597
|
+
except:
|
598
|
+
sys.exit("ERROR ADJUSTING SOUND VOLUME: please try again.")
|
599
|
+
|
600
|
+
page.wait_for_selector("div.TUXButton-label:has-text('Save edit')")
|
601
|
+
page.click("div.TUXButton-label:has-text('Save edit')")
|
602
|
+
if suppressprint == False:
|
603
|
+
print("Added sound")
|
604
|
+
|
605
|
+
page.wait_for_selector('div[data-contents="true"]')
|
606
|
+
|
607
|
+
if copyrightcheck == True:
|
608
|
+
page.locator(".TUXSwitch-input").nth(0).click()
|
609
|
+
while copyrightcheck == True:
|
610
|
+
time.sleep(0.2)
|
611
|
+
if page.locator("span", has_text="No issues detected.").is_visible():
|
612
|
+
if suppressprint == False:
|
613
|
+
print("Copyright check complete")
|
614
|
+
break
|
615
|
+
if page.locator("span", has_text="Copyright issues detected.").is_visible():
|
616
|
+
sys.exit("COPYRIGHT CHECK FAILED: VIDEO SAVED AS DRAFT, COPYRIGHT AUDIO DETECTED FROM TIKTOK")
|
617
|
+
|
618
|
+
|
619
|
+
try:
|
620
|
+
if schedule == None:
|
621
|
+
page.click('button.TUXButton.TUXButton--default.TUXButton--large.TUXButton--primary:has-text("Post")', timeout=10000)
|
622
|
+
uploaded = False
|
623
|
+
checks = 0
|
624
|
+
while uploaded == False:
|
625
|
+
if page.locator(':has-text("Leaving the page does not interrupt")').nth(0).is_visible():
|
626
|
+
time.sleep(0.2)
|
627
|
+
break
|
628
|
+
time.sleep(0.2)
|
629
|
+
checks += 1
|
630
|
+
if checks > 100:
|
631
|
+
time.sleep(10)
|
632
|
+
if checks == 150:
|
633
|
+
break
|
634
|
+
else:
|
635
|
+
page.click('button.TUXButton.TUXButton--default.TUXButton--large.TUXButton--primary:has-text("Schedule")', timeout=10000)
|
636
|
+
uploaded = False
|
637
|
+
checks = 0
|
638
|
+
while uploaded == False:
|
639
|
+
if page.locator(':has-text("Leaving the page does not interrupt")').nth(0).is_visible():
|
640
|
+
time.sleep(0.2)
|
641
|
+
break
|
642
|
+
time.sleep(0.2)
|
643
|
+
checks += 1
|
644
|
+
if checks > 100:
|
645
|
+
time.sleep(10)
|
646
|
+
if checks == 150:
|
647
|
+
break
|
648
|
+
if suppressprint == False:
|
649
|
+
print("Done uploading video, NOTE: it may take a minute or two to show on TikTok")
|
650
|
+
except:
|
651
|
+
time.sleep(5)
|
652
|
+
sys.exit("ERROR UPLOADING: VIDEO HAS SAVED AS DRAFT BUT CANT UPLOAD")
|
653
|
+
time.sleep(1)
|
558
654
|
|
559
|
-
|
655
|
+
page.close()
|
@@ -0,0 +1,138 @@
|
|
1
|
+
Metadata-Version: 2.3
|
2
|
+
Name: tiktokautouploader
|
3
|
+
Version: 2.9
|
4
|
+
Summary: Upload or schedule videos to TikTok with viral TikTok sounds and hashtags.
|
5
|
+
Project-URL: Homepage, https://github.com/haziq-exe/TikTokAutoUploader
|
6
|
+
Author-email: HAZIQ <haziqmk123@gmail.com>
|
7
|
+
License-File: LICENSE.md
|
8
|
+
Keywords: autoupload,tiktok,tiktokautoupload
|
9
|
+
Classifier: License :: OSI Approved :: MIT License
|
10
|
+
Classifier: Operating System :: OS Independent
|
11
|
+
Classifier: Programming Language :: Python :: 3
|
12
|
+
Classifier: Programming Language :: Python :: 3.8
|
13
|
+
Classifier: Programming Language :: Python :: 3.9
|
14
|
+
Classifier: Programming Language :: Python :: 3.10
|
15
|
+
Classifier: Programming Language :: Python :: 3.11
|
16
|
+
Requires-Dist: pillow>=8.0.0
|
17
|
+
Requires-Dist: playwright>=1.0.0
|
18
|
+
Requires-Dist: requests>=2.0.0
|
19
|
+
Requires-Dist: scikit-learn>=0.24.0
|
20
|
+
Description-Content-Type: text/markdown
|
21
|
+
|
22
|
+
<div align="center">
|
23
|
+
<h1>tiktokautouploader</h1>
|
24
|
+
</div>
|
25
|
+
|
26
|
+
|
27
|
+
### AUTOMATE TIKTOK UPLOADS 🤖. USE TRENDING SOUNDS 🔊, ADD WORKING HASHTAGS 💯, SCHEDULE UPLOADS 🗓️, AUTOSOLVES CAPTCHAS 🧠, AND MORE 🎁
|
28
|
+
|
29
|
+
[](https://pypi.org/project/tiktokautouploader/) [](https://opensource.org/licenses/MIT)
|
30
|
+
|
31
|
+
|
32
|
+
<p align="center">
|
33
|
+
<img src="READMEimage/Image.png" alt="" width="400"/>
|
34
|
+
</p>
|
35
|
+
|
36
|
+
## 🚀 Features
|
37
|
+
|
38
|
+
- **🔐 Bypass/Auto Solve Captchas:** No more manual captcha solving; fully automated process!
|
39
|
+
- **🎵 Use TikTok Sounds:** Seamlessly add popular TikTok sounds to your videos.
|
40
|
+
- **🗓 Schedule Uploads:** Upload videos at specific times or upto 10 days in advance with our scheduling feature.
|
41
|
+
- **🔍 Copyright Check:** Ensure your video is safe from copyright claims before uploading.
|
42
|
+
- **🏷 Add Working Hashtags:** Increase your reach by adding effective hashtags that actually work.
|
43
|
+
|
44
|
+
|
45
|
+
## 📦 Installation
|
46
|
+
|
47
|
+
1. **Python Installation:** Install the package using `pip`:
|
48
|
+
|
49
|
+
```bash
|
50
|
+
pip install tiktokautouploader
|
51
|
+
```
|
52
|
+
|
53
|
+
---
|
54
|
+
|
55
|
+
## ⚙️ Pre-requisites
|
56
|
+
|
57
|
+
1. **Node.js:** You must have Node.js installed on your system, as some parts of this package rely on JavaScript code. If you don't have Node.js installed, you can download it from [nodejs.org](https://nodejs.org/).
|
58
|
+
|
59
|
+
- **Note:** The necessary JavaScript dependencies (`playwright`,`playwright-extra`, `puppeteer-extra-plugin-stealth`) will be AUTOMATICALLY installed the first time you run the function, so you don't need to install them manually. Make sure that `npm` (Node.js package manager) is available in your system's PATH.
|
60
|
+
|
61
|
+
|
62
|
+
2. **Browser Binaries:** If you don't have them already, you'll need to install the chromium browser binary for `playwright`.
|
63
|
+
|
64
|
+
To do so, just run the following command AFTER installing the package:
|
65
|
+
|
66
|
+
```bash
|
67
|
+
python -m playwright install chromium
|
68
|
+
```
|
69
|
+
|
70
|
+
|
71
|
+
## 📝 Quick-Start
|
72
|
+
|
73
|
+
Here's how to upload a video to TikTok with hashtags using `tiktokautouploader`:
|
74
|
+
|
75
|
+
NOTE: It is highly recommended you read DOCUMENTATION.md before using the library.
|
76
|
+
|
77
|
+
The first time you run the code, you will be prompted to log-in, this will only occur the first time the function is used. Check documentation for more info.
|
78
|
+
|
79
|
+
```python
|
80
|
+
from tiktokautouploader import upload_tiktok
|
81
|
+
|
82
|
+
video_path = 'path/to/your/video.mp4'
|
83
|
+
description = 'Check out my latest TikTok video!'
|
84
|
+
hashtags = ['#fun', '#viral']
|
85
|
+
|
86
|
+
upload_tiktok(video=video_path, description=description, hashtags=hashtags)
|
87
|
+
|
88
|
+
```
|
89
|
+
|
90
|
+
### Upload with TikTok Sound
|
91
|
+
|
92
|
+
```python
|
93
|
+
upload_tiktok(video=video_path, description=description, sound_name='trending_sound', sound_aud_vol='main')
|
94
|
+
```
|
95
|
+
|
96
|
+
PLEASE READ DOCUMENTATION FOR MORE INFO.
|
97
|
+
|
98
|
+
### Schedule an Upload
|
99
|
+
|
100
|
+
```python
|
101
|
+
upload_tiktok(video=video_path, description=description, schedule='03:10', day=11)
|
102
|
+
```
|
103
|
+
|
104
|
+
PLEASE READ DOCUMENTATION FOR MORE INFO
|
105
|
+
|
106
|
+
### Perform Copyright Check Before Uploading
|
107
|
+
|
108
|
+
```python
|
109
|
+
upload_tiktok(video=video_path, description=description, hashtags=hashtags, copyrightcheck=True)
|
110
|
+
```
|
111
|
+
|
112
|
+
## 🎯 Why Choose `autotiktokuploader`?
|
113
|
+
|
114
|
+
- **No more captchas:** Fully automated uploads without interruptions, If captchas do show up, no worries, they will be solved. (read documentation for more info)
|
115
|
+
- **Maximize your reach:** Add popular sounds and effective hashtags that work to boost visibility.
|
116
|
+
- **Stay compliant:** Built-in copyright checks to avoid unforeseen takedowns.
|
117
|
+
- **Convenient scheduling:** Post at the right time, even when you're away.
|
118
|
+
|
119
|
+
## 🛠 Dependencies
|
120
|
+
|
121
|
+
This library requires the following dependencies:
|
122
|
+
|
123
|
+
- `playwright`
|
124
|
+
- `requests`
|
125
|
+
- `Pillow`
|
126
|
+
- `scikit-learn`
|
127
|
+
- `inference`
|
128
|
+
|
129
|
+
These will be automatically installed when you install the package.
|
130
|
+
|
131
|
+
## 👤 Author
|
132
|
+
|
133
|
+
Created by **Haziq Khalid**. Feel free to reach out at [haziqmk123@gmail.com](mailto:haziqmk123@gmail.com) or my LinkedIn.
|
134
|
+
|
135
|
+
## 📄 License
|
136
|
+
|
137
|
+
This project is licensed under the MIT License. See the [LICENSE](LICENSE.md) file for details.
|
138
|
+
```
|
@@ -0,0 +1,7 @@
|
|
1
|
+
tiktokautouploader/__init__.py,sha256=u7OWCK_u68ST8dfrkSF4Yw44CJOzV9NXI6ASRuoDfmE,64
|
2
|
+
tiktokautouploader/function.py,sha256=xH9h_yIxwuUo-CfvR1NpHDyGBhrS-6n-enNMx1l2Dqw,32635
|
3
|
+
tiktokautouploader/Js_assets/login.js,sha256=SLhtPYo8ZfTRUnbR7Xqp084lSuAOqIWUxi75FlFH3vs,966
|
4
|
+
tiktokautouploader-2.9.dist-info/METADATA,sha256=nIX_7UaWoh3EGi-RqElLzdZpT026RxvO0pABzqm7LZo,4878
|
5
|
+
tiktokautouploader-2.9.dist-info/WHEEL,sha256=fl6v0VwpzfGBVsGtkAkhILUlJxROXbA3HvRL6Fe3140,105
|
6
|
+
tiktokautouploader-2.9.dist-info/licenses/LICENSE.md,sha256=hYds_VJIpnS5gC73WhuWk2IY_e9BWjuEJthQCb9ThyU,1073
|
7
|
+
tiktokautouploader-2.9.dist-info/RECORD,,
|
@@ -1,19 +0,0 @@
|
|
1
|
-
Metadata-Version: 2.3
|
2
|
-
Name: tiktokautouploader
|
3
|
-
Version: 2.7
|
4
|
-
Summary: Upload or schedule videos to TikTok with TikTok sounds and hashtags that work.
|
5
|
-
Project-URL: Homepage, https://github.com/haziq-exe/TikTokAutoUploader
|
6
|
-
Author-email: HAZIQ KHALID <haziqmk123@gmail.com>
|
7
|
-
License-File: LICENSE.md
|
8
|
-
Keywords: autoupload,tiktok,tiktokautoupload
|
9
|
-
Classifier: License :: OSI Approved :: MIT License
|
10
|
-
Classifier: Operating System :: OS Independent
|
11
|
-
Classifier: Programming Language :: Python :: 3
|
12
|
-
Classifier: Programming Language :: Python :: 3.8
|
13
|
-
Classifier: Programming Language :: Python :: 3.9
|
14
|
-
Classifier: Programming Language :: Python :: 3.10
|
15
|
-
Classifier: Programming Language :: Python :: 3.11
|
16
|
-
Requires-Dist: pillow>=8.0.0
|
17
|
-
Requires-Dist: playwright>=1.0.0
|
18
|
-
Requires-Dist: requests>=2.0.0
|
19
|
-
Requires-Dist: scikit-learn>=0.24.0
|
@@ -1,7 +0,0 @@
|
|
1
|
-
tiktokautouploader/__init__.py,sha256=u7OWCK_u68ST8dfrkSF4Yw44CJOzV9NXI6ASRuoDfmE,64
|
2
|
-
tiktokautouploader/function.py,sha256=3S18jcePZCEDSQ2RQNs27Z3Xbp3QW9vyu1aNtbSAHsE,28340
|
3
|
-
tiktokautouploader/Js_assets/login.js,sha256=Qk_zUzRPeIN0ZnQVDtzahDNqIUJbIBbKoKw8GWW6et4,1318
|
4
|
-
tiktokautouploader-2.7.dist-info/METADATA,sha256=dQtuZ0yRpDNo1X_X6tjm2iNyCPmF7pRNQXZH1YgS1gs,816
|
5
|
-
tiktokautouploader-2.7.dist-info/WHEEL,sha256=fl6v0VwpzfGBVsGtkAkhILUlJxROXbA3HvRL6Fe3140,105
|
6
|
-
tiktokautouploader-2.7.dist-info/licenses/LICENSE.md,sha256=hYds_VJIpnS5gC73WhuWk2IY_e9BWjuEJthQCb9ThyU,1073
|
7
|
-
tiktokautouploader-2.7.dist-info/RECORD,,
|
File without changes
|
File without changes
|