Skip to content

Commit 08a4ebc

Browse files
Merge branch 'master' into snyk-fix-688d5aa50af99bbf53ce9b383e137980
2 parents 768f99a + 05b3d82 commit 08a4ebc

18 files changed

+456
-83
lines changed

.github/workflows/cloudinary-cli-test.yml

+3-3
Original file line numberDiff line numberDiff line change
@@ -13,12 +13,12 @@ jobs:
1313
strategy:
1414
fail-fast: false
1515
matrix:
16-
python-version: ["3.7", "3.8", "3.9", "3.10", "3.11"]
16+
python-version: ["3.9", "3.10", "3.11", "3.12", "3.13"]
1717

1818
steps:
19-
- uses: actions/checkout@v2
19+
- uses: actions/checkout@v4
2020
- name: Set up Python ${{ matrix.python-version }}
21-
uses: actions/setup-python@v2
21+
uses: actions/setup-python@v5
2222
with:
2323
python-version: ${{ matrix.python-version }}
2424
- name: Install dependencies

CHANGELOG.md

+10
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,13 @@
1+
1.10.3 / 2024-07-31
2+
==================
3+
4+
* Add missing delivery types in `url` command
5+
6+
1.10.2 / 2024-07-15
7+
==================
8+
9+
* Fix folder existence error in `sync` command
10+
111
1.10.1 / 2024-01-29
212
==================
313

Dockerfile

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
# syntax=docker/dockerfile:1
2+
3+
ARG PYTHON_VERSION
4+
FROM python:${PYTHON_VERSION:-3.12-slim}
5+
6+
# For available labels, see OCI Annotations Spec docs:
7+
# https://specs.opencontainers.org/image-spec/annotations/#pre-defined-annotation-keys
8+
LABEL org.opencontainers.image.source="https://github.com/cloudinary/cloudinary-cli"
9+
10+
RUN pip3 install --no-cache cloudinary-cli
11+
12+
ENTRYPOINT [ "cloudinary" ]

cloudinary_cli/core/__init__.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22

33
from cloudinary_cli.core.admin import admin
44
from cloudinary_cli.core.config import config
5-
from cloudinary_cli.core.search import search
5+
from cloudinary_cli.core.search import search, search_folders
66
from cloudinary_cli.core.uploader import uploader
77
from cloudinary_cli.core.provisioning import provisioning
88
from cloudinary_cli.core.utils import url, utils
@@ -13,6 +13,7 @@
1313
commands = [
1414
config,
1515
search,
16+
search_folders,
1617
admin,
1718
uploader,
1819
provisioning,

cloudinary_cli/core/search.py

+73-31
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import cloudinary
22
from click import command, argument, option, launch
3+
from functools import wraps
34

45
from cloudinary_cli.defaults import logger
56
from cloudinary_cli.utils.json_utils import write_json_to_file, print_json
@@ -9,45 +10,86 @@
910
DEFAULT_MAX_RESULTS = 500
1011

1112

13+
def shared_options(func):
14+
@option("-f", "--with_field", multiple=True, help="Specify which non-default asset attributes to include "
15+
"in the result as a comma separated list.")
16+
@option("-fi", "--fields", multiple=True, help="Specify which asset attributes to include in the result "
17+
"(together with a subset of the default attributes) as a comma separated"
18+
" list. This overrides any value specified for with_field.")
19+
@option("-s", "--sort_by", nargs=2, help="Sort search results by (field, <asc|desc>).")
20+
@option("-a", "--aggregate", nargs=1,
21+
help="Specify the attribute for which an aggregation count should be calculated and returned.")
22+
@option("-n", "--max_results", nargs=1, default=10,
23+
help="The maximum number of results to return. Default: 10, maximum: 500.")
24+
@option("-c", "--next_cursor", nargs=1, help="Continue a search using an existing cursor.")
25+
@option("-A", "--auto_paginate", is_flag=True, help="Return all results. Will call Admin API multiple times.")
26+
@option("-F", "--force", is_flag=True, help="Skip confirmation when running --auto-paginate.")
27+
@option("-ff", "--filter_fields", multiple=True, help="Specify which attributes to show in the response. "
28+
"None of the others will be shown.")
29+
@option("-sq", "--search-query", is_flag=True, help="Show the search request query.", hidden=True)
30+
@option("--json", nargs=1, help="Save JSON output to a file. Usage: --json <filename>")
31+
@option("--csv", nargs=1, help="Save CSV output to a file. Usage: --csv <filename>")
32+
@wraps(func)
33+
def wrapper(*args, **kwargs):
34+
return func(*args, **kwargs)
35+
36+
return wrapper
37+
38+
1239
@command("search",
13-
short_help="Run the admin API search method.",
40+
short_help="Run the Admin API search method.",
1441
help="""\b
15-
Run the admin API search method.
42+
Run the Admin API search method.
1643
Format: cld <cli options> search <command options> <Lucene query syntax string>
1744
e.g. cld search cat AND tags:kitten -s public_id desc -f context -f tags -n 10
1845
""")
1946
@argument("query", nargs=-1)
20-
@option("-f", "--with_field", multiple=True, help="Specify which non-default asset attributes to include "
21-
"in the result as a comma separated list. ")
22-
@option("-fi", "--fields", multiple=True, help="Specify which asset attributes to include in the result "
23-
"(together with a subset of the default attributes) as a comma separated"
24-
" list. This overrides any value specified for with_field.")
25-
@option("-s", "--sort_by", nargs=2, help="Sort search results by (field, <asc|desc>).")
26-
@option("-a", "--aggregate", nargs=1,
27-
help="Specify the attribute for which an aggregation count should be calculated and returned.")
28-
@option("-n", "--max_results", nargs=1, default=10,
29-
help="The maximum number of results to return. Default: 10, maximum: 500.")
30-
@option("-c", "--next_cursor", nargs=1, help="Continue a search using an existing cursor.")
31-
@option("-A", "--auto_paginate", is_flag=True, help="Return all results. Will call Admin API multiple times.")
32-
@option("-F", "--force", is_flag=True, help="Skip confirmation when running --auto-paginate.")
33-
@option("-ff", "--filter_fields", multiple=True, help="Specify which attributes to show in the response. "
34-
"None of the others will be shown.")
47+
@shared_options
3548
@option("-t", "--ttl", nargs=1, default=300, help="Set the Search URL TTL in seconds. Default: 300.")
3649
@option("-u", "--url", is_flag=True, help="Build a signed search URL.")
37-
@option("-sq", "--search-query", is_flag=True, help="Show the search request query.", hidden=True)
38-
@option("--json", nargs=1, help="Save JSON output to a file. Usage: --json <filename>")
39-
@option("--csv", nargs=1, help="Save CSV output to a file. Usage: --csv <filename>")
4050
@option("-d", "--doc", is_flag=True, help="Open Search API documentation page.")
4151
def search(query, with_field, fields, sort_by, aggregate, max_results, next_cursor,
4252
auto_paginate, force, filter_fields, ttl, url, search_query, json, csv, doc):
53+
search_instance = cloudinary.search.Search()
54+
doc_url = "https://cloudinary.com/documentation/search_api"
55+
result_field = 'resources'
56+
return _perform_search(query, with_field, fields, sort_by, aggregate, max_results, next_cursor,
57+
auto_paginate, force, filter_fields, ttl, url, search_query, json, csv, doc,
58+
search_instance, doc_url, result_field)
59+
60+
61+
@command("search_folders",
62+
short_help="Run the Admin API search folders method.",
63+
help="""\b
64+
Run the Admin API search folders method.
65+
Format: cld <cli options> search_folders <command options> <Lucene query syntax string>
66+
e.g. cld search_folders name:folder AND path:my_parent AND created_at>4w
67+
""")
68+
@argument("query", nargs=-1)
69+
@shared_options
70+
@option("-d", "--doc", is_flag=True, help="Open Search Folders API documentation page.")
71+
def search_folders(query, with_field, fields, sort_by, aggregate, max_results, next_cursor,
72+
auto_paginate, force, filter_fields, search_query, json, csv, doc):
73+
search_instance = cloudinary.search_folders.SearchFolders()
74+
doc_url = "https://cloudinary.com/documentation/admin_api#search_folders"
75+
result_field = 'folders'
76+
return _perform_search(query, with_field, fields, sort_by, aggregate, max_results, next_cursor,
77+
auto_paginate, force, filter_fields, 300, False, search_query, json, csv, doc,
78+
search_instance, doc_url, result_field)
79+
80+
81+
def _perform_search(query, with_field, fields, sort_by, aggregate, max_results, next_cursor,
82+
auto_paginate, force, filter_fields, ttl, url, search_query, json, csv, doc,
83+
search_instance, doc_url, result_field):
84+
"""Shared logic for running a search."""
4385
if doc:
44-
return launch("https://cloudinary.com/documentation/search_api")
86+
return launch(doc_url)
4587

4688
fields_to_keep = []
4789
if filter_fields:
4890
fields_to_keep = tuple(normalize_list_params(filter_fields)) + tuple(normalize_list_params(with_field))
4991

50-
search = cloudinary.search.Search().expression(" ".join(query))
92+
search = search_instance.expression(" ".join(query))
5193

5294
if auto_paginate:
5395
max_results = DEFAULT_MAX_RESULTS
@@ -74,32 +116,32 @@ def search(query, with_field, fields, sort_by, aggregate, max_results, next_curs
74116
print_json(search.as_dict())
75117
return True
76118

77-
res = execute_single_request(search, fields_to_keep)
119+
res = execute_single_request(search, fields_to_keep, result_field)
78120

79121
if auto_paginate:
80-
res = handle_auto_pagination(res, search, force, fields_to_keep)
122+
res = handle_auto_pagination(res, search, force, fields_to_keep, result_field)
81123

82124
print_json(res)
83125

84126
if json:
85-
write_json_to_file(res['resources'], json)
127+
write_json_to_file(res[result_field], json)
86128
logger.info(f"Saved search JSON to '{json}' file")
87129

88130
if csv:
89-
write_json_list_to_csv(res['resources'], csv, fields_to_keep)
131+
write_json_list_to_csv(res[result_field], csv, fields_to_keep)
90132
logger.info(f"Saved search to '{csv}.csv' file")
91133

92134

93-
def execute_single_request(expression, fields_to_keep):
135+
def execute_single_request(expression, fields_to_keep, result_field='resources'):
94136
res = expression.execute()
95137

96138
if fields_to_keep:
97-
res['resources'] = whitelist_keys(res['resources'], fields_to_keep)
139+
res[result_field] = whitelist_keys(res[result_field], fields_to_keep)
98140

99141
return res
100142

101143

102-
def handle_auto_pagination(res, expression, force, fields_to_keep):
144+
def handle_auto_pagination(res, expression, force, fields_to_keep, result_field='resources'):
103145
if 'next_cursor' not in res:
104146
return res
105147

@@ -119,9 +161,9 @@ def handle_auto_pagination(res, expression, force, fields_to_keep):
119161
while 'next_cursor' in res.keys():
120162
expression.next_cursor(res['next_cursor'])
121163

122-
res = execute_single_request(expression, fields_to_keep)
164+
res = execute_single_request(expression, fields_to_keep, result_field)
123165

124-
all_results['resources'] += res['resources']
166+
all_results[result_field] += res[result_field]
125167
all_results['time'] += res['time']
126168

127169
all_results.pop('next_cursor', None) # it is empty by now

cloudinary_cli/core/utils.py

+5-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,11 @@ def utils(params, optional_parameter, optional_parameter_parsed, ls):
3636
@argument("transformation", default="")
3737
@option("-rt", "--resource_type", default="image", type=Choice(['image', 'video', 'raw']), help="The asset type")
3838
@option("-t", "--type", "delivery_type", default="upload",
39-
type=Choice(['upload', 'private', 'authenticated', 'fetch', 'list', 'url2png']),
39+
type=Choice([
40+
'upload', 'private', 'public', 'authenticated', 'fetch', 'list', 'url2png',
41+
'sprite', 'text', 'multi', 'facebook', 'twitter', 'twitter_name', 'gravatar',
42+
'youtube', 'hulu', 'vimeo', 'animoto', 'worldstarhiphop', 'dailymotion'
43+
]),
4044
help="The delivery type.")
4145
@option("-o", "--open", 'open_in_browser', is_flag=True, help="Generate the derived asset and open it in your browser.")
4246
@option("-s", "--sign", is_flag=True, help="Generate a signed URL.", default=False)

cloudinary_cli/modules/__init__.py

+3-1
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,13 @@
33
from .sync import sync
44
from .upload_dir import upload_dir
55
from .regen_derived import regen_derived
6+
from .clone import clone
67

78
commands = [
89
upload_dir,
910
make,
1011
migrate,
1112
sync,
12-
regen_derived
13+
regen_derived,
14+
clone
1315
]

cloudinary_cli/modules/clone.py

+112
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,112 @@
1+
from click import command, option, style, argument
2+
from cloudinary_cli.utils.utils import normalize_list_params, print_help_and_exit
3+
import cloudinary
4+
from cloudinary_cli.utils.utils import run_tasks_concurrently
5+
from cloudinary_cli.utils.api_utils import upload_file
6+
from cloudinary_cli.utils.config_utils import load_config, get_cloudinary_config, config_to_dict
7+
from cloudinary_cli.defaults import logger
8+
from cloudinary_cli.core.search import execute_single_request, handle_auto_pagination
9+
10+
DEFAULT_MAX_RESULTS = 500
11+
12+
13+
@command("clone",
14+
short_help="""Clone assets from one product environment to another.""",
15+
help="""
16+
\b
17+
Clone assets from one product environment to another with/without tags and/or context (structured metadata is not currently supported).
18+
Source will be your `CLOUDINARY_URL` environment variable but you also can specify a different source using the `-c/-C` option.
19+
Cloning restricted assets is also not supported currently.
20+
Format: cld clone <target_environment> <command options>
21+
`<target_environment>` can be a CLOUDINARY_URL or a saved config (see `config` command)
22+
Example 1 (Copy all assets including tags and context using CLOUDINARY URL):
23+
cld clone cloudinary://<api_key>:<api_secret>@<cloudname> -fi tags,context
24+
Example 2 (Copy all assets with a specific tag via a search expression using a saved config):
25+
cld clone <config_name> -se "tags:<tag_name>"
26+
""")
27+
@argument("target")
28+
@option("-F", "--force", is_flag=True,
29+
help="Skip confirmation.")
30+
@option("-ow", "--overwrite", is_flag=True, default=False,
31+
help="Specify whether to overwrite existing assets.")
32+
@option("-w", "--concurrent_workers", type=int, default=30,
33+
help="Specify the number of concurrent network threads.")
34+
@option("-fi", "--fields", multiple=True,
35+
help="Specify whether to copy tags and/or context. Valid options: `tags,context`.")
36+
@option("-se", "--search_exp", default="",
37+
help="Define a search expression to filter the assets to clone.")
38+
@option("--async", "async_", is_flag=True, default=False,
39+
help="Clone the assets asynchronously.")
40+
@option("-nu", "--notification_url",
41+
help="Webhook notification URL.")
42+
def clone(target, force, overwrite, concurrent_workers, fields, search_exp, async_, notification_url):
43+
if not target:
44+
print_help_and_exit()
45+
46+
target_config = get_cloudinary_config(target)
47+
if not target_config:
48+
logger.error("The specified config does not exist or the CLOUDINARY_URL scheme provided is invalid"
49+
" (expecting to start with 'cloudinary://').")
50+
return False
51+
52+
if cloudinary.config().cloud_name == target_config.cloud_name:
53+
logger.error("Target environment cannot be the same as source environment.")
54+
return False
55+
56+
source_assets = search_assets(force, search_exp)
57+
58+
upload_list = []
59+
for r in source_assets.get('resources'):
60+
updated_options, asset_url = process_metadata(r, overwrite, async_, notification_url,
61+
normalize_list_params(fields))
62+
updated_options.update(config_to_dict(target_config))
63+
upload_list.append((asset_url, {**updated_options}))
64+
65+
if not upload_list:
66+
logger.error(style(f'No assets found in {cloudinary.config().cloud_name}', fg="red"))
67+
return False
68+
69+
logger.info(style(f'Copying {len(upload_list)} asset(s) from {cloudinary.config().cloud_name} to {target_config.cloud_name}', fg="blue"))
70+
71+
run_tasks_concurrently(upload_file, upload_list, concurrent_workers)
72+
73+
return True
74+
75+
76+
def search_assets(force, search_exp):
77+
search = cloudinary.search.Search().expression(search_exp)
78+
search.fields(['tags', 'context', 'access_control', 'secure_url', 'display_name'])
79+
search.max_results(DEFAULT_MAX_RESULTS)
80+
81+
res = execute_single_request(search, fields_to_keep="")
82+
res = handle_auto_pagination(res, search, force, fields_to_keep="")
83+
84+
return res
85+
86+
87+
def process_metadata(res, overwrite, async_, notification_url, copy_fields=""):
88+
cloned_options = {}
89+
asset_url = res.get('secure_url')
90+
cloned_options['public_id'] = res.get('public_id')
91+
cloned_options['type'] = res.get('type')
92+
cloned_options['resource_type'] = res.get('resource_type')
93+
cloned_options['overwrite'] = overwrite
94+
cloned_options['async'] = async_
95+
if "tags" in copy_fields:
96+
cloned_options['tags'] = res.get('tags')
97+
if "context" in copy_fields:
98+
cloned_options['context'] = res.get('context')
99+
if res.get('folder'):
100+
# This is required to put the asset in the correct asset_folder
101+
# when copying from a fixed to DF (dynamic folder) cloud as if
102+
# you just pass a `folder` param to a DF cloud, it will append
103+
# this to the `public_id` and we don't want this.
104+
cloned_options['asset_folder'] = res.get('folder')
105+
elif res.get('asset_folder'):
106+
cloned_options['asset_folder'] = res.get('asset_folder')
107+
if res.get('display_name'):
108+
cloned_options['display_name'] = res.get('display_name')
109+
if notification_url:
110+
cloned_options['notification_url'] = notification_url
111+
112+
return cloned_options, asset_url

0 commit comments

Comments
 (0)