tiktokautouploader 2.7__tar.gz → 2.8__tar.gz
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-2.7/Documentation.md → tiktokautouploader-2.8/DOCUMENTATION.md +12 -23
- {tiktokautouploader-2.7 → tiktokautouploader-2.8}/PKG-INFO +1 -1
- {tiktokautouploader-2.7 → tiktokautouploader-2.8}/README.md +7 -8
- tiktokautouploader-2.8/READMEimage/Captcha1.gif +0 -0
- tiktokautouploader-2.8/READMEimage/Captcha2.gif +0 -0
- {tiktokautouploader-2.7 → tiktokautouploader-2.8}/pyproject.toml +2 -2
- {tiktokautouploader-2.7 → tiktokautouploader-2.8}/tiktokautouploader/Js_assets/login.js +7 -10
- {tiktokautouploader-2.7 → tiktokautouploader-2.8}/tiktokautouploader/function.py +178 -108
- tiktokautouploader-2.7/READMEimage/CaptchaImage1.jpg +0 -0
- tiktokautouploader-2.7/READMEimage/CaptchaImage2.jpg +0 -0
- {tiktokautouploader-2.7 → tiktokautouploader-2.8}/.gitignore +0 -0
- {tiktokautouploader-2.7 → tiktokautouploader-2.8}/LICENSE.md +0 -0
- {tiktokautouploader-2.7 → tiktokautouploader-2.8}/READMEimage/Image.png +0 -0
- {tiktokautouploader-2.7 → tiktokautouploader-2.8}/tiktokautouploader/__init__.py +0 -0
@@ -25,13 +25,15 @@ This document provides detailed information about the parameters and usage of th
|
|
25
25
|
- An array of hashtag strings (e.g., `['#example', '#fun']`) to be added to the video description.
|
26
26
|
|
27
27
|
- **`sound_name`** (str, optional, default: None)
|
28
|
-
- The name of the TikTok sound that you want to use for the video. This sound will be applied during the upload.
|
28
|
+
- The name of the TikTok sound that you want to use for the video. This sound will be applied during the upload.
|
29
|
+
- NOTE: please be specific with sound name (include sound creator name also if possible)
|
29
30
|
|
30
31
|
- **`sound_aud_vol`** (str, optional, default: `'mix'`)
|
31
32
|
- Determines the volume mix between the TikTok sound and the original video audio. Accepts one of the following options:
|
32
33
|
- `'mix'`: The TikTok sound and original audio will have a 50/50 split.
|
33
34
|
- `'background'`: The original audio will be louder, and the TikTok sound will be faintly heard in the background.
|
34
35
|
- `'main'`: The TikTok sound will be louder, and the original audio will be faintly heard in the background.
|
36
|
+
- Defaults to `'mix'` if invalid option chosen
|
35
37
|
|
36
38
|
- **`schedule`** (str, optional, default: None)
|
37
39
|
- The time you want the video to be uploaded. The format should be `HH:MM`, and the minute (`MM`) must be a multiple of 5. The scheduled time must be at least 15 minutes later than the current local time (unless scheduling for a different day). The time should be in your local time zone.
|
@@ -50,10 +52,6 @@ This document provides detailed information about the parameters and usage of th
|
|
50
52
|
- **`suppressprint`** (bool, optional, default: `True`)
|
51
53
|
- Suppresses print messages that indicate the progress of the video upload. It is recommended to set this to `False` when first running the code to see progress and ensure everything works correctly.
|
52
54
|
|
53
|
-
### 📝 IMPORTANT Notes
|
54
|
-
- **`suppressprint`** (bool, optional, default: `False`)
|
55
|
-
- When `True`, uppresses print messages that indicate the progress of the video upload. It is recommended to set this to `False` when first running the code to see progress and ensure everything works correctly.
|
56
|
-
|
57
55
|
|
58
56
|
### 🛠️ Initialization Info
|
59
57
|
|
@@ -63,17 +61,14 @@ This document provides detailed information about the parameters and usage of th
|
|
63
61
|
|
64
62
|
- Javascript dependencies will be automatically downloaded, once downloaded it will not attempt to download it again unless the files get deleted.
|
65
63
|
|
66
|
-
- Runtime might be a 20-30 seconds longer than usual, this is due to libraries being built
|
64
|
+
- Runtime might be a 20-30 seconds longer than usual, this is due to libraries being built. Runtime should return to normal after first run
|
67
65
|
|
68
66
|
|
69
67
|
### 📝 Important Notes
|
70
68
|
|
71
|
-
- **TikTok Account Recommendations**:
|
72
|
-
- It is recommended to have a TikTok account with at least a few weeks of cookies built up for the best results.
|
73
|
-
- Your TikTok account MUST have the ability to save drafts; otherwise, the code will not work correctly. If your account does not have the ability to save drafts, you will be prompted with a warning message during execution
|
74
69
|
- **VERY IMPORTANT: TikTok Account Recommendations**:
|
75
70
|
- It is recommended to have a TikTok account with at least a few weeks of history built up for the best results.
|
76
|
-
-
|
71
|
+
- If you want to upload your video with TikTok sounds, your TikTok account MUST have the ability to save drafts; otherwise, you can just upload/schedule the video with copyright checks and trending hashtags
|
77
72
|
|
78
73
|
- **Scheduling Limitations**:
|
79
74
|
- The function allows scheduling up to 10 days in advance.
|
@@ -83,20 +78,20 @@ This document provides detailed information about the parameters and usage of th
|
|
83
78
|
|
84
79
|
- **Captcha solver currently works perfectly for Captcha's of type:**
|
85
80
|
<p align="center">
|
86
|
-
<img src="READMEimage/
|
81
|
+
<img src="READMEimage/Captcha1.gif" alt="" width="200"/>
|
87
82
|
</p>
|
88
83
|
|
89
84
|
<p align="center">
|
90
|
-
<img src="READMEimage/
|
85
|
+
<img src="READMEimage/Captcha2.gif" alt="" width="200"/>
|
91
86
|
</p>
|
92
87
|
|
93
88
|
## 🕰️ Runtime:
|
94
|
-
**Total runtime depends on how long TikTok takes to upload your video to their servers,
|
89
|
+
**Total runtime depends on how long TikTok takes to upload your video to their servers, however, here are approximations on how much runtime is added by each parameter**
|
95
90
|
|
96
|
-
- **Captcha's:** 3 - 5 secs
|
97
|
-
- **Adding Sound:**
|
91
|
+
- **Captcha's:** 3 - 5 secs (in RARE cases, it can take 10-15 seconds longer)
|
92
|
+
- **Adding Sound:** 5 - 10 secs
|
98
93
|
- **Scheduling:** 1 - 3 secs
|
99
|
-
- **Copyright Check:**
|
94
|
+
- **Copyright Check:** 2 - 7 secs
|
100
95
|
|
101
96
|
- **NOTE:** When running for the FIRST TIME ONLY, it may take an extra 20 - 30 seconds at the beginning for the code to start running as libraries are being built
|
102
97
|
|
@@ -121,10 +116,4 @@ upload_tiktok(
|
|
121
116
|
)
|
122
117
|
```
|
123
118
|
|
124
|
-
For more details, please feel free to contact me at haziqmk123@gmail.com or on LinkedIn (on my github profile)
|
125
|
-
|
126
|
-
### Key Sections:
|
127
|
-
|
128
|
-
- **Parameter Explanations**: Provides detailed descriptions of each parameter, including the valid options and their effects.
|
129
|
-
- **Other Notes**: Highlights recommendations and limitations related to TikTok accounts and scheduling.
|
130
|
-
- **Example Usage**: Demonstrates a practical example of how to use the function.
|
119
|
+
For more details or if errors persist, please feel free to contact me at haziqmk123@gmail.com or on LinkedIn (on my github profile)
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.3
|
2
2
|
Name: tiktokautouploader
|
3
|
-
Version: 2.
|
3
|
+
Version: 2.8
|
4
4
|
Summary: Upload or schedule videos to TikTok with TikTok sounds and hashtags that work.
|
5
5
|
Project-URL: Homepage, https://github.com/haziq-exe/TikTokAutoUploader
|
6
6
|
Author-email: HAZIQ KHALID <haziqmk123@gmail.com>
|
@@ -3,7 +3,7 @@
|
|
3
3
|
</div>
|
4
4
|
|
5
5
|
|
6
|
-
### AUTOMATE TIKTOK UPLOADS
|
6
|
+
### AUTOMATE TIKTOK UPLOADS 🤖. USE TRENDING SOUNDS 🔊, ADD WORKING HASHTAGS 💯, SCHEDULE UPLOADS 🗓️, AUTOSOLVES CAPTCHAS 🧠, AND MORE 🎁
|
7
7
|
|
8
8
|
[](https://pypi.org/project/tiktokautouploader/) [](https://opensource.org/licenses/MIT)
|
9
9
|
|
@@ -31,21 +31,22 @@ pip install tiktokautouploader
|
|
31
31
|
|
32
32
|
---
|
33
33
|
|
34
|
-
|
35
34
|
## ⚙️ Pre-requisites
|
36
35
|
|
37
36
|
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/).
|
38
37
|
|
39
|
-
- **Note:** The necessary JavaScript dependencies (`playwright`,`playwright-extra`, `puppeteer-extra-plugin-stealth`) will be
|
38
|
+
- **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.
|
39
|
+
|
40
40
|
|
41
|
-
2. **Browser Binaries:**
|
41
|
+
2. **Browser Binaries:** If you don't have them already, you'll need to install the chromium browser binary for `playwright`.
|
42
42
|
|
43
43
|
To do so, just run the following command AFTER installing the package:
|
44
44
|
|
45
45
|
```bash
|
46
|
-
playwright install
|
46
|
+
python -m playwright install chromium
|
47
47
|
```
|
48
48
|
|
49
|
+
|
49
50
|
## 📝 Quick-Start
|
50
51
|
|
51
52
|
Here's how to upload a video to TikTok with hashtags using `tiktokautouploader`:
|
@@ -68,7 +69,7 @@ upload_tiktok(video=video_path, description=description, hashtags=hashtags)
|
|
68
69
|
### Upload with TikTok Sound
|
69
70
|
|
70
71
|
```python
|
71
|
-
upload_tiktok(video=video_path, description=description, sound_name='trending_sound')
|
72
|
+
upload_tiktok(video=video_path, description=description, sound_name='trending_sound', sound_aud_vol='main')
|
72
73
|
```
|
73
74
|
|
74
75
|
PLEASE READ DOCUMENTATION FOR MORE INFO.
|
@@ -101,8 +102,6 @@ This library requires the following dependencies:
|
|
101
102
|
- `playwright`
|
102
103
|
- `requests`
|
103
104
|
- `Pillow`
|
104
|
-
- `transformers`
|
105
|
-
- `torch`
|
106
105
|
- `scikit-learn`
|
107
106
|
- `inference`
|
108
107
|
|
Binary file
|
Binary file
|
@@ -4,7 +4,7 @@ build-backend = "hatchling.build"
|
|
4
4
|
|
5
5
|
[project]
|
6
6
|
name = "tiktokautouploader"
|
7
|
-
version = "2.
|
7
|
+
version = "2.8"
|
8
8
|
description = "Upload or schedule videos to TikTok with TikTok sounds and hashtags that work."
|
9
9
|
long-description = "file: README.md"
|
10
10
|
long-description-content-type = "text/markdown"
|
@@ -36,7 +36,7 @@ tags = ["wheel"]
|
|
36
36
|
|
37
37
|
[tool.hatch.metadata]
|
38
38
|
name = "tiktokautouploader"
|
39
|
-
version = "2.
|
39
|
+
version = "2.8"
|
40
40
|
description = "Upload or schedule videos to TikTok with TikTok sounds and hashtags that work."
|
41
41
|
long_description = "file: README.md"
|
42
42
|
long_description_content_type = "text/markdown"
|
@@ -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
|
})();
|
@@ -11,8 +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
14
|
|
17
15
|
def login_warning():
|
18
16
|
print("NO COOKIES FILE FOUND, PLEASE LOG-IN WHEN PROMPTED")
|
@@ -24,7 +22,6 @@ def save_cookies(cookies):
|
|
24
22
|
def run_javascript():
|
25
23
|
js_file_path = pkg_resources.resource_filename(__name__, 'Js_assets/login.js')
|
26
24
|
result = subprocess.run(['node', js_file_path], capture_output=True, text=True)
|
27
|
-
return result.stdout, result.stderr
|
28
25
|
|
29
26
|
def install_js_dependencies():
|
30
27
|
js_dir = pkg_resources.resource_filename(__name__, 'Js_assets')
|
@@ -34,7 +31,6 @@ def install_js_dependencies():
|
|
34
31
|
print("JavaScript dependencies not found. Installing...")
|
35
32
|
subprocess.run(['npm', 'install', 'playwright', 'playwright-extra', 'puppeteer-extra-plugin-stealth', '--silent'], cwd=js_dir, check=True)
|
36
33
|
else:
|
37
|
-
print("found dependencies")
|
38
34
|
time.sleep(0.1)
|
39
35
|
|
40
36
|
|
@@ -197,7 +193,6 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
197
193
|
video (str) -> path to video to upload
|
198
194
|
description (str) -> description for video
|
199
195
|
hashtags (str)(array) -> hashtags for video
|
200
|
-
cookies_path (str) -> path to tik tok cookies .json file
|
201
196
|
sound_name (str) -> name of tik tok sound to use for video
|
202
197
|
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
198
|
schedule (str) -> format HH:MM, your local time to upload video
|
@@ -224,11 +219,14 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
224
219
|
|
225
220
|
|
226
221
|
with sync_playwright() as p:
|
227
|
-
|
222
|
+
|
223
|
+
browser = p.firefox.launch(headless=True)
|
224
|
+
|
228
225
|
context = browser.new_context()
|
229
226
|
context.add_cookies(cookies)
|
230
227
|
page = context.new_page()
|
231
228
|
url = 'https://www.tiktok.com/tiktokstudio/upload?from=upload&lang=en'
|
229
|
+
url2 = 'https://www.tiktok.com/tiktokstudio/content?tab=draft'
|
232
230
|
|
233
231
|
while retries < 2:
|
234
232
|
try:
|
@@ -245,6 +243,7 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
245
243
|
captcha = False
|
246
244
|
while detected == False:
|
247
245
|
if page.locator('.upload-text-container').is_visible():
|
246
|
+
print("Showed up")
|
248
247
|
detected = True
|
249
248
|
else:
|
250
249
|
if page.locator('div.VerifyBar___StyledDiv-sc-12zaxoy-0.hRJhHT').is_visible():
|
@@ -254,7 +253,6 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
254
253
|
time.sleep(0.1)
|
255
254
|
|
256
255
|
if captcha == True:
|
257
|
-
print("CAPTCHA FOUND")
|
258
256
|
image = get_image_src(page)
|
259
257
|
if image:
|
260
258
|
if suppressprint == False:
|
@@ -291,7 +289,7 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
291
289
|
if attempts > 5:
|
292
290
|
sys.exit("FAILED TO SOLVE CAPTCHA")
|
293
291
|
try:
|
294
|
-
page.wait_for_selector('.upload-text-container', timeout=
|
292
|
+
page.wait_for_selector('.upload-text-container', timeout=5000)
|
295
293
|
os.remove('captcha_image.jpg')
|
296
294
|
if suppressprint == False:
|
297
295
|
print("Captcha Solved")
|
@@ -308,7 +306,6 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
308
306
|
while objectclick == 'N.A':
|
309
307
|
page.click('span.secsdk_captcha_refresh--text')
|
310
308
|
page.wait_for_selector('div.VerifyBar___StyledDiv-sc-12zaxoy-0.hRJhHT')
|
311
|
-
time.sleep(2)
|
312
309
|
question = page.locator('div.VerifyBar___StyledDiv-sc-12zaxoy-0.hRJhHT').text_content()
|
313
310
|
objectclick = understood_Qs(question)
|
314
311
|
image = get_image_src(page)
|
@@ -336,7 +333,7 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
336
333
|
if attempts > 5:
|
337
334
|
sys.exit("FAILED TO SOLVE CAPTCHA")
|
338
335
|
try:
|
339
|
-
page.wait_for_selector('.upload-text-container', timeout=
|
336
|
+
page.wait_for_selector('.upload-text-container', timeout=5000)
|
340
337
|
solved = True
|
341
338
|
os.remove('captcha_image.jpg')
|
342
339
|
if suppressprint == False:
|
@@ -344,13 +341,11 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
344
341
|
except:
|
345
342
|
continue
|
346
343
|
|
347
|
-
|
348
344
|
|
349
345
|
try:
|
350
346
|
page.set_input_files('input[type="file"][accept="video/*"]', f'{video}')
|
351
347
|
except:
|
352
348
|
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
349
|
page.wait_for_selector('div[data-contents="true"]')
|
355
350
|
page.click('div[data-contents="true"]')
|
356
351
|
if suppressprint == False:
|
@@ -400,75 +395,10 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
400
395
|
except:
|
401
396
|
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
397
|
|
403
|
-
time.sleep(0.
|
398
|
+
time.sleep(0.2)
|
404
399
|
if suppressprint == False:
|
405
400
|
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"]')
|
401
|
+
|
472
402
|
if schedule != None:
|
473
403
|
try:
|
474
404
|
hour = schedule[0:2]
|
@@ -480,25 +410,25 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
480
410
|
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
411
|
|
482
412
|
page.locator('div.TUXRadioStandalone.TUXRadioStandalone--medium').nth(1).click()
|
483
|
-
time.sleep(0.
|
413
|
+
time.sleep(0.2)
|
484
414
|
if day != None:
|
485
415
|
page.locator('div.TUXTextInputCore-trailingIconWrapper').nth(1).click()
|
486
|
-
time.sleep(0.
|
416
|
+
time.sleep(0.2)
|
487
417
|
try:
|
488
418
|
page.locator(f'span.day.valid:has-text("{day}")').click()
|
489
419
|
except:
|
490
420
|
sys.exit("SCHEDULE DAY ERROR: ERROR WITH SCHEDULED DAY, read documentation for more information on format of day")
|
491
421
|
try:
|
422
|
+
time.sleep(1)
|
492
423
|
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)
|
424
|
+
time.sleep(0.2)
|
499
425
|
page.locator(f'.tiktok-timepicker-option-text.tiktok-timepicker-right:has-text("{minute}")').nth(0).scroll_into_view_if_needed()
|
500
|
-
time.sleep(0.
|
426
|
+
time.sleep(0.2)
|
501
427
|
page.locator(f'.tiktok-timepicker-option-text.tiktok-timepicker-right:has-text("{minute}")').nth(0).click()
|
428
|
+
time.sleep(0.2)
|
429
|
+
page.locator(f'.tiktok-timepicker-option-text:has-text("{hour}")').nth(0).scroll_into_view_if_needed()
|
430
|
+
page.locator(f'.tiktok-timepicker-option-text:has-text("{hour}")').nth(0).click()
|
431
|
+
time.sleep(1)
|
502
432
|
|
503
433
|
if suppressprint == False:
|
504
434
|
print("Done scheduling video")
|
@@ -506,28 +436,29 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
506
436
|
except:
|
507
437
|
sys.exit("SCHEDULING ERROR: VIDEO SAVED AS DRAFT")
|
508
438
|
|
439
|
+
if (schedule == None) and (day != None):
|
440
|
+
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
441
|
|
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
442
|
|
522
|
-
|
443
|
+
if(sound_name == None):
|
444
|
+
if copyrightcheck == True:
|
445
|
+
page.locator(".TUXSwitch-input").nth(0).click()
|
446
|
+
while copyrightcheck == True:
|
447
|
+
time.sleep(0.2)
|
448
|
+
if page.locator("span", has_text="No issues detected.").is_visible():
|
449
|
+
if suppressprint == False:
|
450
|
+
print("Copyright check complete")
|
451
|
+
break
|
452
|
+
if page.locator("span", has_text="Copyright issues detected.").is_visible():
|
453
|
+
sys.exit("COPYRIGHT CHECK FAILED: COPYRIGHT AUDIO DETECTED FROM TIKTOK")
|
454
|
+
|
523
455
|
if schedule == None:
|
524
456
|
page.click('button.TUXButton.TUXButton--default.TUXButton--large.TUXButton--primary:has-text("Post")', timeout=10000)
|
525
457
|
uploaded = False
|
526
458
|
checks = 0
|
527
459
|
while uploaded == False:
|
528
460
|
if page.locator(':has-text("Leaving the page does not interrupt")').nth(0).is_visible():
|
529
|
-
|
530
|
-
time.sleep(1)
|
461
|
+
time.sleep(0.2)
|
531
462
|
break
|
532
463
|
time.sleep(0.2)
|
533
464
|
checks += 1
|
@@ -541,7 +472,7 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
541
472
|
checks = 0
|
542
473
|
while uploaded == False:
|
543
474
|
if page.locator(':has-text("Leaving the page does not interrupt")').nth(0).is_visible():
|
544
|
-
time.sleep(
|
475
|
+
time.sleep(0.2)
|
545
476
|
break
|
546
477
|
time.sleep(0.2)
|
547
478
|
checks += 1
|
@@ -551,9 +482,148 @@ def upload_tiktok(video, description, hashtags=None, sound_name=None, sound_aud_
|
|
551
482
|
break
|
552
483
|
if suppressprint == False:
|
553
484
|
print("Done uploading video, NOTE: it may take a minute or two to show on TikTok")
|
554
|
-
|
555
|
-
|
556
|
-
|
557
|
-
|
485
|
+
|
486
|
+
page.close()
|
487
|
+
|
488
|
+
else:
|
489
|
+
try:
|
490
|
+
page.click('button.TUXButton.TUXButton--default.TUXButton--large.TUXButton--secondary:has-text("Save draft")', timeout=10000)
|
491
|
+
except:
|
492
|
+
sys.exit("SAVE AS DRAFT BUTTON NOT FOUND; CANNOT ADD SOUND WITHOUT ABILITY TO SAVE DRAFTS")
|
493
|
+
|
494
|
+
time.sleep(0.5)
|
495
|
+
page.close()
|
496
|
+
|
497
|
+
browser = p.chromium.launch(headless=True)
|
498
|
+
|
499
|
+
context = browser.new_context()
|
500
|
+
context.add_cookies(cookies)
|
501
|
+
page = context.new_page()
|
502
|
+
url2 = 'https://www.tiktok.com/tiktokstudio/content?tab=draft'
|
503
|
+
|
504
|
+
while retries < 2:
|
505
|
+
try:
|
506
|
+
page.goto(url2, timeout=30000)
|
507
|
+
except:
|
508
|
+
retries +=1
|
509
|
+
time.sleep(5)
|
510
|
+
if retries == 2:
|
511
|
+
sys.exit("ERROR: TIK TOK PAGE FAILED TO LOAD, try again.")
|
512
|
+
else:
|
513
|
+
break
|
514
|
+
|
515
|
+
try:
|
516
|
+
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']")
|
517
|
+
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']")
|
518
|
+
page.wait_for_selector('div[data-contents="true"]')
|
519
|
+
page.wait_for_function("document.querySelector('.info-progress-num').textContent.trim() === '100%'", timeout=3000000)
|
520
|
+
time.sleep(0.2)
|
521
|
+
except:
|
522
|
+
sys.exit("ERROR ADDING SOUND: Video saved as draft")
|
523
|
+
|
524
|
+
if sound_name != None:
|
525
|
+
page.click("div.TUXButton-label:has-text('Edit video')")
|
526
|
+
page.wait_for_selector("input.search-bar-input")
|
527
|
+
page.fill(f"input.search-bar-input", f"{sound_name}")
|
528
|
+
time.sleep(0.2)
|
529
|
+
page.click("div.TUXButton-label:has-text('Search')")
|
530
|
+
try:
|
531
|
+
page.wait_for_selector('div.music-card-container')
|
532
|
+
page.click("div.music-card-container")
|
533
|
+
page.wait_for_selector("div.TUXButton-label:has-text('Use')")
|
534
|
+
page.click("div.TUXButton-label:has-text('Use')")
|
535
|
+
except:
|
536
|
+
sys.exit(f"ERROR: SOUND '{sound_name}' NOT FOUND")
|
537
|
+
try:
|
538
|
+
page.wait_for_selector('img[src=""]')
|
539
|
+
page.click('img[src=""]')
|
540
|
+
time.sleep(0.5)
|
541
|
+
sliders = page.locator("input.scaleInput")
|
542
|
+
|
543
|
+
if sound_aud_vol == 'background':
|
544
|
+
slider1 = sliders.nth(0)
|
545
|
+
bounding_box1 = slider1.bounding_box()
|
546
|
+
if bounding_box1:
|
547
|
+
x1 = bounding_box1["x"] + (bounding_box1["width"] * 0.92)
|
548
|
+
y1 = bounding_box1["y"] + bounding_box1["height"] / 2
|
549
|
+
page.mouse.click(x1, y1)
|
550
|
+
|
551
|
+
slider2 = sliders.nth(1)
|
552
|
+
bounding_box2 = slider2.bounding_box()
|
553
|
+
if bounding_box2:
|
554
|
+
x2 = bounding_box2["x"] + (bounding_box2["width"] * 0.097)
|
555
|
+
y2 = bounding_box2["y"] + bounding_box2["height"] / 2
|
556
|
+
page.mouse.click(x2, y2)
|
557
|
+
|
558
|
+
if sound_aud_vol == 'main':
|
559
|
+
slider1 = sliders.nth(0)
|
560
|
+
bounding_box1 = slider1.bounding_box()
|
561
|
+
if bounding_box1:
|
562
|
+
x1 = bounding_box1["x"] + (bounding_box1["width"] * 0.092)
|
563
|
+
y1 = bounding_box1["y"] + bounding_box1["height"] / 2
|
564
|
+
page.mouse.click(x1, y1)
|
565
|
+
slider2 = sliders.nth(1)
|
566
|
+
bounding_box2 = slider2.bounding_box()
|
567
|
+
if bounding_box2:
|
568
|
+
x2 = bounding_box2["x"] + (bounding_box2["width"] * 0.92)
|
569
|
+
y2 = bounding_box2["y"] + bounding_box2["height"] / 2
|
570
|
+
page.mouse.click(x2, y2)
|
571
|
+
except:
|
572
|
+
sys.exit("ERROR ADJUSTING SOUND VOLUME: please try again.")
|
573
|
+
|
574
|
+
page.wait_for_selector("div.TUXButton-label:has-text('Save edit')")
|
575
|
+
page.click("div.TUXButton-label:has-text('Save edit')")
|
576
|
+
if suppressprint == False:
|
577
|
+
print("Added sound")
|
578
|
+
|
579
|
+
page.wait_for_selector('div[data-contents="true"]')
|
580
|
+
|
581
|
+
if copyrightcheck == True:
|
582
|
+
page.locator(".TUXSwitch-input").nth(0).click()
|
583
|
+
while copyrightcheck == True:
|
584
|
+
time.sleep(0.2)
|
585
|
+
if page.locator("span", has_text="No issues detected.").is_visible():
|
586
|
+
if suppressprint == False:
|
587
|
+
print("Copyright check complete")
|
588
|
+
break
|
589
|
+
if page.locator("span", has_text="Copyright issues detected.").is_visible():
|
590
|
+
sys.exit("COPYRIGHT CHECK FAILED: VIDEO SAVED AS DRAFT, COPYRIGHT AUDIO DETECTED FROM TIKTOK")
|
591
|
+
|
592
|
+
|
593
|
+
try:
|
594
|
+
if schedule == None:
|
595
|
+
page.click('button.TUXButton.TUXButton--default.TUXButton--large.TUXButton--primary:has-text("Post")', timeout=10000)
|
596
|
+
uploaded = False
|
597
|
+
checks = 0
|
598
|
+
while uploaded == False:
|
599
|
+
if page.locator(':has-text("Leaving the page does not interrupt")').nth(0).is_visible():
|
600
|
+
time.sleep(0.2)
|
601
|
+
break
|
602
|
+
time.sleep(0.2)
|
603
|
+
checks += 1
|
604
|
+
if checks > 100:
|
605
|
+
time.sleep(10)
|
606
|
+
if checks == 150:
|
607
|
+
break
|
608
|
+
else:
|
609
|
+
page.click('button.TUXButton.TUXButton--default.TUXButton--large.TUXButton--primary:has-text("Schedule")', timeout=10000)
|
610
|
+
uploaded = False
|
611
|
+
checks = 0
|
612
|
+
while uploaded == False:
|
613
|
+
if page.locator(':has-text("Leaving the page does not interrupt")').nth(0).is_visible():
|
614
|
+
time.sleep(0.2)
|
615
|
+
break
|
616
|
+
time.sleep(0.2)
|
617
|
+
checks += 1
|
618
|
+
if checks > 100:
|
619
|
+
time.sleep(10)
|
620
|
+
if checks == 150:
|
621
|
+
break
|
622
|
+
if suppressprint == False:
|
623
|
+
print("Done uploading video, NOTE: it may take a minute or two to show on TikTok")
|
624
|
+
except:
|
625
|
+
time.sleep(5)
|
626
|
+
sys.exit("ERROR UPLOADING: VIDEO HAS SAVED AS DRAFT BUT CANT UPLOAD")
|
627
|
+
time.sleep(1)
|
558
628
|
|
559
|
-
|
629
|
+
page.close()
|
Binary file
|
Binary file
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|