2020-05-28 01:16:45 +01:00
import os
import re
2021-04-02 23:21:23 +01:00
import subprocess
2023-05-08 01:10:43 +01:00
import tempfile
import zipfile
import urllib
from bs4 import BeautifulSoup
2021-04-02 23:21:23 +01:00
from pathlib import Path
2023-01-19 18:35:56 +00:00
from django . core . exceptions import MultipleObjectsReturned , ObjectDoesNotExist
2023-05-08 01:10:43 +01:00
from django . http import HttpResponse , HttpResponseNotFound , HttpResponseRedirect , FileResponse
2023-09-30 20:09:18 +01:00
from django . shortcuts import render , redirect
2023-04-30 21:42:03 +01:00
from django . urls import NoReverseMatch , reverse
2013-08-01 16:00:01 +01:00
2020-05-28 01:16:45 +01:00
import troggle . settings as settings
2023-01-30 19:04:36 +00:00
from troggle . core . forms import CaveAndEntranceFormSet , CaveForm , EntranceForm , EntranceLetterForm
2023-12-23 18:37:20 +00:00
from troggle . core . models . caves import Cave , CaveAndEntrance , Entrance , GetCaveLookup , get_cave_leniently
2024-07-12 16:18:05 +01:00
from troggle . core . models . logbooks import QM
2023-12-23 18:37:20 +00:00
from troggle . core . models . wallets import Wallet
2023-01-30 23:04:11 +00:00
from troggle . core . utils import write_and_commit
2024-03-15 01:50:34 +00:00
from troggle . core . utils import current_expo
2021-04-01 21:44:03 +01:00
from troggle . core . views import expo
2023-04-22 01:24:32 +01:00
from troggle . settings import CAVEDESCRIPTIONS , ENTRANCEDESCRIPTIONS
2023-04-22 22:05:12 +01:00
from troggle . parsers . caves import read_cave , read_entrance
2023-04-22 01:24:32 +01:00
2023-05-08 01:10:43 +01:00
from django . template import loader
from django . utils . safestring import mark_safe
2023-01-19 18:35:56 +00:00
2021-05-03 20:35:35 +01:00
from . auth import login_required_if_public
2019-02-26 00:43:05 +00:00
2023-01-30 19:04:36 +00:00
""" Manages the complex procedures to assemble a cave description out of the compnoents
2021-04-13 01:37:42 +01:00
Manages the use of cavern to parse survex files to produce 3 d and pos files
2023-01-30 19:04:36 +00:00
"""
2021-04-13 01:37:42 +01:00
2023-02-02 21:50:40 +00:00
todo = """
- in getCaves ( ) search GCavelookup first , which should raise a MultpleObjectsReturned
exception if no duplicates
- Learn to use Django . select_related ( ) and . prefetch_related ( ) to speed things up
especially on the big report pages
https : / / zerotobyte . com / how - to - use - django - select - related - and - prefetch - related /
2024-04-10 21:23:52 +01:00
- Remove all the URL rewriting which is there because we have not yet edited all the caves to use
our new ( 2023 ) standard addressing of / 16 xx / NNN / NNN . html where * all * caves are assumed to have their
own directory 16 xx / NNN / even if they have no images to put in it .
2023-01-30 19:04:36 +00:00
"""
2024-07-10 20:07:45 +01:00
# def cavepagefwd(request, karea=None, subpath=None):
# """archaic, just send to the caves list page
# """
# return redirect("/caves")
2024-07-12 16:18:05 +01:00
2024-07-10 20:07:45 +01:00
def get_cave_from_slug ( caveslug ) :
2024-07-12 16:18:05 +01:00
""" Needs refactoring
2024-07-10 20:07:45 +01:00
"""
caves = [ ]
print ( f " get_cave_from_slug(): { caveslug } ... " )
areacode = caveslug [ : 4 ] # e.g. 1623
id = caveslug [ 5 : ] # e.g. 161 or 2023-MM-02
thisarea = Cave . objects . filter ( areacode = areacode )
caves_k = thisarea . filter ( kataster_number = id )
if len ( caves_k ) == 1 :
caves . append ( caves_k [ 0 ] )
print ( f " get_cave_from_slug(): { caves_k =} { len ( caves_k ) =} " )
caves_id = thisarea . filter ( unofficial_number = id )
if len ( caves_id ) == 1 :
caves . append ( caves_id [ 0 ] )
print ( f " get_cave_from_slug(): { caves_id =} { len ( caves_id ) =} " )
if len ( caves ) > 1 :
print ( f " get_cave_from_slug(): { caveslug } More than 1 \n { caves } " )
2024-07-12 16:18:05 +01:00
return None
2024-07-10 20:07:45 +01:00
if len ( caves ) < 1 :
print ( f " get_cave_from_slug(): { caveslug } Nowt found.. " )
2024-07-12 16:18:05 +01:00
return None
2024-07-10 20:07:45 +01:00
cave = caves [ 0 ]
print ( f " get_cave_from_slug(): { caveslug } FOUND { cave } " )
2024-07-12 16:18:05 +01:00
return cave
2024-07-10 20:07:45 +01:00
try :
cave_zero = Cave . objects . get ( caveslug__slug = caveslug )
print ( f " Getting cave from ' { caveslug } ' " )
if cave_zero != cave :
print ( f " get_cave_from_slug(): { caveslug } BAD DISCREPANCY { cave_zero =} { cave =} " )
else :
print ( f " get_cave_from_slug(): { caveslug } SUCCESS " )
return cave_zero
except :
return None
def caveslugfwd ( request , slug ) :
""" This is ass backwards. It would be better style to have the slug-identified request be the master, and have
other paths redirect to it , rather than what we have here .
Pending a change where we remove cave . url as a field and have an explicit fixed convention instead .
"""
if not slug :
message = f " Failed to find cave from identifier given: { slug } . "
return render ( request , " errors/generic.html " , { " message " : message } )
2023-12-23 18:37:20 +00:00
2024-07-10 20:07:45 +01:00
Gcavelookup = GetCaveLookup ( )
if slug in Gcavelookup :
cave = Gcavelookup [ slug ]
return redirect ( f " / { cave . url } " )
2023-10-21 14:22:20 +01:00
2021-11-06 22:36:44 +00:00
2021-04-02 23:21:23 +01:00
def getCaves ( cave_id ) :
2023-01-30 19:04:36 +00:00
""" Only gets called if a call to getCave() raises a MultipleObjects exception
2021-04-02 23:21:23 +01:00
TO DO : search GCavelookup first , which should raise a MultpleObjectsReturned exception if there
2023-01-30 19:04:36 +00:00
are duplicates """
2021-04-02 23:21:23 +01:00
try :
caves = Cave . objects . filter ( kataster_number = cave_id )
caveset = set ( caves )
2023-01-30 19:04:36 +00:00
Gcavelookup = GetCaveLookup ( ) # dictionary makes strings to Cave objects
2021-04-02 23:21:23 +01:00
if cave_id in Gcavelookup :
caveset . add ( Gcavelookup [ cave_id ] )
return list ( caveset )
except :
return [ ]
2011-07-11 02:10:22 +01:00
def getCave ( cave_id ) :
2023-10-21 14:22:20 +01:00
""" Returns a cave object when given a cave name or number.
It is used by views including cavehref , ent , wallets and qm .
2023-01-30 19:04:36 +00:00
2021-03-28 23:47:47 +01:00
TO DO : search GCavelookup first , which should raise a MultpleObjectsReturned exception if there
2023-01-30 19:04:36 +00:00
are duplicates """
2011-07-11 02:10:22 +01:00
try :
cave = Cave . objects . get ( kataster_number = cave_id )
2021-03-31 20:18:46 +01:00
return cave
2021-03-28 23:47:47 +01:00
except Cave . MultipleObjectsReturned as ex :
2023-01-30 19:04:36 +00:00
raise MultipleObjectsReturned ( " Duplicate kataster number " ) from ex # propagate this up
2021-03-28 23:47:47 +01:00
2021-03-31 20:18:46 +01:00
except Cave . DoesNotExist as ex :
2023-01-30 19:04:36 +00:00
Gcavelookup = GetCaveLookup ( ) # dictionary makes strings to Cave objects
2021-03-28 23:47:47 +01:00
if cave_id in Gcavelookup :
2023-01-30 19:04:36 +00:00
return Gcavelookup [ cave_id ]
2021-03-31 20:18:46 +01:00
else :
2023-01-30 19:04:36 +00:00
raise ObjectDoesNotExist ( " No cave found with this identifier in any id field " ) from ex # propagate this up
2021-03-28 23:47:47 +01:00
except :
raise ObjectDoesNotExist ( " No cave found with this identifier in any id field " )
2011-07-11 02:10:22 +01:00
2023-01-30 19:04:36 +00:00
2012-08-05 18:26:24 +01:00
def pad5 ( x ) :
2023-01-30 19:04:36 +00:00
return " 0 " * ( 5 - len ( x . group ( 0 ) ) ) + x . group ( 0 )
2012-08-05 18:26:24 +01:00
def padnumber ( x ) :
2024-07-10 20:07:45 +01:00
return re . sub ( r " \ d+ " , pad5 , x )
2023-01-30 19:04:36 +00:00
2012-08-05 18:26:24 +01:00
def numericalcmp ( x , y ) :
2023-01-30 19:04:36 +00:00
return cmp ( padnumber ( x ) , padnumber ( y ) )
2012-08-05 18:26:24 +01:00
2021-04-16 16:01:35 +01:00
def caveKey ( c ) :
2024-04-10 21:23:52 +01:00
""" This function goes into a lexicographic sort function, and the values are strings,
2024-06-29 06:55:14 +01:00
but we want to sort numerically on kataster number before sorting on unofficial number .
2023-01-30 19:04:36 +00:00
"""
2021-04-16 16:01:35 +01:00
if not c . kataster_number :
return " 9999. " + c . unofficial_number
else :
if int ( c . kataster_number ) > = 100 :
return " 99. " + c . kataster_number
if int ( c . kataster_number ) > = 10 :
2023-01-30 19:04:36 +00:00
return " 9. " + c . kataster_number
2021-04-16 16:01:35 +01:00
return c . kataster_number
2012-06-10 14:59:21 +01:00
2023-01-30 19:04:36 +00:00
2020-06-07 16:13:59 +01:00
def getnotablecaves ( ) :
notablecaves = [ ]
for kataster_number in settings . NOTABLECAVESHREFS :
try :
2023-09-13 16:45:59 +01:00
cave = Cave . objects . get ( kataster_number = kataster_number , areacode = " 1623 " )
2020-06-07 16:13:59 +01:00
notablecaves . append ( cave )
except :
2023-09-16 17:53:39 +01:00
print ( " ! Notable Caves: FAILED to get only one cave per kataster_number OR invalid number for: " + kataster_number )
2023-09-13 16:45:59 +01:00
2023-09-16 17:53:39 +01:00
try :
hc = Cave . objects . get ( kataster_number = 359 , areacode = " 1626 " )
notablecaves . append ( hc )
except :
# fails during the tests because this cave has not been loaded for tests, so catch it here.
pass
2023-09-13 16:45:59 +01:00
print ( notablecaves )
2020-06-07 16:13:59 +01:00
return notablecaves
2023-01-30 19:04:36 +00:00
2011-07-11 02:10:22 +01:00
def caveindex ( request ) :
2023-10-15 16:39:00 +01:00
""" Should use Django order-by for lazy sorting, not here. But only after we have a proper slug system in place for Caves
"""
# allcaves = Cave.objects.all()
# for c in allcaves:
# if c.entrances:
# pass
2023-10-06 12:57:41 +01:00
2023-09-10 13:42:36 +01:00
caves1623 = list ( Cave . objects . filter ( areacode = " 1623 " ) )
2023-10-07 22:06:12 +01:00
caves1624 = list ( Cave . objects . filter ( areacode = " 1624 " ) )
2023-09-10 13:42:36 +01:00
caves1626 = list ( Cave . objects . filter ( areacode = " 1626 " ) )
caves1627 = list ( Cave . objects . filter ( areacode = " 1627 " ) )
2020-05-25 01:46:52 +01:00
caves1623 . sort ( key = caveKey )
2023-10-15 16:39:00 +01:00
caves1624 . sort ( key = caveKey )
2020-05-25 01:46:52 +01:00
caves1626 . sort ( key = caveKey )
2023-05-20 20:47:09 +01:00
caves1627 . sort ( key = caveKey )
2023-01-30 19:04:36 +00:00
return render (
request ,
" caveindex.html " ,
2023-10-07 22:06:12 +01:00
{ " caves1623 " : caves1623 ,
" caves1626 " : caves1626 ,
" caves1627 " : caves1627 ,
" caves1624 " : caves1624 ,
" notablecaves " : getnotablecaves ( ) ,
2024-03-15 01:50:34 +00:00
" cavepage " : True , " year " : current_expo ( ) } ,
2023-01-30 19:04:36 +00:00
)
2023-07-03 08:00:03 +01:00
def entranceindex ( request ) :
2023-07-03 08:04:26 +01:00
ents = Entrance . objects . all ( ) . order_by ( " slug " )
2023-07-03 08:00:03 +01:00
return render (
request ,
" entranceindex.html " ,
{ " entrances " : ents } ,
)
2011-07-11 02:10:22 +01:00
2023-01-30 19:04:36 +00:00
def cave3d ( request , cave_id = " " ) :
""" This is used to create a download url in templates/cave.html if anyone wants to download the .3d file
2021-04-02 23:21:23 +01:00
The caller template tries kataster first , then unofficial_number if that kataster number does not exist
but only if Cave . survex_file is non - empty
2023-01-30 19:04:36 +00:00
"""
2021-03-28 23:47:47 +01:00
try :
2023-01-30 19:04:36 +00:00
cave = getCave ( cave_id )
2021-04-02 23:21:23 +01:00
except ObjectDoesNotExist :
2021-04-14 16:28:30 +01:00
return HttpResponseNotFound
2023-01-30 19:04:36 +00:00
except Cave . MultipleObjectsReturned :
2023-09-15 20:41:02 +01:00
# should really produce a better error message. This is a failure of ambiguous aliases probably.
caves = Cave . objects . filter ( url = kpath )
2024-03-15 01:50:34 +00:00
return render ( request , " svxcaveseveral.html " , { " settings " : settings , " caves " : caves , " year " : current_expo ( ) } )
2021-04-02 23:21:23 +01:00
else :
return file3d ( request , cave , cave_id )
2023-01-30 19:04:36 +00:00
2021-04-02 23:21:23 +01:00
def file3d ( request , cave , cave_id ) :
2023-01-30 19:04:36 +00:00
""" Produces a .3d file directly for download.
2021-04-03 20:52:35 +01:00
survex_file should be in valid path format ' caves-1623/264/264.svx ' but it might be mis - entered as simply ' 2012-ns-10.svx '
2023-01-30 19:04:36 +00:00
Also the cave . survex_file may well not match the cave description path :
2021-04-03 20:52:35 +01:00
e . g . it might be to the whole system ' smk-system.svx ' instead of just for the specific cave .
2023-01-30 19:04:36 +00:00
2021-04-03 20:52:35 +01:00
- If the expected .3 d file corresponding to cave . survex_file is present , return it .
- If the cave . survex_file exists , generate the 3 d file , cache it and return it
- Use the cave_id to guess what the 3 d file might be and , if in the cache , return it
2023-11-07 16:37:52 +00:00
There is a problem as the filename is shown of all areacode information , so both 1624 - 161 and 1623 - 161
have a file called 161. svx and return a file called " 161.3d " which may
get incorrectly cached by other software ( i . e your browser )
2023-01-30 19:04:36 +00:00
"""
2021-04-03 20:52:35 +01:00
def runcavern ( survexpath ) :
2023-01-30 19:04:36 +00:00
""" This has not yet been properly updated with respect to putting the .3d file in the same folder as the .svx filse
as done in runcavern3d ( ) in parsers / survex . py
2022-07-20 23:22:07 +01:00
Needs testing .
2023-01-30 19:04:36 +00:00
"""
2021-04-03 20:52:35 +01:00
if not survexpath . is_file ( ) :
2021-04-03 20:54:33 +01:00
pass
2021-04-02 23:21:23 +01:00
try :
2023-01-30 19:04:36 +00:00
completed_process = subprocess . run (
[ settings . CAVERN , " --log " , f " --output= { settings . SURVEX_DATA } " , f " { survexpath } " ]
)
2021-04-02 23:21:23 +01:00
except OSError as ex :
2023-01-30 19:04:36 +00:00
# propagate this to caller.
2021-04-03 20:52:35 +01:00
raise OSError ( completed_process . stdout ) from ex
2023-01-30 19:04:36 +00:00
op3d = ( Path ( settings . SURVEX_DATA ) / Path ( survexpath ) . name ) . with_suffix ( " .3d " )
op3dlog = Path ( op3d . with_suffix ( " .log " ) )
2021-04-03 20:52:35 +01:00
if not op3d . is_file ( ) :
2022-11-23 10:48:39 +00:00
print ( f " - - Regeneration FAILED \n - - from ' { survexpath } ' \n - - to ' { op3d } ' " )
2021-04-03 20:52:35 +01:00
print ( " - - Regeneration stdout: " , completed_process . stdout )
print ( " - - Regeneration cavern log output: " , op3dlog . read_text ( ) )
2023-01-30 19:04:36 +00:00
2023-11-07 16:37:52 +00:00
def return3d ( threedpath , cave ) :
newfilename = cave . slug ( ) + " .3d " # add the "1623-" part of the filename effectively.
2021-04-03 20:52:35 +01:00
if threedpath . is_file ( ) :
2023-01-30 19:04:36 +00:00
response = HttpResponse ( content = open ( threedpath , " rb " ) , content_type = " application/3d " )
2023-11-07 16:37:52 +00:00
response [ " Content-Disposition " ] = f " attachment; filename= { newfilename } "
2021-04-03 20:52:35 +01:00
return response
else :
2022-11-23 10:48:39 +00:00
message = f ' <h1>Path provided does not correspond to any actual 3d file.</h1><p>path: " { threedpath } " '
2023-01-30 19:04:36 +00:00
return HttpResponseNotFound ( message )
2023-11-07 16:37:52 +00:00
survexname = Path ( cave . survex_file ) . name # removes directories ie 1623/161/161.svx -> 161.svx
2021-04-03 20:52:35 +01:00
survexpath = Path ( settings . SURVEX_DATA , cave . survex_file )
2023-11-07 16:37:52 +00:00
survexdir = survexpath . parent
threedname = Path ( survexname ) . with_suffix ( " .3d " ) # removes .svx, replaces with .3d AND DISCARDS PATH arrgh
threedpath = survexpath . parent / threedname
2021-04-03 20:52:35 +01:00
# These if statements need refactoring more cleanly
if cave . survex_file :
2022-08-25 03:31:54 +01:00
if threedpath . is_file ( ) :
2021-04-03 20:52:35 +01:00
if survexpath . is_file ( ) :
if os . path . getmtime ( survexpath ) > os . path . getmtime ( threedpath ) :
runcavern ( survexpath )
2023-11-07 16:37:52 +00:00
return return3d ( threedpath , cave )
2021-04-03 20:52:35 +01:00
else :
if survexpath . is_file ( ) :
runcavern ( survexpath )
2023-11-07 16:37:52 +00:00
return return3d ( threedpath , cave )
2021-04-03 20:52:35 +01:00
# Get here if cave.survex_file was set but did not correspond to a valid svx file
if survexpath . is_file ( ) :
# a file, but invalid format
2023-01-30 19:04:36 +00:00
message = f ' <h1>File is not valid .svx format.</h1><p>Could not generate 3d file from " { survexpath } " '
2021-04-02 23:21:23 +01:00
else :
2021-04-03 20:52:35 +01:00
# we could try to guess that 'caves-1623/' is missing,... nah.
2022-11-23 10:48:39 +00:00
message = f ' <h1>Path provided does not correspond to any actual file.</h1><p>path: " { survexpath } " '
2021-04-03 20:52:35 +01:00
2023-01-30 19:04:36 +00:00
return HttpResponseNotFound ( message )
def rendercave ( request , cave , slug , cave_id = " " ) :
""" Gets the data and files ready and then triggers Django to render the template.
2021-04-03 20:52:35 +01:00
The resulting html contains urls which are dispatched independently , e . g . the ' download ' link
2023-01-30 19:04:36 +00:00
"""
2021-04-03 20:52:35 +01:00
2021-04-07 21:53:17 +01:00
if cave . non_public and settings . PUBLIC_SITE and not request . user . is_authenticated :
2023-01-30 19:04:36 +00:00
return render ( request , " nonpublic.html " , { " instance " : cave , " cavepage " : True , " cave_id " : cave_id } )
2021-04-03 20:52:35 +01:00
else :
2022-03-18 20:00:15 +00:00
# print(f" ! rendercave: slug:'{slug}' survex file:'{cave.survex_file}'")
try :
2023-01-30 19:04:36 +00:00
svx3d = Path ( cave . survex_file ) . stem
2022-03-18 20:00:15 +00:00
svxstem = Path ( settings . SURVEX_DATA ) / Path ( cave . survex_file )
# print(f" ! rendercave: slug:'{slug}' '' ++ '{svxstem}'")
except :
2023-07-05 17:43:57 +01:00
svx3d = " "
svxstem = " "
2022-03-18 20:00:15 +00:00
print ( f " ! rendercave: slug: ' { slug } ' FAIL TO MANAGE survex file: ' { cave . survex_file } ' " )
2021-04-03 20:52:35 +01:00
# NOTE the template itself loads the 3d file using javascript before it loads anything else.
# Django cannot see what this javascript is doing, so we need to ensure that the 3d file exists first.
2024-06-29 06:55:14 +01:00
# So only do this render if a valid .3d file exists. TO BE DONE
2021-11-06 20:59:10 +00:00
# see design docum in troggle/templates/cave.html
# see rendercave() in troggle/core/views/caves.py
2023-01-30 19:04:36 +00:00
templatefile = " cave.html "
2023-12-23 18:37:20 +00:00
wallets = Wallet . objects . filter ( caves = cave )
2023-01-30 19:04:36 +00:00
2021-04-25 04:04:53 +01:00
if not cave_id :
2023-01-30 19:04:36 +00:00
cave_id = slug # cave.unofficial_number
context = {
" cave_editable " : True ,
" settings " : settings ,
" cave " : cave ,
" cavepage " : True ,
" cave_id " : cave_id ,
" svxstem " : str ( svxstem ) ,
" svx3d " : svx3d ,
2024-03-15 01:50:34 +00:00
" wallets " : wallets ,
" year " : current_expo ( )
2023-01-30 19:04:36 +00:00
}
2018-04-17 21:57:02 +01:00
2022-07-20 08:08:23 +01:00
# Do not catch any exceptions here: propagate up to caller
2023-01-30 19:04:36 +00:00
r = render (
request , templatefile , context
) # crashes here with NoReverseMatch if url not set up for 'edit_cave' in urls.py
2022-07-20 08:08:23 +01:00
return r
2023-09-30 20:09:18 +01:00
2023-01-30 19:04:36 +00:00
2023-09-30 20:09:18 +01:00
def cavepage ( request , karea = None , subpath = None ) :
2023-01-30 19:04:36 +00:00
""" Displays a cave description page
2021-04-03 20:52:35 +01:00
accessed by kataster area number specifically
2022-03-18 20:00:15 +00:00
OR
2023-01-30 19:04:36 +00:00
accessed by cave . url specifically set in data , e . g .
2023-05-02 15:42:58 +01:00
" 1623/000/000 " < = cave - data / 1623 - 000. html
2023-01-30 19:04:36 +00:00
" 1623/41/115.htm " < = cave - data / 1623 - 115. html
2023-09-06 22:01:03 +01:00
so we have to query the database to find the URL as we cannot rely on the url actually telling us the cave by inspection .
2023-05-02 15:42:58 +01:00
NOTE that old caves have " .html " ( or " .htm " ) in the URL as they used to be actual files . But since 2006 these URLs
refer to virtual pages generated on the fly by troggle , so the " .html " is confusing and redundant .
2024-04-10 21:23:52 +01:00
But this confused Becka so it was re - instated . Thus creating more confusion for future generations . . .
2023-01-30 19:04:36 +00:00
2023-05-02 15:42:58 +01:00
There are also A LOT OF URLS to e . g . / 1623 / 161 / l / rl89a . htm which are IMAGES and real html files
in cave descriptions . These need to be handled HERE too ( accident of history ) .
2024-08-04 06:48:37 +01:00
2023-01-30 19:04:36 +00:00
"""
2024-04-10 21:23:52 +01:00
if not subpath or subpath == ' / ' :
print ( f " { karea =} { subpath =} " )
return render ( request , " pagenotfound.html " , { " path " : f " { karea } / { subpath } " } , status = 404 )
2023-11-11 23:23:20 +00:00
# lack of validation for karea, 162x
2023-09-30 20:09:18 +01:00
# subpath has an initial /
2022-03-18 20:00:15 +00:00
kpath = karea + subpath
2023-11-11 23:23:20 +00:00
#print(f" ! cavepage:'{kpath}' kataster area:'{karea}' rest of path:'{subpath}'")
caves = Cave . objects . filter ( url = kpath )
if len ( caves ) == 1 :
cave = caves [ 0 ]
return rendercave ( request , cave , cave . slug ( ) )
2024-04-10 21:23:52 +01:00
subpath = subpath . strip ( " // " )
2023-11-11 23:23:20 +00:00
# re do all this using pathlib functions
parts = subpath . strip ( " / " ) . split ( " / " )
if len ( parts ) > 5 :
# recursive loop. break out of it.
subparts = parts [ 0 ] . split ( " . " )
caveid = subparts [ 0 ]
slug = f " { karea } - { caveid } "
2024-07-10 20:07:45 +01:00
if cave := get_cave_from_slug ( slug ) : # walrus operator
2023-11-11 23:23:20 +00:00
return redirect ( f " / { cave . url } " )
2022-07-31 20:29:17 +01:00
else :
2023-11-11 23:23:20 +00:00
return redirect ( f " /caves " )
2024-08-04 06:48:37 +01:00
# epath = karea + subpath # e.g. 1623 /204
# return expo.expopage(request, epath)
2024-04-10 21:23:52 +01:00
# BUGGER the real problem is the the cave descript has embedded in it images like
2023-11-11 23:23:20 +00:00
# src="110/entrance.jpeg and since the cave url is now /1623/110/110.html
# the images try to load from /1623/110/110/entrance.jpeg and of course fail.
# THIS IS A HORRIBLE HACK
2024-08-04 06:48:37 +01:00
if len ( parts ) == 1 :
# simple filename, no folders in path,
# either need to insert caveid OR leave as relative link as we are already "in" /1623/nn/
2023-11-11 23:23:20 +00:00
subparts = parts [ 0 ] . split ( " . " )
2024-08-04 06:48:37 +01:00
caveid = subparts [ 0 ] # e.g. 204.htm
2023-11-11 23:23:20 +00:00
k2path = karea + " / " + caveid + subpath
return redirect ( f " / { k2path } " ) # infinite loop
elif len ( parts ) > 2 :
2024-08-04 06:48:37 +01:00
# e.g. i/204.jpg, but that's ok as we are already "in" /1623/nn/
2023-11-11 23:23:20 +00:00
if parts [ 0 ] == parts [ 1 ] : # double caveid
epath = karea
for i in parts [ 1 : ] :
epath + = " / " + i
#print(f"{subpath=}\n {epath=}")
return expo . expopage ( request , epath )
# if either the first two parts are not /caveid/caveid/
# or the number of parts == 2,
# print(f"2 {subpath=}")
epath = karea + " / " + subpath
return expo . expopage ( request , epath )
2023-01-30 19:04:36 +00:00
2011-07-11 02:10:22 +01:00
@login_required_if_public
2023-01-30 19:04:36 +00:00
def edit_cave ( request , path = " " , slug = None ) :
""" This is the form that edits all the cave data and writes out an XML file in the :expoweb: repo folder
2021-04-02 23:21:23 +01:00
The format for the file being saved is in templates / dataformat / cave . xml
2023-09-11 18:38:14 +01:00
Warning . This uses Django deep magic in the CaveForm processing .
2023-01-30 19:04:36 +00:00
2023-04-22 01:24:32 +01:00
It saves the data into into the database and into the html file , which it then commits to git .
2024-06-28 16:31:23 +01:00
We basically ignore the < path > as the < slug > is of the format 1624 - 114 and contains the area code
2023-01-30 19:04:36 +00:00
"""
2024-07-02 10:23:33 +01:00
print ( f " edit_cave(): { path =} { slug =} " )
2021-04-14 16:28:30 +01:00
message = " "
2024-07-10 20:07:45 +01:00
if slug is None :
cave = Cave ( ) # create a New Cave
2012-08-12 18:10:23 +01:00
else :
2024-07-10 20:07:45 +01:00
print ( f " { slug =} " )
if not ( cave := get_cave_from_slug ( slug ) ) : # walrus operator
return render ( request , " errors/badslug.html " , { " badslug " : f " for cave { caveslug } - from edit_cave() " } )
2012-05-23 09:23:40 +01:00
if request . POST :
form = CaveForm ( request . POST , instance = cave )
2023-07-05 18:22:08 +01:00
#ceFormSet = CaveAndEntranceFormSet(request.POST)
if form . is_valid ( ) : # and ceFormSet.is_valid():
2024-07-10 20:07:45 +01:00
print ( f ' edit_cave(): POST is valid. Editing { cave } ' )
2023-01-30 19:04:36 +00:00
cave = form . save ( commit = False )
2024-07-10 20:07:45 +01:00
# print(cave)
2023-07-05 17:43:57 +01:00
if not cave . filename :
2023-09-11 18:38:14 +01:00
cave . filename = cave . areacode + " - " + cave . number ( ) + " .html "
2023-07-05 17:43:57 +01:00
if not cave . url :
2023-09-11 18:38:14 +01:00
cave . url = cave . areacode + " / " + cave . number ( )
2012-08-12 18:10:23 +01:00
cave . save ( )
2024-07-10 20:07:45 +01:00
form . save_m2m ( ) # this does the many-to-many relationship saving between caves and entrances
2012-08-12 18:10:23 +01:00
if slug is None :
2023-09-25 23:10:50 +01:00
# it is not visible on the form so it always will be None
slug = f " { cave . areacode } - { cave . number ( ) } "
cs = CaveSlug ( cave = cave , slug = slug , primary = True )
2024-07-10 20:07:45 +01:00
print ( f " edit_cave(): New CaveSlug saved { slug } " )
2023-01-30 19:04:36 +00:00
cs . save ( )
2023-07-05 18:22:08 +01:00
#ceinsts = ceFormSet.save(commit=False)
#for ceinst in ceinsts:
# ceinst.cave = cave
# ceinst.save()
2021-12-30 21:13:34 +00:00
try :
2022-08-01 15:04:22 +01:00
cave_file = cave . file_output ( )
2023-04-22 03:26:53 +01:00
write_and_commit ( [ cave_file ] , f " Online edit of cave { cave } " )
2021-12-30 21:13:34 +00:00
# leave other exceptions unhandled so that they bubble up to user interface
except PermissionError :
2023-01-30 19:04:36 +00:00
message = f " CANNOT save this file. \n PERMISSIONS incorrectly set on server for this file { cave . filename } . Ask a nerd to fix this. "
return render ( request , " errors/generic.html " , { " message " : message } )
2022-07-19 17:06:56 +01:00
except subprocess . SubprocessError :
2023-01-30 19:04:36 +00:00
message = f " CANNOT git on server for this file { cave . filename } . Edits may not be committed. \n Ask a nerd to fix this. "
return render ( request , " errors/generic.html " , { " message " : message } )
2024-07-10 20:07:45 +01:00
except :
raise
2023-04-30 21:42:03 +01:00
if cave . entrances ( ) . count ( ) > 0 :
return HttpResponseRedirect ( " / " + cave . url )
else :
return HttpResponseRedirect ( reverse ( " newentrance " , args = [ cave . url_parent ( ) , cave . slug ( ) ] ) )
2023-04-30 19:05:57 +01:00
2012-05-23 09:23:40 +01:00
else :
2023-04-30 19:05:57 +01:00
if slug is not None :
# re-read cave data from file.
2024-07-10 20:07:45 +01:00
print ( f " edit_cave(): { cave =} { cave . filename =} " )
print ( f " edit_cave(): { cave . slug ( ) =} " )
2023-07-05 18:22:08 +01:00
if cave . filename :
2024-06-28 16:31:23 +01:00
try :
read_cave ( cave . filename , cave = cave )
except Exception as e :
2024-07-02 10:23:33 +01:00
print ( f " edit_cave(): EXCEPTION attempting to read_cave( { cave . filename } ) \n { e } " )
2024-06-28 16:31:23 +01:00
raise
2023-04-22 01:24:32 +01:00
2023-04-30 19:05:57 +01:00
form = CaveForm ( instance = cave , initial = { ' cave_slug ' : cave . slug ( ) } )
2023-07-05 18:22:08 +01:00
#ceFormSet = CaveAndEntranceFormSet(queryset=cave.caveandentrance_set.all())
2023-04-30 19:05:57 +01:00
else :
form = CaveForm ( )
2023-07-05 18:22:08 +01:00
#ceFormSet = CaveAndEntranceFormSet(queryset=CaveAndEntrance.objects.none())
2024-07-10 20:07:45 +01:00
print ( f " edit_cave(): returning render() " )
2023-01-30 19:04:36 +00:00
return render (
request ,
" editcave.html " ,
{
" form " : form ,
" cave " : cave ,
2024-03-15 01:50:34 +00:00
" message " : message , " year " : current_expo ( ) ,
2023-07-05 18:22:08 +01:00
#"caveAndEntranceFormSet": ceFormSet,
2023-10-07 00:26:52 +01:00
" path " : path + " / " , # used for saving images if attached
2023-01-30 19:04:36 +00:00
} ,
)
2012-06-10 14:59:21 +01:00
2024-07-10 20:07:45 +01:00
2012-06-10 14:59:21 +01:00
@login_required_if_public
2023-04-22 03:26:53 +01:00
def edit_entrance ( request , path = " " , caveslug = None , entslug = None ) :
2023-01-30 19:04:36 +00:00
""" This is the form that edits the entrance data for a single entrance and writes out
2021-04-14 16:28:30 +01:00
an XML file in the : expoweb : repo folder
2023-03-28 20:30:00 +01:00
2021-04-14 16:28:30 +01:00
The format for the file being saved is in templates / dataformat / entrance . xml
2023-03-28 20:30:00 +01:00
Warning . This uses Django deep magic for multiple forms and the CaveAndEntrance class .
2023-01-30 19:04:36 +00:00
2021-04-14 16:28:30 +01:00
It does save the data into into the database directly , not by parsing the file .
2023-10-11 15:02:11 +01:00
2024-07-10 20:07:45 +01:00
' path ' comes from the urls . py regex but is usually empty ( ! )
So we make the proper path for storing the images ourselves .
2023-10-11 15:02:11 +01:00
GET RID of all this entranceletter stuff . Far too overcomplexified .
We don ' t need it. Just the entrance slug is fine, then check uniqueness.
2023-01-30 19:04:36 +00:00
"""
2023-11-07 00:12:57 +00:00
def check_new_slugname_ok ( slug , letter ) :
""" In Nov.2023 it is possible to create a 2nd entrance and not set an entrance letter,
which leads to a constraint uniqueness crash . FIX THIS .
The letter may be set to an existing letter , OR it may be unset , but there may
be an existing unlettered single entrance . Both of these will crash unless fixed .
"""
slugname = f " { slug } { letter } "
nents = Entrance . objects . filter ( slug = slugname ) . count ( )
2024-06-29 06:55:14 +01:00
print ( f " check_new_slugname_ok() { slugname =} { letter =} => { nents } " )
2023-11-07 00:12:57 +00:00
if nents == 0 :
# looks good, but we need to check the CaveaAndEntrance object too
e = entrance #Entrance.objects.get(slug=slugname) # does not exist yet!
2024-06-29 06:55:14 +01:00
e . save ( )
2023-11-07 00:12:57 +00:00
gcl = GetCaveLookup ( )
c = gcl [ slug ]
nce = CaveAndEntrance . objects . filter ( cave = c , entrance = e ) . count ( )
if nce == 0 :
return slugname , letter
# That entrance already exists, or the CE does, OK.. do recursive call, starting at "b"
if letter == " " :
return check_new_slugname_ok ( slug , " b " )
else :
nextletter = chr ( ord ( letter ) + 1 )
return check_new_slugname_ok ( slug , nextletter )
2023-01-30 19:04:36 +00:00
2024-07-10 20:07:45 +01:00
if not ( cave := get_cave_from_slug ( caveslug ) ) : # walrus operator
2023-04-22 03:26:53 +01:00
return render ( request , " errors/badslug.html " , { " badslug " : f " for cave { caveslug } - from edit_entrance() " } )
2022-08-01 15:04:22 +01:00
2023-04-30 19:05:57 +01:00
if entslug :
try :
entrance = Entrance . objects . get ( slug = entslug )
except :
return render ( request , " errors/badslug.html " , { " badslug " : f " for entrance { entslug } - from edit_entrance() " } )
else :
2023-08-04 21:33:49 +01:00
# a new entrance on a cave
2023-04-30 19:05:57 +01:00
entrance = None
2023-04-22 03:26:53 +01:00
if entslug :
2024-07-10 20:07:45 +01:00
print ( f " Edit Entrance { caveslug =} { entslug =} { path =} number of ents: { cave . entrances ( ) . count ( ) } " )
2023-01-30 19:04:36 +00:00
caveAndEntrance = CaveAndEntrance . objects . get ( entrance = entrance , cave = cave )
2023-08-04 07:08:18 +01:00
entlettereditable = False
2012-08-12 18:10:23 +01:00
else :
2023-11-07 00:12:57 +00:00
caveAndEntrance = CaveAndEntrance ( cave = cave , entrance = Entrance ( ) ) # creates a new Entrance object as well as a new CE object
2022-08-01 15:04:22 +01:00
entlettereditable = True
2023-08-04 07:08:18 +01:00
2023-11-07 00:12:57 +00:00
ce = caveAndEntrance
if ce . entranceletter == " " and cave . entrances ( ) . count ( ) > 0 :
2023-08-04 07:08:18 +01:00
# this should not be blank on a multiple-entrance cave
2023-10-11 15:02:11 +01:00
# but it doesn't trigger the entrance letter form unless entletter has a value
2023-11-07 00:12:57 +00:00
entlettereditable = True # but the user has to remember to actually set it...
2023-08-04 07:08:18 +01:00
print ( f " { entlettereditable =} " )
2023-11-07 00:12:57 +00:00
# if the entletter is not editable, then the entletterform does not appear and so is always invalid.
2024-07-10 20:07:45 +01:00
print ( f " { caveslug =} " )
print ( f " { cave =} " )
imgpath = Path ( path ) / cave . areacode / cave . number ( )
print ( f " Edit Entrance { imgpath =} " )
2023-08-04 07:08:18 +01:00
2012-06-10 14:59:21 +01:00
if request . POST :
2023-08-04 21:33:49 +01:00
print ( f " POST Online edit of entrance: ' { entrance } ' where { cave =} " )
2023-10-11 15:02:11 +01:00
entform = EntranceForm ( request . POST , instance = entrance )
2023-11-07 00:12:57 +00:00
2023-10-11 15:02:11 +01:00
if not entlettereditable :
2023-11-07 00:12:57 +00:00
entranceletter = ce . entranceletter
2023-10-11 15:02:11 +01:00
else :
2023-11-07 00:12:57 +00:00
entletterform = EntranceLetterForm ( request . POST , instance = ce )
2023-10-11 15:02:11 +01:00
if entletterform . is_valid ( ) :
ce = entletterform . save ( commit = False )
entranceletter = entletterform . cleaned_data [ " entranceletter " ]
2023-11-07 00:12:57 +00:00
message = f " - POST valid { caveslug =} { entslug =} { path =} entletterform valid \n { entletterform =} . "
print ( message )
2023-10-11 15:02:11 +01:00
else :
2023-11-07 00:12:57 +00:00
# maybe this doesn't matter? It just means entranceletter unset ?
# probably because 'Cave and entrance with this Cave and Entranceletter already exists.'
message = f " - POST INVALID { caveslug =} { entslug =} { path =} entletterform invalid \n { entletterform . errors =} \n { entletterform =} . "
print ( message )
# if entletterform.errors:
# for field in entletterform:
# for error in field.errors:
# print(f"ERR {field=} {error=}")
# return render(request, "errors/generic.html", {"message": message})
entranceletter = " "
if not entform . is_valid ( ) :
message = f " - POST INVALID { caveslug =} { entslug =} { path =} entform valid: { entform . is_valid ( ) } entletterform valid: { entletterform . is_valid ( ) } "
2023-10-11 15:02:11 +01:00
entrance = entform . save ( commit = False )
2023-11-07 00:12:57 +00:00
print ( message )
return render ( request , " errors/generic.html " , { " message " : message } )
else :
2023-10-11 15:02:11 +01:00
print ( f " - POST { caveslug =} { entslug =} { entranceletter =} { path =} " )
2023-04-22 03:26:53 +01:00
if entslug is None :
2023-11-07 00:12:57 +00:00
# we are creating a new entrance
2023-11-08 00:12:37 +00:00
entrance = entform . save ( commit = False )
# entrance = ce.entrance # the one we created earlier?
2024-06-29 06:55:14 +01:00
try :
if entranceletter :
slugname , letter = check_new_slugname_ok ( cave . slug ( ) , entranceletter )
else :
slugname , letter = check_new_slugname_ok ( cave . slug ( ) , " " )
ce . entranceletter = letter
except Exception as e :
print ( f " - EXCEPTION entranceletter { caveslug =} { entslug =} { entranceletter =} { path =} \n { e } " )
raise
2023-04-22 03:26:53 +01:00
entrance . slug = slugname
2012-08-14 21:51:15 +01:00
entrance . cached_primary_slug = slugname
entrance . filename = slugname + " .html "
2023-07-26 14:54:37 +01:00
else :
2023-11-08 00:12:37 +00:00
# an existing entrance ?
2023-07-26 14:54:37 +01:00
entrance . slug = entslug
entrance . cached_primary_slug = entslug
entrance . filename = entslug + " .html "
2023-08-04 21:33:49 +01:00
try :
entrance . save ( )
2023-11-08 00:12:37 +00:00
print ( f " - post { entrance . slug =} { entrance . tag_station =} { entrance . other_station =} " )
except Exception as e :
2023-11-07 00:12:57 +00:00
# fails with uniqueness constraint failure. Which is on CaveAndEntrance, not just on entrance,
# which is confusing to a user who is just editing an Entrance.
# Can happen when user specifies an existing letter! (or none, when they should set one)
2023-08-04 21:33:49 +01:00
print ( f " SAVE EXCEPTION FAIL { entrance =} " )
2023-11-08 00:12:37 +00:00
print ( f " CAVE { cave } \n { e } " )
2023-08-04 21:33:49 +01:00
for ce in cave . entrances ( ) :
print ( f " CAVE: { ce . cave } - ENT: { ce . entrance } - LETTER: ' { ce . entranceletter } ' " )
raise
ce . entrance = entrance
2023-11-08 00:12:37 +00:00
# try not to invoke this:
2023-11-07 00:12:57 +00:00
# UNIQUE constraint failed: core_caveandentrance.cave_id, core_caveandentrance.entranceletter
2023-08-04 21:33:49 +01:00
ce . save ( )
2023-07-24 23:03:12 +01:00
2022-08-01 15:04:22 +01:00
entrance_file = entrance . file_output ( )
cave_file = cave . file_output ( )
2023-11-08 00:12:37 +00:00
2023-08-03 14:11:46 +01:00
2023-08-04 21:33:49 +01:00
print ( f " - POST WRITE letter: ' { ce } ' { entrance =} " )
2023-11-08 00:12:37 +00:00
try :
write_and_commit ( [ entrance_file , cave_file ] , f " Online edit of entrance { entrance . slug } " )
2023-11-07 00:12:57 +00:00
return HttpResponseRedirect ( " / " + cave . url )
2023-11-08 00:12:37 +00:00
except Exception as e :
2023-11-07 00:12:57 +00:00
efilepath , econtent , eencoding = entrance_file
cfilepath , ccontent , cencoding = cave_file
2023-11-08 00:12:37 +00:00
message = f " - FAIL write_and_commit \n entr: ' { efilepath } ' \n cave: ' { cfilepath } ' \n \n { e } "
2023-11-07 00:12:57 +00:00
print ( message )
return render ( request , " errors/generic.html " , { " message " : message } )
2023-08-02 08:17:48 +01:00
else : # GET the page, not POST, or if either of the forms were invalid when POSTed
2023-08-04 08:29:20 +01:00
entletterform = None
2023-08-04 14:17:52 +01:00
entletter = " "
2023-11-07 00:12:57 +00:00
print ( f " ENTRANCE in db: entranceletter = ' { ce . entranceletter } ' " )
2023-04-30 19:05:57 +01:00
if entrance :
# re-read entrance data from file.
filename = str ( entrance . slug + " .html " )
2023-11-08 00:12:37 +00:00
try :
ent = read_entrance ( filename , ent = entrance )
print ( f " ENTRANCE from file: entranceletter = ' { ce . entranceletter } ' " )
except :
# ent only in db not on file. Interesting, let's run with it using whatever we have in the db
print ( f " ENTRANCE NOT read from file: entranceletter = ' { ce . entranceletter } ' " )
2023-04-22 03:26:53 +01:00
2023-08-04 14:11:35 +01:00
entform = EntranceForm ( instance = entrance )
2023-04-30 19:05:57 +01:00
if entslug is None :
2023-08-04 07:29:25 +01:00
entletterform = EntranceLetterForm ( )
# print(f" Getting entletter from EntranceLetterForm")
2023-04-30 19:05:57 +01:00
else :
2023-11-07 00:12:57 +00:00
entletter = ce . entranceletter
2023-08-04 07:29:25 +01:00
if entletter == " " :
entletterform = EntranceLetterForm ( )
print ( f " Blank value: getting entletter from EntranceLetterForm " )
2023-08-04 08:29:20 +01:00
print ( f " { entletter =} " )
2012-08-14 21:51:15 +01:00
else :
2023-08-04 07:29:25 +01:00
entform = EntranceForm ( )
entletterform = EntranceLetterForm ( )
2022-08-01 15:04:22 +01:00
2023-01-30 19:04:36 +00:00
return render (
request ,
" editentrance.html " ,
2024-03-15 01:50:34 +00:00
{ " year " : current_expo ( ) ,
2023-08-04 14:17:52 +01:00
" entform " : entform ,
2023-01-30 19:04:36 +00:00
" cave " : cave ,
2023-11-18 13:27:08 +00:00
" ent " : entrance ,
2023-01-30 19:04:36 +00:00
" entletter " : entletter ,
2023-08-04 08:29:20 +01:00
" entletterform " : entletterform , # is unset if not being used
2023-01-30 19:04:36 +00:00
" entlettereditable " : entlettereditable ,
2024-07-10 20:07:45 +01:00
" path " : str ( imgpath ) + " / " , # used for saving images if attached
2023-01-30 19:04:36 +00:00
} ,
)
2011-07-11 02:10:22 +01:00
def ent ( request , cave_id , ent_letter ) :
2023-01-30 19:04:36 +00:00
cave = Cave . objects . filter ( kataster_number = cave_id ) [ 0 ]
2023-08-03 14:11:46 +01:00
cave_and_ent = CaveAndEntrance . objects . filter ( cave = cave ) . filter ( entranceletter = ent_letter ) [ 0 ]
2023-01-30 19:04:36 +00:00
return render (
request ,
" entrance.html " ,
2024-03-15 01:50:34 +00:00
{ " year " : current_expo ( ) ,
2023-01-30 19:04:36 +00:00
" cave " : cave ,
" entrance " : cave_and_ent . entrance ,
2023-08-03 14:11:46 +01:00
" letter " : cave_and_ent . entranceletter ,
2023-01-30 19:04:36 +00:00
} ,
)
2023-03-28 15:37:25 +01:00
def cave_debug ( request ) :
2023-03-28 17:08:55 +01:00
ents = Entrance . objects . all ( ) . order_by ( ' id ' )
2023-11-17 19:26:26 +00:00
caves = Cave . objects . all ( ) . order_by ( ' id ' )
2023-03-28 15:37:25 +01:00
return render (
request ,
" cave_debug.html " ,
2024-03-15 01:50:34 +00:00
{ " ents " : ents , " caves " : caves , " year " : current_expo ( ) } ,
2023-03-28 15:37:25 +01:00
)
2023-11-07 00:12:57 +00:00
def caveslist ( request ) :
caves = Cave . objects . all ( )
print ( " CAVESLIST " )
return render (
request ,
" caveslist.html " ,
2024-03-15 01:50:34 +00:00
{ " caves " : caves , " year " : current_expo ( ) } ,
2023-11-07 00:12:57 +00:00
)
2011-07-11 02:10:22 +01:00
def get_entrances ( request , caveslug ) :
2024-07-10 20:07:45 +01:00
if not ( cave := get_cave_from_slug ( caveslug ) ) : # walrus operator
2023-02-02 11:19:46 +00:00
return render ( request , " errors/badslug.html " , { " badslug " : f " { caveslug } - from get_entrances() " } )
2023-01-30 19:04:36 +00:00
return render (
2024-03-15 01:50:34 +00:00
request , " options.html " , { " year " : current_expo ( ) , " items " : [ ( e . entrance . slug ( ) , e . entrance . slug ( ) ) for e in cave . entrances ( ) ] }
2023-01-30 19:04:36 +00:00
)
2011-07-11 02:10:22 +01:00
2023-04-05 23:13:12 +01:00
def caveQMs ( request , slug , open = False ) :
2023-01-30 19:04:36 +00:00
""" Lists all the QMs on a particular cave
2022-07-05 13:38:23 +01:00
relies on the template to find all the QMs for the cave specified in the slug , e . g . ' 1623-161 '
Now working in July 2022
2023-01-30 19:04:36 +00:00
"""
2024-07-16 17:13:07 +01:00
if not ( cave := get_cave_from_slug ( slug ) ) : # walrus operator
2023-02-02 11:19:46 +00:00
return render ( request , " errors/badslug.html " , { " badslug " : f " { slug } - from caveQMs() " } )
2023-01-30 19:04:36 +00:00
2022-03-18 20:00:15 +00:00
if cave . non_public and settings . PUBLIC_SITE and not request . user . is_authenticated :
2023-01-30 19:04:36 +00:00
return render ( request , " nonpublic.html " , { " instance " : cave } )
2023-04-05 23:13:12 +01:00
elif open :
2024-03-15 01:50:34 +00:00
return render ( request , " cave_open_qms.html " , { " cave " : cave , " year " : current_expo ( ) } )
2022-03-18 20:00:15 +00:00
else :
2024-03-15 01:50:34 +00:00
return render ( request , " cave_qms.html " , { " cave " : cave , " year " : current_expo ( ) } )
2023-01-30 19:04:36 +00:00
2022-03-18 20:00:15 +00:00
2023-01-30 19:04:36 +00:00
def qm ( request , cave_id , qm_id , year , grade = None , blockname = None ) :
""" Reports on one specific QM
Fixed and working July 2022 , for both CSV imported QMs
2023-03-17 20:01:52 +00:00
Needs refactoring though ! Uses extremely baroque way of getting the QMs instead of querying for QM objects
directly , presumably as a result of a baroque history .
2023-01-30 19:04:36 +00:00
2023-03-17 20:01:52 +00:00
Many caves have several QMS with the same number , grade , year ( 2018 ) and first 8 chars of the survexblock . This crashes things , so the terminal char of the survexblock name was added
2023-01-30 19:04:36 +00:00
"""
year = int ( year )
if blockname == " " or not blockname :
2022-07-06 13:38:53 +01:00
# CSV import QMs, use old technique
try :
2023-01-30 19:04:36 +00:00
c = getCave ( cave_id )
2023-03-17 20:01:52 +00:00
manyqms = c . get_open_QMs ( ) | c . get_ticked_QMs ( ) # set union operation
2023-03-18 03:03:06 +00:00
qm = manyqms . get ( number = qm_id , expoyear = year , grade = grade )
2023-01-30 19:04:36 +00:00
return render ( request , " qm.html " , { " qm " : qm } )
2022-07-06 13:38:53 +01:00
except QM . DoesNotExist :
2023-01-30 19:04:36 +00:00
# raise
return render (
request ,
" errors/badslug.html " ,
{
" badslug " : f " QM.DoesNotExist blockname is empty string: { cave_id =} { year =} { qm_id =} { grade =} { blockname =} "
} ,
)
2023-03-18 03:03:06 +00:00
except QM . MultipleObjectsReturned :
# raise
qms = manyqms . filter ( number = qm_id , expoyear = year )
return render (
request ,
" errors/badslug.html " ,
{
" badslug " : f " QM.MultipleObjectsReturned { cave_id =} { year =} { qm_id =} { grade =} { blockname =} { qms =} "
} ,
)
2022-07-06 13:38:53 +01:00
else :
try :
2023-01-30 19:04:36 +00:00
qmslug = f " { cave_id } - { year } - { blockname =} { qm_id } { grade } "
print ( f " { qmslug =} " )
c = getCave ( cave_id )
2023-03-17 20:01:52 +00:00
manyqms = c . get_open_QMs ( ) | c . get_ticked_QMs ( ) # set union operation
2023-01-30 19:04:36 +00:00
qmqs = manyqms . filter ( expoyear = year , blockname = blockname , number = qm_id , grade = grade )
if len ( qmqs ) > 1 :
2022-07-06 15:35:08 +01:00
for q in qmqs :
print ( qmqs )
2023-03-17 20:01:52 +00:00
message = f " Multiple QMs with the same cave, year, number, grade AND first-several+terminal chars of the survexblock name. (Could be caused by incomplete databasereset). Fix this in the survex file(s). { cave_id =} { year =} { qm_id =} { blockname =} "
2023-01-30 19:04:36 +00:00
return render ( request , " errors/generic.html " , { " message " : message } )
2022-07-06 13:38:53 +01:00
else :
2023-01-30 19:04:36 +00:00
qm = qmqs . get ( expoyear = year , blockname = blockname , number = qm_id , grade = grade )
2022-07-06 15:35:08 +01:00
if qm :
2023-01-30 19:04:36 +00:00
print (
qm ,
f " { qmslug =} : { cave_id =} { year =} { qm_id =} { blockname =} { qm . expoyear =} { qm . completion_description =} " ,
)
2024-03-15 01:50:34 +00:00
return render ( request , " qm.html " , { " qm " : qm , " year " : current_expo ( ) } )
2022-07-06 15:35:08 +01:00
else :
2023-01-30 19:04:36 +00:00
# raise
return render (
request ,
" errors/badslug.html " ,
{ " badslug " : f " Failed get { cave_id =} { year =} { qm_id =} { grade =} { blockname =} " } ,
)
2022-07-06 13:38:53 +01:00
except MultipleObjectsReturned :
2023-03-17 20:01:52 +00:00
message = f " Multiple QMs with the same cave, year, number, grade AND first-several+terminal chars of the survexblock name. (Could be caused by incomplete databasereset). Fix this in the survex file(s). { cave_id =} { year =} { qm_id =} { blockname =} "
2023-01-30 19:04:36 +00:00
return render ( request , " errors/generic.html " , { " message " : message } )
2022-07-06 13:38:53 +01:00
except QM . DoesNotExist :
2023-01-30 19:04:36 +00:00
# raise
return render (
request ,
" errors/badslug.html " ,
{
" badslug " : f " QM.DoesNotExist blockname is not empty string { cave_id =} { year =} { qm_id =} { grade =} { blockname =} "
} ,
)
2023-05-01 00:01:41 +01:00
def expo_kml ( request ) :
return render (
request ,
" expo.kml " ,
{
" entrances " : Entrance . objects . all ( )
} ,
content_type = " application/vnd.google-earth.kml+xml "
)
2023-05-08 01:10:43 +01:00
def expo_kmz ( request ) :
notablecaves = set ( getnotablecaves ( ) )
#Zip file written to a file, to save this function using too much memory
with tempfile . TemporaryDirectory ( ) as tmpdirname :
zippath = os . path . join ( tmpdirname , ' expo.kmz ' )
with zipfile . ZipFile ( zippath , ' w ' , compression = zipfile . ZIP_DEFLATED ) as myzip :
entrances = [ ]
for e in Entrance . objects . all ( ) :
html = loader . get_template ( " entrance_html.kml " ) . render ( { " entrance " : e } , request )
soup = BeautifulSoup ( html )
for img in soup . find_all ( " img " ) :
#src_orig = img['src']
src = urllib . parse . urljoin ( e . cavelist ( ) [ 0 ] . url . rpartition ( " / " ) [ 0 ] + " / " , img [ ' src ' ] )
img [ ' src ' ] = src
p = os . path . join ( settings . EXPOWEB , src )
#print(e.cavelist()[0].url, e.cavelist()[0].url.rpartition("/")[0] + "/", src_orig, p)
if os . path . isfile ( p ) :
myzip . write ( p , src )
for a in soup . find_all ( " a " ) :
try :
ao = a [ ' href ' ]
aa = urllib . parse . urljoin ( e . cavelist ( ) [ 0 ] . url . rpartition ( " / " ) [ 0 ] + " / " , ao )
a [ ' href ' ] = urllib . parse . urljoin ( " https://expo.survex.com/ " , aa )
print ( e . cavelist ( ) [ 0 ] . url . rpartition ( " / " ) [ 0 ] + " / " , ao , a [ ' href ' ] )
except :
pass
html = mark_safe ( soup . prettify ( " utf-8 " ) . decode ( " utf-8 " ) )
size = { True : " large " , False : " small " } [ bool ( set ( e . cavelist ( ) ) & notablecaves ) ]
entrances . append ( loader . get_template ( " entrance.kml " ) . render ( { " entrance " : e , " html " : html , " size " : size } , request ) )
s = loader . get_template ( " expo.kml " ) . render ( { " entrances " : entrances } , request )
myzip . writestr ( " expo.kml " , s )
for f in os . listdir ( settings . KMZ_ICONS_PATH ) :
p = os . path . join ( settings . KMZ_ICONS_PATH , f )
if os . path . isfile ( p ) :
myzip . write ( p , os . path . join ( " icons " , f ) )
return FileResponse ( open ( zippath , ' rb ' ) , content_type = " application/vnd.google-earth.kmz .kmz " )