mirror of
https://github.com/osm-search/Nominatim.git
synced 2024-12-01 14:31:42 +03:00
Merge pull request #2280 from AntoJvlt/Fix-special-phrases-import-and-tests-cleaning
Fix regex and sanity check for the import of special phrases and tests cleaning.
This commit is contained in:
commit
abdba5fdc7
@ -27,7 +27,7 @@ class SpecialPhrasesImporter():
|
|||||||
self.black_list, self.white_list = self._load_white_and_black_lists()
|
self.black_list, self.white_list = self._load_white_and_black_lists()
|
||||||
#Compile the regex here to increase performances.
|
#Compile the regex here to increase performances.
|
||||||
self.occurence_pattern = re.compile(
|
self.occurence_pattern = re.compile(
|
||||||
r'\| ([^\|]+) \|\| ([^\|]+) \|\| ([^\|]+) \|\| ([^\|]+) \|\| ([\-YN])'
|
r'\| *([^\|]+) *\|\| *([^\|]+) *\|\| *([^\|]+) *\|\| *([^\|]+) *\|\| *([\-YN])'
|
||||||
)
|
)
|
||||||
self.sanity_check_pattern = re.compile(r'^\w+$')
|
self.sanity_check_pattern = re.compile(r'^\w+$')
|
||||||
self.transliterator = Transliterator.createFromRules("special-phrases normalizer",
|
self.transliterator = Transliterator.createFromRules("special-phrases normalizer",
|
||||||
@ -153,8 +153,10 @@ class SpecialPhrasesImporter():
|
|||||||
class_matchs = self.sanity_check_pattern.findall(phrase_class)
|
class_matchs = self.sanity_check_pattern.findall(phrase_class)
|
||||||
|
|
||||||
if len(class_matchs) < 1 or len(type_matchs) < 1:
|
if len(class_matchs) < 1 or len(type_matchs) < 1:
|
||||||
raise UsageError("Bad class/type for language {}: {}={}".format(
|
LOG.warning("Bad class/type for language %s: %s=%s. It will not be imported",
|
||||||
lang, phrase_class, phrase_type))
|
lang, phrase_class, phrase_type)
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
def _process_xml_content(self, xml_content, lang):
|
def _process_xml_content(self, xml_content, lang):
|
||||||
"""
|
"""
|
||||||
@ -205,7 +207,8 @@ class SpecialPhrasesImporter():
|
|||||||
continue
|
continue
|
||||||
|
|
||||||
#sanity check, in case somebody added garbage in the wiki
|
#sanity check, in case somebody added garbage in the wiki
|
||||||
self._check_sanity(lang, phrase_class, phrase_type)
|
if not self._check_sanity(lang, phrase_class, phrase_type):
|
||||||
|
continue
|
||||||
|
|
||||||
class_type_pairs.add((phrase_class, phrase_type))
|
class_type_pairs.add((phrase_class, phrase_type))
|
||||||
|
|
||||||
|
@ -13,66 +13,46 @@ from nominatim.tools.special_phrases import SpecialPhrasesImporter
|
|||||||
TEST_BASE_DIR = Path(__file__) / '..' / '..'
|
TEST_BASE_DIR = Path(__file__) / '..' / '..'
|
||||||
|
|
||||||
def test_fetch_existing_words_phrases_basic(special_phrases_importer, word_table,
|
def test_fetch_existing_words_phrases_basic(special_phrases_importer, word_table,
|
||||||
temp_db_conn):
|
temp_db_cursor):
|
||||||
"""
|
"""
|
||||||
Check for the fetch_existing_words_phrases() method.
|
Check for the fetch_existing_words_phrases() method.
|
||||||
It should return special phrase term added to the word
|
It should return special phrase term added to the word
|
||||||
table.
|
table.
|
||||||
"""
|
"""
|
||||||
with temp_db_conn.cursor() as temp_db_cursor:
|
query ="""
|
||||||
query ="""
|
INSERT INTO word VALUES(99999, 'lookup_token', 'normalized_word',
|
||||||
INSERT INTO word VALUES(99999, 'lookup_token', 'normalized_word',
|
'class', 'type', null, 0, 'near');
|
||||||
'class', 'type', null, 0, 'near');
|
"""
|
||||||
"""
|
temp_db_cursor.execute(query)
|
||||||
temp_db_cursor.execute(query)
|
|
||||||
|
|
||||||
assert not special_phrases_importer.words_phrases_to_delete
|
assert not special_phrases_importer.words_phrases_to_delete
|
||||||
special_phrases_importer._fetch_existing_words_phrases()
|
special_phrases_importer._fetch_existing_words_phrases()
|
||||||
contained_phrase = special_phrases_importer.words_phrases_to_delete.pop()
|
contained_phrase = special_phrases_importer.words_phrases_to_delete.pop()
|
||||||
assert contained_phrase == ('normalized_word', 'class', 'type', 'near')
|
assert contained_phrase == ('normalized_word', 'class', 'type', 'near')
|
||||||
|
|
||||||
def test_fetch_existing_words_phrases_housenumber(special_phrases_importer, word_table,
|
@pytest.mark.parametrize("house_type", ['house', 'postcode'])
|
||||||
temp_db_conn):
|
def test_fetch_existing_words_phrases_special_cases(special_phrases_importer, word_table,
|
||||||
|
house_type, temp_db_cursor):
|
||||||
"""
|
"""
|
||||||
Check for the fetch_existing_words_phrases() method.
|
Check for the fetch_existing_words_phrases() method.
|
||||||
It should return nothing as the term added correspond
|
It should return nothing as the terms added correspond
|
||||||
to a housenumber term.
|
to a housenumber and postcode term.
|
||||||
"""
|
"""
|
||||||
with temp_db_conn.cursor() as temp_db_cursor:
|
query ="""
|
||||||
query ="""
|
INSERT INTO word VALUES(99999, 'lookup_token', 'normalized_word',
|
||||||
INSERT INTO word VALUES(99999, 'lookup_token', 'normalized_word',
|
'place', %s, null, 0, 'near');
|
||||||
'place', 'house', null, 0, 'near');
|
"""
|
||||||
"""
|
temp_db_cursor.execute(query, (house_type,))
|
||||||
temp_db_cursor.execute(query)
|
|
||||||
|
|
||||||
special_phrases_importer._fetch_existing_words_phrases()
|
special_phrases_importer._fetch_existing_words_phrases()
|
||||||
assert not special_phrases_importer.words_phrases_to_delete
|
assert not special_phrases_importer.words_phrases_to_delete
|
||||||
|
|
||||||
def test_fetch_existing_words_phrases_postcode(special_phrases_importer, word_table,
|
def test_fetch_existing_place_classtype_tables(special_phrases_importer, temp_db_cursor):
|
||||||
temp_db_conn):
|
|
||||||
"""
|
|
||||||
Check for the fetch_existing_words_phrases() method.
|
|
||||||
It should return nothing as the term added correspond
|
|
||||||
to a postcode term.
|
|
||||||
"""
|
|
||||||
with temp_db_conn.cursor() as temp_db_cursor:
|
|
||||||
query ="""
|
|
||||||
INSERT INTO word VALUES(99999, 'lookup_token', 'normalized_word',
|
|
||||||
'place', 'postcode', null, 0, 'near');
|
|
||||||
"""
|
|
||||||
temp_db_cursor.execute(query)
|
|
||||||
|
|
||||||
special_phrases_importer._fetch_existing_words_phrases()
|
|
||||||
assert not special_phrases_importer.words_phrases_to_delete
|
|
||||||
|
|
||||||
def test_fetch_existing_place_classtype_tables(special_phrases_importer, temp_db_conn):
|
|
||||||
"""
|
"""
|
||||||
Check for the fetch_existing_place_classtype_tables() method.
|
Check for the fetch_existing_place_classtype_tables() method.
|
||||||
It should return the table just created.
|
It should return the table just created.
|
||||||
"""
|
"""
|
||||||
with temp_db_conn.cursor() as temp_db_cursor:
|
temp_db_cursor.execute('CREATE TABLE place_classtype_testclasstypetable()')
|
||||||
query = 'CREATE TABLE place_classtype_testclasstypetable()'
|
|
||||||
temp_db_cursor.execute(query)
|
|
||||||
|
|
||||||
special_phrases_importer._fetch_existing_place_classtype_tables()
|
special_phrases_importer._fetch_existing_place_classtype_tables()
|
||||||
contained_table = special_phrases_importer.table_phrases_to_delete.pop()
|
contained_table = special_phrases_importer.table_phrases_to_delete.pop()
|
||||||
@ -84,13 +64,11 @@ def test_check_sanity_class(special_phrases_importer):
|
|||||||
If a wrong class or type is given, an UsageError should raise.
|
If a wrong class or type is given, an UsageError should raise.
|
||||||
If a good class and type are given, nothing special happens.
|
If a good class and type are given, nothing special happens.
|
||||||
"""
|
"""
|
||||||
with pytest.raises(UsageError):
|
|
||||||
special_phrases_importer._check_sanity('en', '', 'type')
|
|
||||||
|
|
||||||
with pytest.raises(UsageError):
|
assert not special_phrases_importer._check_sanity('en', '', 'type')
|
||||||
special_phrases_importer._check_sanity('en', 'class', '')
|
assert not special_phrases_importer._check_sanity('en', 'class', '')
|
||||||
|
|
||||||
special_phrases_importer._check_sanity('en', 'class', 'type')
|
assert special_phrases_importer._check_sanity('en', 'class', 'type')
|
||||||
|
|
||||||
def test_load_white_and_black_lists(special_phrases_importer):
|
def test_load_white_and_black_lists(special_phrases_importer):
|
||||||
"""
|
"""
|
||||||
@ -147,7 +125,7 @@ def test_convert_settings_giving_json(special_phrases_importer):
|
|||||||
assert returned == json_file
|
assert returned == json_file
|
||||||
|
|
||||||
def test_process_amenity_with_operator(special_phrases_importer, getorcreate_amenityoperator_funcs,
|
def test_process_amenity_with_operator(special_phrases_importer, getorcreate_amenityoperator_funcs,
|
||||||
temp_db_conn):
|
temp_db_conn, word_table):
|
||||||
"""
|
"""
|
||||||
Test that _process_amenity() execute well the
|
Test that _process_amenity() execute well the
|
||||||
getorcreate_amenityoperator() SQL function and that
|
getorcreate_amenityoperator() SQL function and that
|
||||||
@ -157,13 +135,13 @@ def test_process_amenity_with_operator(special_phrases_importer, getorcreate_ame
|
|||||||
special_phrases_importer._process_amenity('', '', '', '', 'in')
|
special_phrases_importer._process_amenity('', '', '', '', 'in')
|
||||||
|
|
||||||
with temp_db_conn.cursor() as temp_db_cursor:
|
with temp_db_conn.cursor() as temp_db_cursor:
|
||||||
temp_db_cursor.execute("SELECT * FROM temp_with_operator WHERE op='near' OR op='in'")
|
temp_db_cursor.execute("SELECT * FROM word WHERE operator='near' OR operator='in'")
|
||||||
results = temp_db_cursor.fetchall()
|
results = temp_db_cursor.fetchall()
|
||||||
|
|
||||||
assert len(results) == 2
|
assert len(results) == 2
|
||||||
|
|
||||||
def test_process_amenity_without_operator(special_phrases_importer, getorcreate_amenity_funcs,
|
def test_process_amenity_without_operator(special_phrases_importer, getorcreate_amenity_funcs,
|
||||||
temp_db_conn):
|
temp_db_conn, word_table):
|
||||||
"""
|
"""
|
||||||
Test that _process_amenity() execute well the
|
Test that _process_amenity() execute well the
|
||||||
getorcreate_amenity() SQL function.
|
getorcreate_amenity() SQL function.
|
||||||
@ -171,7 +149,7 @@ def test_process_amenity_without_operator(special_phrases_importer, getorcreate_
|
|||||||
special_phrases_importer._process_amenity('', '', '', '', '')
|
special_phrases_importer._process_amenity('', '', '', '', '')
|
||||||
|
|
||||||
with temp_db_conn.cursor() as temp_db_cursor:
|
with temp_db_conn.cursor() as temp_db_cursor:
|
||||||
temp_db_cursor.execute("SELECT * FROM temp_without_operator WHERE op='no_operator'")
|
temp_db_cursor.execute("SELECT * FROM word WHERE operator='no_operator'")
|
||||||
result = temp_db_cursor.fetchone()
|
result = temp_db_cursor.fetchone()
|
||||||
|
|
||||||
assert result
|
assert result
|
||||||
@ -221,8 +199,8 @@ def test_grant_access_to_web_user(temp_db_conn, def_config, special_phrases_impo
|
|||||||
assert check_grant_access(temp_db_conn, def_config.DATABASE_WEBUSER, phrase_class, phrase_type)
|
assert check_grant_access(temp_db_conn, def_config.DATABASE_WEBUSER, phrase_class, phrase_type)
|
||||||
|
|
||||||
def test_create_place_classtype_table_and_indexes(
|
def test_create_place_classtype_table_and_indexes(
|
||||||
temp_db_conn, def_config, placex_table, getorcreate_amenity_funcs,
|
temp_db_conn, def_config, placex_table,
|
||||||
getorcreate_amenityoperator_funcs, special_phrases_importer):
|
special_phrases_importer):
|
||||||
"""
|
"""
|
||||||
Test that _create_place_classtype_table_and_indexes()
|
Test that _create_place_classtype_table_and_indexes()
|
||||||
create the right place_classtype tables and place_id indexes
|
create the right place_classtype tables and place_id indexes
|
||||||
@ -238,7 +216,7 @@ def test_create_place_classtype_table_and_indexes(
|
|||||||
assert check_placeid_and_centroid_indexes(temp_db_conn, pair[0], pair[1])
|
assert check_placeid_and_centroid_indexes(temp_db_conn, pair[0], pair[1])
|
||||||
assert check_grant_access(temp_db_conn, def_config.DATABASE_WEBUSER, pair[0], pair[1])
|
assert check_grant_access(temp_db_conn, def_config.DATABASE_WEBUSER, pair[0], pair[1])
|
||||||
|
|
||||||
def test_process_xml_content(temp_db_conn, def_config, special_phrases_importer,
|
def test_process_xml_content(temp_db_conn, def_config, special_phrases_importer, word_table,
|
||||||
getorcreate_amenity_funcs, getorcreate_amenityoperator_funcs):
|
getorcreate_amenity_funcs, getorcreate_amenityoperator_funcs):
|
||||||
"""
|
"""
|
||||||
Test that _process_xml_content() process the given xml content right
|
Test that _process_xml_content() process the given xml content right
|
||||||
@ -310,17 +288,22 @@ def test_import_from_wiki(monkeypatch, temp_db_conn, def_config, special_phrases
|
|||||||
Check that the main import_from_wiki() method is well executed.
|
Check that the main import_from_wiki() method is well executed.
|
||||||
It should create the place_classtype table, the place_id and centroid indexes,
|
It should create the place_classtype table, the place_id and centroid indexes,
|
||||||
grand access to the web user and executing the SQL functions for amenities.
|
grand access to the web user and executing the SQL functions for amenities.
|
||||||
|
It should also update the database well by deleting or preserving existing entries
|
||||||
|
of the database.
|
||||||
"""
|
"""
|
||||||
mock_fetch_existing_words_phrases = MockParamCapture()
|
#Add some data to the database before execution in order to test
|
||||||
mock_fetch_existing_place_classtype_tables = MockParamCapture()
|
#what is deleted and what is preserved.
|
||||||
mock_remove_non_existent_phrases_from_db = MockParamCapture()
|
with temp_db_conn.cursor() as temp_db_cursor:
|
||||||
|
temp_db_cursor.execute("""
|
||||||
|
INSERT INTO word VALUES(99999, ' animal shelter', 'animal shelter',
|
||||||
|
'amenity', 'animal_shelter', null, 0, null);
|
||||||
|
|
||||||
|
INSERT INTO word VALUES(99999, ' wrong_lookup_token', 'wrong_normalized_word',
|
||||||
|
'wrong_class', 'wrong_type', null, 0, 'near');
|
||||||
|
|
||||||
|
CREATE TABLE place_classtype_amenity_animal_shelter();
|
||||||
|
CREATE TABLE place_classtype_wrongclass_wrongtype();""")
|
||||||
|
|
||||||
monkeypatch.setattr('nominatim.tools.special_phrases.SpecialPhrasesImporter._fetch_existing_words_phrases',
|
|
||||||
mock_fetch_existing_words_phrases)
|
|
||||||
monkeypatch.setattr('nominatim.tools.special_phrases.SpecialPhrasesImporter._fetch_existing_place_classtype_tables',
|
|
||||||
mock_fetch_existing_place_classtype_tables)
|
|
||||||
monkeypatch.setattr('nominatim.tools.special_phrases.SpecialPhrasesImporter._remove_non_existent_phrases_from_db',
|
|
||||||
mock_remove_non_existent_phrases_from_db)
|
|
||||||
monkeypatch.setattr('nominatim.tools.special_phrases.SpecialPhrasesImporter._get_wiki_content', mock_get_wiki_content)
|
monkeypatch.setattr('nominatim.tools.special_phrases.SpecialPhrasesImporter._get_wiki_content', mock_get_wiki_content)
|
||||||
special_phrases_importer.import_from_wiki(['en'])
|
special_phrases_importer.import_from_wiki(['en'])
|
||||||
|
|
||||||
@ -332,9 +315,45 @@ def test_import_from_wiki(monkeypatch, temp_db_conn, def_config, special_phrases
|
|||||||
assert check_grant_access(temp_db_conn, def_config.DATABASE_WEBUSER, class_test, type_test)
|
assert check_grant_access(temp_db_conn, def_config.DATABASE_WEBUSER, class_test, type_test)
|
||||||
assert check_amenities_with_op(temp_db_conn)
|
assert check_amenities_with_op(temp_db_conn)
|
||||||
assert check_amenities_without_op(temp_db_conn)
|
assert check_amenities_without_op(temp_db_conn)
|
||||||
assert mock_fetch_existing_words_phrases.called == 1
|
assert check_table_exist(temp_db_conn, 'amenity', 'animal_shelter')
|
||||||
assert mock_fetch_existing_place_classtype_tables.called == 1
|
assert not check_table_exist(temp_db_conn, 'wrong_class', 'wrong_type')
|
||||||
assert mock_remove_non_existent_phrases_from_db.called == 1
|
|
||||||
|
#Format (query, should_return_something_bool) use to easily execute all asserts
|
||||||
|
queries_tests = set()
|
||||||
|
|
||||||
|
#Used to check that the correct phrase already in the word table before is still there.
|
||||||
|
query_correct_word = "SELECT * FROM word WHERE word = 'animal shelter'"
|
||||||
|
queries_tests.add((query_correct_word, True))
|
||||||
|
|
||||||
|
#Used to check if wrong phrase was deleted from the word table of the database.
|
||||||
|
query_wrong_word = "SELECT word FROM word WHERE word = 'wrong_normalized_word'"
|
||||||
|
queries_tests.add((query_wrong_word, False))
|
||||||
|
|
||||||
|
#Used to check that correct place_classtype table already in the datase before is still there.
|
||||||
|
query_existing_table = """
|
||||||
|
SELECT table_name
|
||||||
|
FROM information_schema.tables
|
||||||
|
WHERE table_schema='public'
|
||||||
|
AND table_name = 'place_classtype_amenity_animal_shelter';
|
||||||
|
"""
|
||||||
|
queries_tests.add((query_existing_table, True))
|
||||||
|
|
||||||
|
#Used to check that wrong place_classtype table was deleted from the database.
|
||||||
|
query_wrong_table = """
|
||||||
|
SELECT table_name
|
||||||
|
FROM information_schema.tables
|
||||||
|
WHERE table_schema='public'
|
||||||
|
AND table_name = 'place_classtype_wrongclass_wrongtype';
|
||||||
|
"""
|
||||||
|
queries_tests.add((query_wrong_table, False))
|
||||||
|
|
||||||
|
with temp_db_conn.cursor() as temp_db_cursor:
|
||||||
|
for query in queries_tests:
|
||||||
|
temp_db_cursor.execute(query[0])
|
||||||
|
if (query[1] == True):
|
||||||
|
assert temp_db_cursor.fetchone()
|
||||||
|
else:
|
||||||
|
assert not temp_db_cursor.fetchone()
|
||||||
|
|
||||||
def mock_get_wiki_content(lang):
|
def mock_get_wiki_content(lang):
|
||||||
"""
|
"""
|
||||||
@ -400,7 +419,7 @@ def check_amenities_with_op(temp_db_conn):
|
|||||||
contains more than one value (so that the SQL function was call more than one time).
|
contains more than one value (so that the SQL function was call more than one time).
|
||||||
"""
|
"""
|
||||||
with temp_db_conn.cursor() as temp_db_cursor:
|
with temp_db_conn.cursor() as temp_db_cursor:
|
||||||
temp_db_cursor.execute("SELECT * FROM temp_with_operator")
|
temp_db_cursor.execute("SELECT * FROM word WHERE operator != 'no_operator'")
|
||||||
return len(temp_db_cursor.fetchall()) > 1
|
return len(temp_db_cursor.fetchall()) > 1
|
||||||
|
|
||||||
def check_amenities_without_op(temp_db_conn):
|
def check_amenities_without_op(temp_db_conn):
|
||||||
@ -409,7 +428,7 @@ def check_amenities_without_op(temp_db_conn):
|
|||||||
contains more than one value (so that the SQL function was call more than one time).
|
contains more than one value (so that the SQL function was call more than one time).
|
||||||
"""
|
"""
|
||||||
with temp_db_conn.cursor() as temp_db_cursor:
|
with temp_db_conn.cursor() as temp_db_cursor:
|
||||||
temp_db_cursor.execute("SELECT * FROM temp_without_operator")
|
temp_db_cursor.execute("SELECT * FROM word WHERE operator = 'no_operator'")
|
||||||
return len(temp_db_cursor.fetchall()) > 1
|
return len(temp_db_cursor.fetchall()) > 1
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
@ -458,13 +477,12 @@ def make_strandard_name_func(temp_db_cursor):
|
|||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def getorcreate_amenity_funcs(temp_db_cursor, make_strandard_name_func):
|
def getorcreate_amenity_funcs(temp_db_cursor, make_strandard_name_func):
|
||||||
temp_db_cursor.execute("""
|
temp_db_cursor.execute("""
|
||||||
CREATE TABLE temp_without_operator(op TEXT);
|
|
||||||
|
|
||||||
CREATE OR REPLACE FUNCTION getorcreate_amenity(lookup_word TEXT, normalized_word TEXT,
|
CREATE OR REPLACE FUNCTION getorcreate_amenity(lookup_word TEXT, normalized_word TEXT,
|
||||||
lookup_class text, lookup_type text)
|
lookup_class text, lookup_type text)
|
||||||
RETURNS void as $$
|
RETURNS void as $$
|
||||||
BEGIN
|
BEGIN
|
||||||
INSERT INTO temp_without_operator VALUES('no_operator');
|
INSERT INTO word VALUES(null, lookup_word, normalized_word,
|
||||||
|
lookup_class, lookup_type, null, 0, 'no_operator');
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE plpgsql""")
|
$$ LANGUAGE plpgsql""")
|
||||||
|
|
||||||
@ -477,6 +495,7 @@ def getorcreate_amenityoperator_funcs(temp_db_cursor, make_strandard_name_func):
|
|||||||
lookup_class text, lookup_type text, op text)
|
lookup_class text, lookup_type text, op text)
|
||||||
RETURNS void as $$
|
RETURNS void as $$
|
||||||
BEGIN
|
BEGIN
|
||||||
INSERT INTO temp_with_operator VALUES(op);
|
INSERT INTO word VALUES(null, lookup_word, normalized_word,
|
||||||
|
lookup_class, lookup_type, null, 0, op);
|
||||||
END;
|
END;
|
||||||
$$ LANGUAGE plpgsql""")
|
$$ LANGUAGE plpgsql""")
|
@ -3,7 +3,7 @@
|
|||||||
<sitename>OpenStreetMap Wiki</sitename>
|
<sitename>OpenStreetMap Wiki</sitename>
|
||||||
<dbname>wiki</dbname>
|
<dbname>wiki</dbname>
|
||||||
<base>https://wiki.openstreetmap.org/wiki/Main_Page</base>
|
<base>https://wiki.openstreetmap.org/wiki/Main_Page</base>
|
||||||
<generator>MediaWiki 1.35.1</generator>
|
<generator>MediaWiki 1.35.2</generator>
|
||||||
<case>first-letter</case>
|
<case>first-letter</case>
|
||||||
<namespaces>
|
<namespaces>
|
||||||
<namespace key="-2" case="first-letter">Media</namespace>
|
<namespace key="-2" case="first-letter">Media</namespace>
|
||||||
@ -70,7 +70,7 @@
|
|||||||
<model>wikitext</model>
|
<model>wikitext</model>
|
||||||
<format>text/x-wiki</format>
|
<format>text/x-wiki</format>
|
||||||
<text bytes="158218" sha1="cst5x7tt58izti1pxzgljf27tx8qjcj" xml:space="preserve">
|
<text bytes="158218" sha1="cst5x7tt58izti1pxzgljf27tx8qjcj" xml:space="preserve">
|
||||||
== en == {| class="wikitable sortable" |- ! Word / Phrase !! Key !! Value !! Operator !! Plural |- | Zip Line || aerialway || zip_line || - || N |- | Zip Lines || aerialway || zip_line || - || Y |- | Zip Line in || aerialway || zip_line || in || N |- | Zip Lines in || aerialway || zip_line || in || Y |- | Zip Line near || aerialway || zip_line || near || N |- | Zip Lines near || aerialway || zip_line || near || Y |- | Zip Wire || aerialway || zip_line || - || N |- | Zip Wires || aerialway || zip_line || - || Y |- | Zip Wire in || aerialway || zip_line || in || N |- | Zip Wires in || aerialway || zip_line || in || Y |- | Zip Wire near || aerialway || zip_line || near || N |} [[Category:Word list]]
|
== en == {| class="wikitable sortable" |- ! Word / Phrase !! Key !! Value !! Operator !! Plural |- | Zip Line || aerialway || zip_line || - || N |- | Zip Lines || aerialway || zip_line || - || Y |- | Zip Line in || aerialway || zip_line || in || N |- | Zip Lines in || aerialway || zip_line || in || Y |- | Zip Line near || aerialway || zip_line || near || N |- | Animal shelter || amenity || animal_shelter || - || N |- | Animal shelters || amenity || animal_shelter || - || Y |- | Animal shelter in || amenity || animal_shelter || in || N |- | Animal shelters in || amenity || animal_shelter || in || Y |- | Animal shelter near || amenity || animal_shelter || near|| N |- | Animal shelters near || amenity || animal_shelter || near|| Y |- | Drinking Water near || amenity || drinking_water || near || N |- | Water || amenity || drinking_water || - || N |- | Water in || amenity || drinking_water || in || N |- | Water near || amenity || drinking_water || near || N |- | Embassy || amenity || embassy || - || N |- | Embassys || amenity || embassy || - || Y |- | Embassies || amenity || embassy || - || Y |- |Coworkings near |amenity |coworking_space |near |Y |} [[Category:Word list]]
|
||||||
</text>
|
</text>
|
||||||
<sha1>cst5x7tt58izti1pxzgljf27tx8qjcj</sha1>
|
<sha1>cst5x7tt58izti1pxzgljf27tx8qjcj</sha1>
|
||||||
</revision>
|
</revision>
|
||||||
|
Loading…
Reference in New Issue
Block a user