Skip to content

Commit eb7146e

Browse files
authored
Merge pull request #153 from developmentseed/enh/152-auth
HTTP authentication
2 parents 089ce95 + 8b8663a commit eb7146e

12 files changed

+49
-33
lines changed

.circleci/config.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ version: 2
22
jobs:
33
build:
44
docker:
5-
- image: circleci/python:3.5
5+
- image: circleci/python:3.6-stretch
66

77
working_directory: ~/label-maker
88
steps:

CHANGES.txt

+5-3
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,15 @@
1-
0.6.0 (2018-11-06)
1+
0.6.1 (2019-11-11)
2+
------------------
3+
- Added ability to use HTTP Authentication for TMS endpoints (#152)
4+
5+
0.6.0 (2019-11-06)
26
------------------
37
- Use sys.exectuable in place of python string (#124)
48
- Correct script reference to fix bug in skynet train example (#129)
59
- Add s3 requirement to rasterio (#137)
610
- users can split data into more groups than train and test, for example train/test/validate, and specify the ratio for
711
each split (#149)
812

9-
10-
1113
0.5.1 (2018-11-12)
1214
------------------
1315
- Skip invalid or empty geometries which prevent segmentation rendering (#118)

docs/parameters.rst

+3
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,9 @@ Here is the full list of configuration parameters you can specify in a ``config.
3434

3535
Remote files like a `WMS endpoint <http://www.opengeospatial.org/standards/wms>`_ ``GetMap`` request. Fill out all necessary parameters except ``bbox`` which should be set as ``{bbox}``. Ex: ``'https://basemap.nationalmap.gov/arcgis/services/USGSImageryOnly/MapServer/WMSServer?SERVICE=WMS&REQUEST=GetMap&VERSION=1.1.1&LAYERS=0&STYLES=&FORMAT=image%2Fjpeg&TRANSPARENT=false&HEIGHT=256&WIDTH=256&SRS=EPSG%3A3857&BBOX={bbox}'``
3636

37+
**http_auth**: list
38+
Optional parameter to specify a username and password for restricted WMS services. For example, ``['my_username', 'my_password']``.
39+
3740
**background_ratio**: float
3841
Specify how many background (or "negative") training examples to create when there is only one class specified with the ``classes`` parameter. Label Maker will generate ``background_ratio`` times the number of images matching the one class.
3942

label_maker/images.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -72,4 +72,4 @@ def class_test(value):
7272
image_function = get_image_function(imagery)
7373

7474
for tile in tiles:
75-
image_function(tile, imagery, tiles_dir, imagery_offset)
75+
image_function(tile, imagery, tiles_dir, imagery_offset, kwargs)

label_maker/main.py

+4
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,10 @@ def cli():
9393
config['country'] = op.splitext(op.basename(config.get('geojson')))[0]
9494
config['bounding_box'] = get_bounds(json.load(open(config.get('geojson'), 'r')))
9595

96+
# Convert HTTP auth from list to tuple if it exists
97+
if 'http_auth' in config.keys():
98+
config['http_auth'] = tuple(config['http_auth'])
99+
96100
if cmd == 'download':
97101
download_mbtiles(dest_folder=dest_folder, **config)
98102
elif cmd == 'labels':

label_maker/package.py

+21-17
Original file line numberDiff line numberDiff line change
@@ -9,43 +9,46 @@
99
from label_maker.utils import is_tif
1010

1111

12-
def package_directory(dest_folder, classes, imagery, ml_type, seed=False, split_names=['train', 'test'],
13-
split_vals=[0.8, .2], **kwargs):
12+
def package_directory(dest_folder, classes, imagery, ml_type, seed=False,
13+
split_names=('train', 'test'), split_vals=(0.8, .2),
14+
**kwargs):
1415
"""Generate an .npz file containing arrays for training machine learning algorithms
1516
1617
Parameters
1718
------------
1819
dest_folder: str
1920
Folder to save labels, tiles, and final numpy arrays into
2021
classes: list
21-
A list of classes for machine learning training. Each class is defined as a dict
22-
with two required properties:
22+
A list of classes for machine learning training. Each class is defined
23+
as a dict with two required properties:
2324
- name: class name
2425
- filter: A Mapbox GL Filter.
2526
See the README for more details
2627
imagery: str
2728
Imagery template to download satellite images from.
2829
Ex: http://a.tiles.mapbox.com/v4/mapbox.satellite/{z}/{x}/{y}.jpg?access_token=ACCESS_TOKEN
2930
ml_type: str
30-
Defines the type of machine learning. One of "classification", "object-detection", or "segmentation"
31+
Defines the type of machine learning. One of "classification",
32+
"object-detection", or "segmentation"
3133
seed: int
3234
Random generator seed. Optional, use to make results reproducible.
33-
split_vals: list
34-
Default: [0.8, 0.2]
35-
Percentage of data to put in each catagory listed in split_names.
36-
Must be floats and must sum to one.
37-
split_names: list
38-
Default: ['train', 'test']
35+
split_vals: tuple
36+
Percentage of data to put in each catagory listed in split_names. Must
37+
be floats and must sum to one. Default: (0.8, 0.2)
38+
split_names: tupel
39+
Default: ('train', 'test')
3940
List of names for each subset of the data.
4041
**kwargs: dict
41-
Other properties from CLI config passed as keywords to other utility functions
42+
Other properties from CLI config passed as keywords to other utility
43+
functions.
4244
"""
4345
# if a seed is given, use it
4446
if seed:
4547
np.random.seed(seed)
4648

4749
if len(split_names) != len(split_vals):
48-
raise ValueError('`split_names` and `split_vals` must be the same length. Please update your config.')
50+
raise ValueError('`split_names` and `split_vals` must be the same '
51+
'length. Please update your config.')
4952
if not np.isclose(sum(split_vals), 1):
5053
raise ValueError('`split_vals` must sum to one. Please update your config.')
5154

@@ -105,7 +108,8 @@ def package_directory(dest_folder, classes, imagery, ml_type, seed=False, split_
105108
split_n_samps = [len(x_vals) * val for val in split_vals]
106109

107110
if np.any(split_n_samps == 0):
108-
raise ValueError('split must not generate zero samples per partition, change ratio of values in config file.')
111+
raise ValueError('Split must not generate zero samples per partition. '
112+
'Change ratio of values in config file.')
109113

110114
# Convert into a cumulative sum to get indices
111115
split_inds = np.cumsum(split_n_samps).astype(np.integer)
@@ -117,8 +121,8 @@ def package_directory(dest_folder, classes, imagery, ml_type, seed=False, split_
117121
save_dict = {}
118122

119123
for si, split_name in enumerate(split_names):
120-
save_dict[f'x_{split_name}'] = split_arrs_x[si]
121-
save_dict[f'y_{split_name}'] = split_arrs_y[si]
124+
save_dict['x_{}'.format(split_name)] = split_arrs_x[si]
125+
save_dict['y_{}'.format(split_name)] = split_arrs_y[si]
122126

123127
np.savez(op.join(dest_folder, 'data.npz'), **save_dict)
124-
print('Saving packaged file to {}'.format(op.join(dest_folder, 'data.npz')))
128+
print('Saving packaged file to {}'.format(op.join(dest_folder, 'data.npz')))

label_maker/preview.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,8 @@ def preview(dest_folder, number, classes, imagery, ml_type, imagery_offset=False
6464
if n >= number:
6565
break
6666

67-
tile_img = image_function(tile, imagery, class_dir, imagery_offset)
67+
tile_img = image_function(tile, imagery, class_dir, imagery_offset,
68+
kwargs)
6869

6970
if ml_type == 'object-detection':
7071
img = Image.open(tile_img)

label_maker/utils.py

+6-5
Original file line numberDiff line numberDiff line change
@@ -24,17 +24,18 @@ def class_match(ml_type, label, i):
2424
return np.count_nonzero(label == i)
2525
return None
2626

27-
def download_tile_tms(tile, imagery, folder, *args):
27+
def download_tile_tms(tile, imagery, folder, kwargs):
2828
"""Download a satellite image tile from a tms endpoint"""
2929
o = urlparse(imagery)
3030
_, image_format = op.splitext(o.path)
31-
r = requests.get(url(tile.split('-'), imagery))
31+
r = requests.get(url(tile.split('-'), imagery),
32+
auth=kwargs.get('http_auth'))
3233
tile_img = op.join(folder, '{}{}'.format(tile, image_format))
3334
with open(tile_img, 'wb')as w:
3435
w.write(r.content)
3536
return tile_img
3637

37-
def get_tile_tif(tile, imagery, folder, imagery_offset):
38+
def get_tile_tif(tile, imagery, folder, imagery_offset, kwargs):
3839
"""
3940
Read a GeoTIFF with a window corresponding to a TMS tile
4041
@@ -87,7 +88,7 @@ def get_tile_tif(tile, imagery, folder, imagery_offset):
8788

8889
return tile_img
8990

90-
def get_tile_wms(tile, imagery, folder, imagery_offset):
91+
def get_tile_wms(tile, imagery, folder, imagery_offset, kwargs):
9192
"""
9293
Read a WMS endpoint with query parameters corresponding to a TMS tile
9394
@@ -118,7 +119,7 @@ def get_tile_wms(tile, imagery, folder, imagery_offset):
118119

119120
# request the image with the transformed bounding box and save
120121
wms_url = imagery.replace('{bbox}', ','.join([str(b) for b in bbox]))
121-
r = requests.get(wms_url)
122+
r = requests.get(wms_url, auth=kwargs.get('http_auth'))
122123
tile_img = op.join(folder, '{}.{}'.format(tile, image_format))
123124
with open(tile_img, 'wb') as w:
124125
w.write(r.content)

label_maker/validate.py

+1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727
'zoom': {'type': 'integer', 'required': True},
2828
'classes': {'type': 'list', 'schema': class_schema, 'required': True},
2929
'imagery': {'type': 'string', 'required': True},
30+
'http_auth': {'type': 'list', 'schema': {'type': 'string'}},
3031
'background_ratio': {'type': 'float'},
3132
'ml_type': {'allowed': ['classification', 'object-detection', 'segmentation'], 'required': True},
3233
'seed': {'type': 'integer'},

label_maker/version.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,2 @@
11
"""Library verison"""
2-
__version__ = '0.6.0'
2+
__version__ = '0.6.1'

test/fixtures/146-195-9.jpeg

10 Bytes
Loading

test/unit/test_utils.py

+4-4
Original file line numberDiff line numberDiff line change
@@ -81,7 +81,7 @@ def test_get_tile_tif(self):
8181
if not op.isdir(tiles_dir):
8282
makedirs(tiles_dir)
8383

84-
get_tile_tif(tile, 'test/fixtures/drone.tif', tiles_dir, None)
84+
get_tile_tif(tile, 'test/fixtures/drone.tif', tiles_dir, None, {})
8585
test_tile = Image.open('test/tiles/{}.jpg'.format(tile))
8686
fixture_tile = Image.open('test/fixtures/{}.jpg'.format(tile))
8787
self.assertEqual(test_tile, fixture_tile)
@@ -95,7 +95,7 @@ def test_get_tile_tif_offset(self):
9595
if not op.isdir(tiles_dir):
9696
makedirs(tiles_dir)
9797

98-
get_tile_tif(tile, 'test/fixtures/drone.tif', tiles_dir, [128, 64])
98+
get_tile_tif(tile, 'test/fixtures/drone.tif', tiles_dir, [128, 64], {})
9999
test_tile = Image.open('test/tiles/{}.jpg'.format(tile))
100100
fixture_tile = Image.open('test/fixtures/{}_offset.jpg'.format(tile))
101101
self.assertEqual(test_tile, fixture_tile)
@@ -109,7 +109,7 @@ def test_get_tile_vrt(self):
109109
if not op.isdir(tiles_dir):
110110
makedirs(tiles_dir)
111111

112-
get_tile_tif(tile, 'test/fixtures/drone.vrt', tiles_dir, None)
112+
get_tile_tif(tile, 'test/fixtures/drone.vrt', tiles_dir, None, {})
113113
test_tile = Image.open('test/tiles/{}.jpg'.format(tile))
114114
fixture_tile = Image.open('test/fixtures/{}.jpg'.format(tile))
115115
self.assertEqual(test_tile, fixture_tile)
@@ -125,7 +125,7 @@ def test_get_tile_wms(self):
125125

126126
nasa_url = 'https://gibs.earthdata.nasa.gov/wms/epsg4326/best/wms.cgi?SERVICE=WMS&REQUEST=GetMap&layers=MODIS_Aqua_CorrectedReflectance_TrueColor&version=1.3.0&crs=EPSG:4326&transparent=false&width=256&height=256&bbox={bbox}&format=image/jpeg&time=2019-03-05'
127127

128-
get_tile_wms(tile, nasa_url, tiles_dir, None)
128+
get_tile_wms(tile, nasa_url, tiles_dir, None, {})
129129
test_tile = Image.open('test/tiles/{}.jpeg'.format(tile))
130130
fixture_tile = Image.open('test/fixtures/{}.jpeg'.format(tile))
131131
self.assertEqual(test_tile, fixture_tile)

0 commit comments

Comments
 (0)