[0001]
[0002]
[0003]
[0004]
[0005]
[0006]
[0007]
[0008]
[0009]
[0010]
[0011]
[0012]
[0013]
[0014]
[0015]
[0016]
[0017]
[0018]
[0019]
[0020]
[0021]
[0022]
[0023]
[0024]
[0025]
[0026]
[0027]
[0028]
[0029]
[0030]
[0031]
[0032]
[0033]
[0034]
[0035]
[0036]
[0037]
[0038]
[0039]
[0040]
[0041]
[0042]
[0043]
[0044]
[0045]
[0046]
[0047]
[0048]
[0049]
[0050]
[0051]
[0052]
[0053]
[0054]
[0055]
[0056]
[0057]
[0058]
[0059]
[0060]
[0061]
[0062]
[0063]
[0064]
[0065]
[0066]
[0067]
[0068]
[0069]
[0070]
[0071]
[0072]
[0073]
[0074]
[0075]
[0076]
[0077]
[0078]
[0079]
[0080]
[0081]
[0082]
[0083]
[0084]
[0085]
[0086]
[0087]
[0088]
[0089]
[0090]
[0091]
[0092]
[0093]
[0094]
[0095]
[0096]
[0097]
[0098]
[0099]
[0100]
[0101]
[0102]
[0103]
[0104]
[0105]
[0106]
[0107]
[0108]
[0109]
[0110]
[0111]
[0112]
[0113]
[0114]
[0115]
[0116]
[0117]
[0118]
[0119]
[0120]
[0121]
[0122]
[0123]
[0124]
[0125]
[0126]
[0127]
[0128]
[0129]
[0130]
[0131]
[0132]
[0133]
[0134]
[0135]
[0136]
[0137]
[0138]
[0139]
[0140]
[0141]
[0142]
[0143]
[0144]
[0145]
[0146]
[0147]
[0148]
[0149]
[0150]
[0151]
[0152]
[0153]
[0154]
[0155]
[0156]
[0157]
[0158]
[0159]
[0160]
[0161]
[0162]
[0163]
[0164]
[0165]
[0166]
[0167]
[0168]
[0169]
[0170]
[0171]
[0172]
[0173]
[0174]
[0175]
[0176]
[0177]
[0178]
[0179]
[0180]
[0181]
[0182]
[0183]
[0184]
[0185]
[0186]
[0187]
[0188]
[0189]
[0190]
[0191]
[0192]
[0193]
[0194]
[0195]
[0196]
[0197]
[0198]
[0199]
[0200]
[0201]
[0202]
[0203]
[0204]
[0205]
[0206]
[0207]
[0208]
[0209]
[0210]
[0211]
[0212]
[0213]
[0214]
[0215]
[0216]
[0217]
[0218]
[0219]
[0220]
[0221]
[0222]
[0223]
[0224]
[0225]
[0226]
[0227]
[0228]
[0229]
[0230]
[0231]
[0232]
[0233]
[0234]
[0235]
[0236]
[0237]
[0238]
[0239]
[0240]
[0241]
[0242]
[0243]
[0244]
[0245]
[0246]
[0247]
[0248]
[0249]
[0250]
[0251]
[0252]
[0253]
[0254]
[0255]
[0256]
[0257]
[0258]
[0259]
[0260]
[0261]
[0262]
[0263]
[0264]
[0265]
[0266]
[0267]
[0268]
[0269]
[0270]
[0271]
[0272]
[0273]
[0274]
[0275]
[0276]
[0277]
[0278]
[0279]
[0280]
[0281]
[0282]
[0283]
[0284]
[0285]
[0286]
[0287]
[0288]
[0289]
[0290]
[0291]
[0292]
[0293]
[0294]
[0295]
[0296]
[0297]
[0298]
[0299]
[0300]
[0301]
[0302]
[0303]
[0304]
[0305]
[0306]
[0307]
[0308]
[0309]
[0310]
[0311]
[0312]
[0313]
[0314]
[0315]
[0316]
[0317]
[0318]
[0319]
[0320]
[0321]
[0322]
[0323]
[0324]
[0325]
[0326]
[0327]
[0328]
[0329]
[0330]
[0331]
[0332]
[0333]
[0334]
[0335]
[0336]
[0337]
[0338]
[0339]
[0340]
[0341]
[0342]
[0343]
[0344]
[0345]
[0346]
[0347]
[0348]
[0349]
[0350]
[0351]
[0352]
[0353]
[0354]
[0355]
[0356]
[0357]
[0358]
[0359]
[0360]
[0361]
[0362]
[0363]
[0364]
[0365]
[0366]
[0367]
[0368]
[0369]
[0370]
[0371]
[0372]
[0373]
[0374]
[0375]
[0376]
[0377]
[0378]
[0379]
[0380]
[0381]
[0382]
[0383]
[0384]
[0385]
[0386]
[0387]
[0388]
[0389]
[0390]
[0391]
[0392]
[0393]
[0394]
[0395]
[0396]
[0397]
[0398]
[0399]
[0400]
[0401]
[0402]
[0403]
[0404]
[0405]
[0406]
[0407]
[0408]
[0409]
[0410]
[0411]
[0412]
[0413]
[0414]
[0415]
[0416]
[0417]
[0418]
[0419]
[0420]
[0421]
[0422]
[0423]
[0424]
[0425]
[0426]
[0427]
[0428]
[0429]
[0430]
[0431]
[0432]
[0433]
[0434]
[0435]
[0436]
[0437]
[0438]
[0439]
[0440]
[0441]
[0442]
[0443]
[0444]
[0445]
[0446]
[0447]
[0448]
[0449]
[0450]
[0451]
[0452]
[0453]
[0454]
[0455]
[0456]
[0457]
[0458]
[0459]
[0460]
[0461]
[0462]
[0463]
[0464]
[0465]
[0466]
[0467]
[0468]
[0469]
[0470]
[0471]
[0472]
[0473]
[0474]
[0475]
[0476]
[0477]
[0478]
[0479]
[0480]
[0481]
[0482]
[0483]
[0484]
[0485]
[0486]
[0487]
[0488]
[0489]
[0490]
[0491]
[0492]
[0493]
[0494]
[0495]
[0496]
[0497]
[0498]
[0499]
[0500]
[0501]
[0502]
[0503]
[0504]
[0505]
[0506]
[0507]
[0508]
[0509]
[0510]
[0511]
[0512]
[0513]
[0514]
[0515]
[0516]
[0517]
[0518]
[0519]
[0520]
[0521]
[0522]
[0523]
[0524]
[0525]
[0526]
[0527]
[0528]
[0529]
[0530]
[0531]
[0532]
[0533]
[0534]
[0535]
[0536]
[0537]
[0538]
[0539]
[0540]
[0541]
[0542]
[0543]
[0544]
[0545]
[0546]
[0547]
[0548]
[0549]
[0550]
[0551]
[0552]
[0553]
[0554]
[0555]
[0556]
[0557]
[0558]
[0559]
[0560]
[0561]
[0562]
[0563]
[0564]
[0565]
[0566]
[0567]
[0568]
[0569]
[0570]
[0571]
[0572]
[0573]
[0574]
[0575]
[0576]
[0577]
[0578]
[0579]
[0580]
[0581]
[0582]
[0583]
[0584]
[0585]
[0586]
[0587]
[0588]
[0589]
[0590]
[0591]
[0592]
[0593]
[0594]
[0595]
[0596]
[0597]
[0598]
[0599]
[0600]
[0601]
[0602]
[0603]
[0604]
[0605]
[0606]
[0607]
[0608]
[0609]
[0610]
[0611]
[0612]
[0613]
[0614]
[0615]
[0616]
[0617]
[0618]
[0619]
[0620]
[0621]
[0622]
[0623]
[0624]
[0625]
[0626]
[0627]
[0628]
[0629]
[0630]
[0631]
[0632]
[0633]
[0634]
[0635]
[0636]
[0637]
[0638]
[0639]
[0640]
[0641]
[0642]
[0643]
[0644]
[0645]
[0646]
[0647]
[0648]
[0649]
[0650]
[0651]
[0652]
[0653]
[0654]
[0655]
[0656]
[0657]
[0658]
[0659]
[0660]
[0661]
[0662]
[0663]
[0664]
[0665]
[0666]
[0667]
[0668]
[0669]
[0670]
[0671]
[0672]
[0673]
[0674]
[0675]
[0676]
[0677]
[0678]
[0679]
[0680]
[0681]
[0682]
[0683]
[0684]
[0685]
[0686]
[0687]
[0688]
[0689]
[0690]
[0691]
[0692]
[0693]
[0694]
[0695]
[0696]
[0697]
[0698]
[0699]
[0700]
[0701]
[0702]
[0703]
[0704]
[0705]
[0706]
[0707]
[0708]
[0709]
[0710]
[0711]
[0712]
[0713]
[0714]
[0715]
[0716]
[0717]
[0718]
[0719]
[0720]
[0721]
[0722]
[0723]
[0724]
[0725]
[0726]
[0727]
[0728]
[0729]
[0730]
[0731]
[0732]
[0733]
[0734]
[0735]
[0736]
[0737]
[0738]
[0739]
[0740]
[0741]
[0742]
[0743]
[0744]
[0745]
[0746]
[0747]
[0748]
[0749]
[0750]
[0751]
[0752]
[0753]
[0754]
[0755]
[0756]
[0757]
[0758]
[0759]
[0760]
[0761]
[0762]
[0763]
[0764]
[0765]
[0766]
[0767]
[0768]
[0769]
[0770]
[0771]
[0772]
[0773]
[0774]
[0775]
[0776]
[0777]
[0778]
[0779]
[0780]
[0781]
[0782]
[0783]
[0784]
[0785]
[0786]
[0787]
[0788]
[0789]
[0790]
[0791]
[0792]
[0793]
[0794]
[0795]
[0796]
[0797]
[0798]
[0799]
[0800]
[0801]
[0802]
[0803]
[0804]
[0805]
[0806]
[0807]
[0808]
[0809]
[0810]
[0811]
[0812]
[0813]
[0814]
[0815]
[0816]
[0817]
[0818]
[0819]
[0820]
[0821]
[0822]
[0823]
[0824]
[0825]
[0826]
[0827]
[0828]
[0829]
[0830]
[0831]
[0832]
[0833]
[0834]
[0835]
[0836]
[0837]
[0838]
[0839]
[0840]
[0841]
[0842]
[0843]
[0844]
[0845]
[0846]
[0847]
[0848]
[0849]
[0850]
[0851]
[0852]
[0853]
[0854]
[0855]
[0856]
[0857]
[0858]
[0859]
[0860]
[0861]
[0862]
[0863]
[0864]
[0865]
[0866]
[0867]
[0868]
[0869]
[0870]
[0871]
[0872]
[0873]
[0874]
[0875]
[0876]
[0877]
[0878]
[0879]
[0880]
[0881]
[0882]
[0883]
[0884]
[0885]
[0886]
[0887]
[0888]
[0889]
[0890]
[0891]
[0892]
[0893]
[0894]
[0895]
[0896]
[0897]
[0898]
[0899]
[0900]
[0901]
[0902]
[0903]
[0904]
[0905]
[0906]
[0907]
[0908]
[0909]
[0910]
[0911]
[0912]
[0913]
[0914]
[0915]
[0916]
[0917]
[0918]
[0919]
[0920]
[0921]
[0922]
[0923]
[0924]
[0925]
[0926]
[0927]
[0928]
[0929]
[0930]
[0931]
[0932]
[0933]
[0934]
[0935]
[0936]
[0937]
[0938]
[0939]
[0940]
[0941]
[0942]
[0943]
[0944]
[0945]
[0946]
[0947]
[0948]
[0949]
[0950]
[0951]
[0952]
[0953]
[0954]
[0955]
[0956]
[0957]
[0958]
[0959]
[0960]
[0961]
[0962]
[0963]
[0964]
[0965]
[0966]
[0967]
[0968]
[0969]
[0970]
[0971]
[0972]
[0973]
[0974]
[0975]
[0976]
[0977]
[0978]
[0979]
[0980]
[0981]
[0982]
[0983]
[0984]
[0985]
[0986]
[0987]
[0988]
[0989]
[0990]
[0991]
[0992]
[0993]
[0994]
[0995]
[0996]
[0997]
[0998]
[0999]
[1000]
[1001]
[1002]
[1003]
[1004]
[1005]
[1006]
[1007]
[1008]
[1009]
[1010]
[1011]
[1012]
[1013]
[1014]
[1015]
[1016]
[1017]
[1018]
[1019]
[1020]
[1021]
[1022]
[1023]
[1024]
[1025]
[1026]
[1027]
[1028]
[1029]
[1030]
[1031]
[1032]
[1033]
[1034]
[1035]
[1036]
[1037]
[1038]
[1039]
[1040]
[1041]
[1042]
[1043]
[1044]
[1045]
[1046]
[1047]
[1048]
[1049]
[1050]
[1051]
[1052]
[1053]
[1054]
[1055]
[1056]
[1057]
[1058]
[1059]
[1060]
[1061]
[1062]
[1063]
[1064]
[1065]
[1066]
[1067]
[1068]
[1069]
[1070]
[1071]
[1072]
[1073]
[1074]
[1075]
[1076]
[1077]
[1078]
[1079]
[1080]
[1081]
[1082]
[1083]
[1084]
[1085]
[1086]
[1087]
[1088]
[1089]
[1090]
[1091]
[1092]
[1093]
[1094]
[1095]
[1096]
[1097]
[1098]
[1099]
[1100]
[1101]
[1102]
[1103]
[1104]
[1105]
[1106]
[1107]
[1108]
[1109]
[1110]
[1111]
[1112]
[1113]
[1114]
[1115]
[1116]
[1117]
[1118]
[1119]
[1120]
[1121]
[1122]
[1123]
[1124]
[1125]
[1126]
[1127]
[1128]
[1129]
[1130]
[1131]
[1132]
[1133]
[1134]
[1135]
[1136]
[1137]
[1138]
[1139]
[1140]
[1141]
[1142]
[1143]
[1144]
[1145]
[1146]
[1147]
[1148]
[1149]
[1150]
[1151]
[1152]
[1153]
[1154]
[1155]
[1156]
[1157]
[1158]
[1159]
[1160]
[1161]
[1162]
[1163]
[1164]
[1165]
[1166]
[1167]
[1168]
[1169]
[1170]
[1171]
[1172]
[1173]
[1174]
[1175]
[1176]
[1177]
[1178]
[1179]
[1180]
[1181]
[1182]
[1183]
[1184]
[1185]
[1186]
[1187]
[1188]
[1189]
[1190]
[1191]
[1192]
[1193]
[1194]
[1195]
[1196]
[1197]
[1198]
[1199]
[1200]
[1201]
[1202]
[1203]
[1204]
[1205]
[1206]
[1207]
[1208]
[1209]
[1210]
[1211]
[1212]
[1213]
[1214]
[1215]
[1216]
[1217]
[1218]
[1219]
[1220]
[1221]
[1222]
[1223]
[1224]
[1225]
[1226]
[1227]
[1228]
[1229]
[1230]
[1231]
[1232]
[1233]
[1234]
[1235]
[1236]
[1237]
[1238]
[1239]
[1240]
[1241]
[1242]
[1243]
[1244]
[1245]
[1246]
[1247]
[1248]
[1249]
[1250]
[1251]
[1252]
[1253]
[1254]
[1255]
[1256]
[1257]
[1258]
[1259]
[1260]
[1261]
[1262]
[1263]
[1264]
[1265]
[1266]
[1267]
[1268]
[1269]
[1270]
[1271]
[1272]
[1273]
[1274]
[1275]
[1276]
[1277]
[1278]
[1279]
[1280]
[1281]
[1282]
[1283]
[1284]
[1285]
[1286]
[1287]
[1288]
[1289]
[1290]
[1291]
[1292]
[1293]
[1294]
[1295]
[1296]
[1297]
[1298]
[1299]
[1300]
[1301]
[1302]
[1303]
[1304]
[1305]
[1306]
[1307]
[1308]
[1309]
[1310]
[1311]
[1312]
[1313]
[1314]
[1315]
[1316]
[1317]
[1318]
[1319]
[1320]
[1321]
[1322]
[1323]
[1324]
[1325]
[1326]
[1327]
[1328]
[1329]
[1330]
[1331]
[1332]
[1333]
[1334]
[1335]
[1336]
[1337]
[1338]
[1339]
[1340]
[1341]
[1342]
[1343]
[1344]
[1345]
[1346]
[1347]
[1348]
[1349]
[1350]
[1351]
[1352]
[1353]
[1354]
[1355]
[1356]
[1357]
[1358]
[1359]
[1360]
[1361]
[1362]
[1363]
[1364]
[1365]
[1366]
[1367]
[1368]
[1369]
[1370]
[1371]
[1372]
[1373]
[1374]
[1375]
[1376]
[1377]
[1378]
[1379]
[1380]
[1381]
[1382]
[1383]
[1384]
[1385]
[1386]
[1387]
[1388]
[1389]
[1390]
[1391]
[1392]
[1393]
[1394]
[1395]
[1396]
[1397]
[1398]
[1399]
[1400]
[1401]
[1402]
[1403]
[1404]
[1405]
[1406]
[1407]
[1408]
[1409]
[1410]
[1411]
[1412]
[1413]
[1414]
[1415]
[1416]
[1417]
[1418]
[1419]
[1420]
[1421]
[1422]
[1423]
[1424]
[1425]
[1426]
[1427]
[1428]
[1429]
[1430]
[1431]
[1432]
[1433]
[1434]
[1435]
[1436]
[1437]
[1438]
[1439]
[1440]
/*****************************************************************************/
/*
                                  Put.c

PUT, POST or DELETE a document.  Although the HTTP/1.1 specification
differentiates between the PUT and POST functionality this module does not.
Allows documents (files) and directories (i.e. specification ending in a slash)
to be created.  Either the DELETE method or a kludge allows these to be
deleted.  The kludge: if a wildcard version (";*") is included with the
specification the respective file or directory is deleted with either of the
non-DELETE methods.


Access Control
--------------

Relies on HTTPd authentication.  If a remote username has not been verified an
automatic no-privilege error is generated.

Access to create or delete documents is also controlled by any
permissions/protections/ACLs, etc. on the parent directory controlling access
by the HTTPd server account (usually HTTP$SERVER).  The explicit action
required to grant the server account access to a directory it needs to write
into is a bit of a nuisance, but deemed a good double check, preventing access
due to flawed authentication/authorization configuration (and hopefully even
a range of possible design problems or coding errors within this server :^)

Setting ACLs is preferable to granting world write access (obviously):

This ACE would grant PUT, POST or DELETE access to the server:

   $ SET ACL directory.DIR /ACL=(IDENT=HTTP$SERVER,ACCESS=READ+WRITE)

This ACE would explcitly deny POST access to the server:

   $ SET ACL directory.DIR /ACL=(IDENT=HTTP$SERVER,ACCESS=READ)

This ACE would explcitly deny ALL access to the server:

   $ SET ACL directory.DIR /ACL=(IDENT=HTTP$SERVER,ACCESS=NONE)

BE ULTRA-CAUTIOUS ... check access to directory before undertaking any action
that will alter anything within it, even if doing that potentially involves
some redundant processesing/checking!  In this way any programming oversights
will hopefully be ameliorated.

File Record Format
------------------

Text files are created with a STREAM_LF format and CR implied carriage-control. 
Binary files are created in undefined (UDF) record format with no record
attributes (carriage-control).  File writing is done using block I/O for
efficiency and record independence.


Content-Type: application/x-www-form-urlencoded
-----------------------------------------------

If the document 'Content-Type:' is "application/x-www-form-urlencoded" (i.e.
generated from an HTML form) all field names and delimiting symbols are
eliminated and the field(s) content only converted into plain text.  Hence a
text document can be POSTed using an HTML form with only the content ending up
in the file.  This processing is  performed by BodProcessUrlEncoded().


Content-Type: multipart/form-data
---------------------------------

This module can process a request body according to RFC-1867, "Form-based File
Upload in HTML".  As yet it is not a full implementation.  It will not process
"multipart/mixed" subsections.  The 'Content-Type:' is "multipart/form-data".
in the file.  This processing is performed by BodProcessMultipartFormData().


Other Considerations
--------------------

PUT/POSTed files automatically have a three version limit imposed.

If an error occurs (message generated) a file created by the request is
deleted.


WebDAV Support
--------------
The PUT module was essentially suitable for WebDAV support.  Only minor
modifications were made to support WebDAV's authorization model, agent
idiosyncracies, etc, and to shield WebDAV requests from the historical
idiosyncracies of WASD's PUT method support :-)


VERSION HISTORY
---------------
22-NOV-2020  MGD  content length now 64 bit
01-NOV-2018  MGD  bugfix; PutWriteFileOpen() override incompatible existing
                    file characteristics by first erasing the file
06-JUL-2014  MGD  PutWebDavBegin() migrate into DavWebPutBegin()
                  bugfix; PutWriteFileOpen() WebDAV should not use default
                    protection mask and instead propagate from profile
17-AUG-2013  MGD  PutWriteFileOpen() support FAB$C_STM and FAB$C_STMCR
10-MAY-2010  JPP  bugfix; PutWriteFileOpen() ensure SYSPRV enabled before
                  $ERASE() if not WebDAV request (for access and ownership)
24-MAR-2010  MGD  bugfix; PutWriteFileOpen() ensure SYSPRV enabled before
                  $CREATE() if not WebDAV request (for access and ownership)
21-JUN-2009  MGD  PutWriteFileOpen() allow record format of binary files
                  to be specified by global configuration and path setting
01-MAY-2007  MGD  WebDAV considerations,
                  "If-Match:.." and "If-Not-Match:.." conditional processing,
                  pre-allocate space if the size is fixed (viz. not a form),
                  FAB and RAB flags to enable truncate-on-put,
                  PutBegin() remove 'SpecificDirectory' parameter
30-OCT-2004  MGD  bugfix; PutWriteFileOpen() always check 'prptr' non-NULL
24-JUL-2004  MGD  bugfix; (potential anyway) PutWriteFileClose()/PutEnd(),
                  in the absence of a content-type assume bag-o'-bytes
10-JAN-2004  MGD  PutWriteFileOpen() 'delete-on-close' file specification
                  extended to include a four digit global 'uniquifier'
                  (with faster systems and multiple instances it was entirely
                  possible that purely a time-based name might be inadequite)
23-AUG-2002  MGD  set fab$b_rfm and fab$b_rat (is supplied with body) to
                  establish file attributes (falling back to type analysis)
02-FEB-2002  MGD  rework file processing for request body processing changes
04-AUG-2001  MGD  support module WATCHing
04-JUL-2000  MGD  redirect from POST (success=)
04-JAN-2000  MGD  support ODS-2 and ODS-5 using ODS module
12-MAR-1998  MGD  file protection may now be specified (as hexadecimal value)
25-SEP-1997  MGD  bugfix; PutProcessText() removed CRLF munging
17-AUG-1997  MGD  message database,
                  SYSUAF-authenticated users security-profile
27-MAR-1997  MGD  provide file edit preview (see UPD.C)
01-FEB-1997  MGD  HTTPd version 4
01-SEP-1996  MGD  provide "Content-Type: multipart/form-data" for file upload
06-APR-1996  MGD  initial development
*/
/*****************************************************************************/

#ifdef WASD_VMS_V7
#undef _VMS__V6__SOURCE
#define _VMS__V6__SOURCE
#undef __VMS_VER
#define __VMS_VER 70000000
#undef __CRTL_VER
#define __CRTL_VER 70000000
#endif

/* standard C header files */
#include <ctype.h>
#include <stdio.h>
#include <string.h>

/* VMS related header files */
#include <acldef.h>
#include <armdef.h>
#include <chpdef.h>
#include <descrip.h>
#include <iodef.h>
#include <jpidef.h>
#include <prvdef.h>
#include <rms.h>
#include <rmsdef.h>
#include <ssdef.h>
#include <stsdef.h>

/* application header files */
#include "wasd.h"

#define WASD_MODULE "PUT"

/***************/
/* definitions */
/***************/

#define PUT_FILE 0x01
#define PUT_DIRECTORY 0x02
#define PUT_UPLOAD 0x04
#define PUT_CREATED 0x10
#define PUT_DELETED 0x20
#define PUT_SUPERCEDED 0x40

/******************/
/* global storage */
/******************/

BOOL  PutOnlyTextFilesStreamLf = true;

/********************/
/* external storage */
/********************/

#ifdef DBUG
extern BOOL Debug;
#else
#define Debug 0 
#endif

extern BOOL  OdsExtended;

extern unsigned long  SysPrvMask[];

extern char  SoftwareID[],
             ErrorSanityCheck[];

extern ACCOUNTING_STRUCT  *AccountingPtr;
extern CONFIG_STRUCT  Config;
extern MSG_STRUCT  Msgs;
extern WATCH_STRUCT  Watch;

/*****************************************************************************/
/*
Begin PUT processing.
*/ 
 
PutBegin
(
REQUEST_STRUCT *rqptr,
REQUEST_AST NextTaskFunction
)
{
   PUT_TASK  *tkptr;

   /*********/
   /* begin */
   /*********/

   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "PutBegin() !&A !@SQ !&Z",
                 NextTaskFunction, rqptr->rqHeader.ContentLength64,
                 rqptr->ParseOds.NamDevicePtr);

   if (ERROR_REPORTED (rqptr))
   {
      /* previous error, cause threaded processing to unravel */
      SysDclAst (NextTaskFunction, rqptr);
      return;
   }

   if (!rqptr->AccountingDone++)
      InstanceGblSecIncrLong (&AccountingPtr->DoPutCount);

   /* authentication is mandatory for a PUT, DELETE or POST */
   if (!rqptr->RemoteUser[0] ||
       !((rqptr->rqAuth.RequestCan & HTTP_METHOD_PUT) ||
         (rqptr->rqAuth.RequestCan & HTTP_METHOD_POST) ||
         (rqptr->rqAuth.RequestCan & HTTP_METHOD_DELETE)))
   {
      rqptr->rqResponse.ErrorTextPtr = rqptr->rqHeader.PathInfoPtr;
      ErrorVmsStatus (rqptr, SS$_NOPRIV, FI_LI);
      SysDclAst (NextTaskFunction, rqptr);
      return;
   }

   /* set up the task structure (only ever one per request!) */
   rqptr->PutTaskPtr = tkptr = (PUT_TASK*) VmGetHeap (rqptr, sizeof(PUT_TASK));
   tkptr->NextTaskFunction = NextTaskFunction;

   OdsStructInit (&tkptr->SearchOds, false);

   if (rqptr->WebDavTaskPtr)
      tkptr->ProtectionMask = 0;
   else
      tkptr->ProtectionMask = PUT_DEFAULT_FILE_PROTECTION;

   if (rqptr->rqHeader.IfMatchPtr ||
       rqptr->rqHeader.IfNoneMatchPtr)
      PutIfBegin (rqptr);
   else
   if (rqptr->WebDavTaskPtr)
      DavWebPutBegin (rqptr);
   else
      PutWasdBegin (rqptr);
}

/*****************************************************************************/
/*
Begin the WASD-idiosyncratic PUT processing.
*/ 
 
PutWasdBegin (REQUEST_STRUCT *rqptr)

{
   PUT_TASK  *tkptr;

   /*********/
   /* begin */
   /*********/

   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "PutWasdBegin()");

   tkptr = rqptr->PutTaskPtr;

   if ((rqptr->rqHeader.Method == HTTP_METHOD_DELETE) ||
       (rqptr->ParseOds.NamVersionPtr[0] == ';' &&
        rqptr->ParseOds.NamVersionPtr[1] == '*'))
   {
      /**********************************************/
      /* delete (or file/directory deletion kludge) */
      /**********************************************/

      /* terminate on either the directory or file name */
      if (!rqptr->ParseOds.NamNameLength &&
          !rqptr->ParseOds.NamTypeLength)
         rqptr->ParseOds.NamNamePtr[0] = '\0';
      else
         rqptr->ParseOds.NamVersionPtr[0] = '\0';

      if (WATCHING (rqptr, WATCH_RESPONSE))
         WatchThis (WATCHITM(rqptr), WATCH_RESPONSE,
                    "PUT !AZ/delete !AZ",
                    rqptr->rqHeader.MethodName,
                    rqptr->ParseOds.NamDevicePtr);

      PutDelete (rqptr);
      PutEnd (rqptr);
      return;
   }

   if (!rqptr->ParseOds.NamNameLength &&
       !rqptr->ParseOds.NamTypeLength)
   {
      /********************/
      /* create directory */
      /********************/

      /* multipart/form-data supplies the parent directory as the path */
      if (!ConfigSameContentType (rqptr->rqHeader.ContentTypePtr,
                                  "multipart/", 10))
      {
         if (WATCHING (rqptr, WATCH_RESPONSE))
            WatchThis (WATCHITM(rqptr), WATCH_RESPONSE,
                       "PUT !AZ/create !AZ",
                       rqptr->rqHeader.MethodName,
                       rqptr->ParseOds.NamDevicePtr);

         PutCreateDirectory (rqptr);
         PutEnd (rqptr);
         return;
      }
   }

   /* terminate on version for file name, name for directory */
   if (!rqptr->ParseOds.NamNameLength &&
       rqptr->ParseOds.NamTypeLength == 1 &&
       rqptr->ParseOds.NamVersionLength == 1)
      *rqptr->ParseOds.NamNamePtr = '\0';
   else
      *rqptr->ParseOds.NamVersionPtr = '\0';

   if (WATCHING (rqptr, WATCH_RESPONSE))
      WatchThis (WATCHITM(rqptr), WATCH_RESPONSE,
                 "PUT !AZ/create !AZ as \"!AZ\"",
                 rqptr->rqHeader.MethodName,
                 rqptr->ParseOds.NamDevicePtr,
                 rqptr->rqHeader.ContentTypePtr ?
                    rqptr->rqHeader.ContentTypePtr :
                    "(unspecified content-type)");

   if (ConfigSameContentType (rqptr->rqHeader.ContentTypePtr,
                              "application/x-www-form-urlencoded", -1))
      BodyReadBegin (rqptr, &PutWriteFile, &BodyProcessUrlEncoded);
   else
   if (ConfigSameContentType (rqptr->rqHeader.ContentTypePtr,
                              "multipart/", 10))
      BodyReadBegin (rqptr, &PutWriteFile, &BodyProcessMultipartFormData);
   else
   {
      tkptr->ContentLengthFixed = true;
      BodyReadBegin (rqptr, &PutWriteFile, &BodyProcessByVirtualBlock);
   }
}

/*****************************************************************************/
/*
If "If-Match: <entity>" or "If-Not-Match: <entity>" conditional processing has
been requested then get the (possible) file details, generate entity data from
those details, and continue processing (or not) according to the match.
This function (re)calls itself multiple times during processing.
*/ 
 
PutIfBegin (REQUEST_STRUCT *rqptr)

{
   int  status;
   char  EntityTag [32];
   PUT_TASK  *tkptr;

   /*********/
   /* begin */
   /*********/

   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "PutIfBegin()");

   tkptr = rqptr->PutTaskPtr;

   if (!tkptr->SearchOds.Fab.fab$l_sts)
   {
      /* use SYSPRV to ensure access */
      sys$setprv (1, &SysPrvMask, 0, 0);
      OdsParse (&tkptr->SearchOds,
                rqptr->ParseOds.ExpFileName,
                rqptr->ParseOds.ExpFileNameLength,
                NULL, 0, 0, &PutIfBegin, rqptr);
      sys$setprv (0, &SysPrvMask, 0, 0);
      return;
   }

   if (VMSnok(status = tkptr->SearchOds.Fab.fab$l_sts))
   {
      OdsParseRelease (&tkptr->SearchOds);
      if (rqptr->WebDavTaskPtr)
         DavWebResponse (rqptr, 0, status, "entity tag", FI_LI);
      else
         ErrorVmsStatus (rqptr, status, FI_LI);
      PutEnd (rqptr);
      return;
   }

   if (!tkptr->SearchOds.FileQio.IOsb.Status)
   {
      /* use SYSPRV to ensure access */
      sys$setprv (1, &SysPrvMask, 0, 0);
      OdsFileAcpInfo (&tkptr->SearchOds, &PutIfBegin, rqptr);
      sys$setprv (0, &SysPrvMask, 0, 0);
      return;
   }

   /* deassign the channel allocated by OdsFileAcpInfo() */
   sys$dassgn (tkptr->SearchOds.FileQio.AcpChannel);

   if (VMSok(status = tkptr->SearchOds.FileQio.IOsb.Status))
   {
      /* file exists, generate entity tag from the data */
      OdsParseRelease (&tkptr->SearchOds);
      FileGenerateEntityTag (EntityTag, &tkptr->SearchOds.FileQio);
   }
   else
   {
      /* error accessing file */
      OdsParseRelease (&tkptr->SearchOds);
      if (rqptr->WebDavTaskPtr)
         DavWebResponse (rqptr, 0, status, "entity tag", FI_LI);
      else
         ErrorVmsStatus (rqptr, status, FI_LI);
      PutEnd (rqptr);
      return;
   }

   if (!ResponseEntityMatch (rqptr, EntityTag))
   {
      PutEnd (rqptr);
      return;
   }

   if (rqptr->WebDavTaskPtr)
      DavWebPutBegin (rqptr);
   else
      PutWasdBegin (rqptr);
}

/*****************************************************************************/
/*
Conclude processing the request.  If a temporary file name/path was generated
then it's a preview only, turn the POST into a GET of the temporary file path.
*/

PutEnd (REQUEST_STRUCT *rqptr)

{
   int  status;
   PUT_TASK  *tkptr;

   /*********/
   /* begin */
   /*********/

   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "PutEnd()");

   tkptr = rqptr->PutTaskPtr;

   /* indicate the PUT task has concluded correctly */
   rqptr->PutTaskPtr = NULL;

   if (tkptr->FileOpen) PutWriteFileClose (rqptr);
   SysDclAst (tkptr->NextTaskFunction, rqptr);
}

/*****************************************************************************/
/*
Check the status result of the preceding request body network read or content
processing using 'rqptr->rqBody.DataStatus'.  Write (i.e. block I/O) the data
represented by 'rqptr->rqBody.DataPtr' and 'rqptr->rqBody.DataCount'.  For all
writes except the possible final one this should be a number of complete
virtual blocks (512 bytes) beginning at 'rqptr->rqBody.DataVBN'.  The final one
may be any number of bytes between 1 and 511.
*/ 

PutWriteFile (REQUEST_STRUCT *rqptr)

{
   int  status;
   PUT_TASK  *tkptr;

   /*********/
   /* begin */
   /*********/

   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT,
                 "PutWriteFile() !&F !&X !UL !UL !&S", &PutWriteFile,
                 rqptr->rqBody.DataPtr, rqptr->rqBody.DataCount,
                 rqptr->rqBody.DataVBN, rqptr->rqBody.DataStatus);

   tkptr = rqptr->PutTaskPtr;

   if (VMSok (rqptr->rqBody.DataStatus))
   {
      if (!tkptr->FileOpen)
      {
         status = PutWriteFileOpen (rqptr);
         if (VMSnok (status)) return;
      }
      tkptr->FileSizeBytes += rqptr->rqBody.DataCount;
      tkptr->FileOds.Rab.rab$l_rbf = rqptr->rqBody.DataPtr;
      tkptr->FileOds.Rab.rab$w_rsz = rqptr->rqBody.DataCount;
      tkptr->FileOds.Rab.rab$l_bkt = rqptr->rqBody.DataVBN;
      sys$write (&tkptr->FileOds.Rab, &PutWriteFileAst, &PutWriteFileAst);
      return;
   }

   if (rqptr->rqBody.DataStatus == SS$_ENDOFFILE)
   {
      /* body is exhausted (aren't we all?) */
      if (!tkptr->FileOpen)
      {
         status = PutWriteFileOpen (rqptr);
         if (VMSnok (status)) return;
      }
      PutWriteFileClose (rqptr);
      PutEnd (rqptr);
      return;
   }

   /* error reading or processing request body */
   if (rqptr->WebDavTaskPtr)
      if (WATCHING (rqptr, WATCH_WEBDAV))
         WatchThis (WATCHITM(rqptr), WATCH_WEBDAV, "PUT write !&S !AZ",
                    rqptr->rqBody.DataStatus, tkptr->FileOds.ExpFileName);

   rqptr->rqResponse.ErrorTextPtr = MsgFor(rqptr,MSG_REQUEST_BODY_READ);
   ErrorVmsStatus (rqptr, rqptr->rqBody.DataStatus, FI_LI);
   PutWriteFileClose (rqptr);
   PutEnd (rqptr);
}

/*****************************************************************************/
/*
After each block I/O written this AST function is called to either write more
data, or on conlusion  to call the end file function.  If an error is reported
from the write the end file function is called, with the presence of the error
message causing the file to be deleted.
*/ 

PutWriteFileAst (struct RAB *RabPtr)

{
   int  status;
   REQUEST_STRUCT *rqptr;
   PUT_TASK  *tkptr;

   /*********/
   /* begin */
   /*********/

   rqptr = RabPtr->rab$l_ctx;

   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT,
                 "PutWriteFileAst() !&F sts:!&X stv:!&X rsz:!UL",
                 &PutWriteFileAst,
                 RabPtr->rab$l_sts, RabPtr->rab$l_stv, RabPtr->rab$w_rsz);

   tkptr = rqptr->PutTaskPtr;

   if (VMSok (tkptr->FileOds.Rab.rab$l_sts))
   {
      /* read more form the request body */
      BodyRead (rqptr);
      return;
   }

   rqptr->rqResponse.ErrorTextPtr = rqptr->rqHeader.PathInfoPtr;
   rqptr->rqResponse.ErrorOtherTextPtr = tkptr->FileName;
   ErrorVmsStatus (rqptr, tkptr->FileOds.Rab.rab$l_sts, FI_LI);
   PutWriteFileClose (rqptr);
   PutEnd (rqptr);
}

/*****************************************************************************/
/*
Create a file using '->FileName'.  Fill with '->ContentFileLength' bytes
from '->ContentFilePtr'.  Return to processing at '->NextFunction'
(providing there was no problem!)
*/

PutWriteFileOpen (REQUEST_STRUCT *rqptr)

{
   int  status,
        EraseCount,
        PutRFM,
        TotalCount;
   char  *cptr, *sptr, *zptr,
         *ContentTypePtr;
   char  FileName [256];
   CONTENT_TYPE  ContType;
   BODY_PROCESS  *prptr;
   PUT_TASK  *tkptr;

   /*********/
   /* begin */
   /*********/

   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "PutWriteFileOpen()");

   tkptr = rqptr->PutTaskPtr;
   prptr = rqptr->rqBody.ProcessPtr;

   if (rqptr->rqBody.ProcessedAs == BODY_PROCESSED_AS_MULTIPART_FORMDATA)
   {
      if (!prptr)
      {
         /* must have had the body processed before we can store it! */
         ErrorInternal (rqptr, 0, ErrorSanityCheck, FI_LI);
         PutEnd (rqptr);
         return (SS$_BUGCHECK);
      }
      if (!prptr->MultipartFileName[0])
      {
         rqptr->rqResponse.HttpStatus = 400;
         ErrorGeneral (rqptr, MsgFor(rqptr,MSG_PUT_MULTIPART_FILENAME), FI_LI);
         PutEnd (rqptr);
         return (SS$_BADPARAM);
      }

      if (WATCHMOD (rqptr, WATCH_MOD_PUT))
         WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "!&Z !&Z !&Z !&B",
                    rqptr->ParseOds.NamDevicePtr,
                    prptr->MultipartUploadFileName,
                    prptr->MultipartFileName, prptr->PreviewOnly);
   }
   else
   {
      if (!rqptr->ParseOds.NamNameLength &&
          !rqptr->ParseOds.NamTypeLength)
      {
         /* must have the parsed file name */
         ErrorInternal (rqptr, 0, ErrorSanityCheck, FI_LI);
         PutEnd (rqptr);
         return (SS$_BUGCHECK);
      }
      if (WATCHMOD (rqptr, WATCH_MOD_PUT))
         WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "!&Z",
                    rqptr->ParseOds.NamDevicePtr);
   }

   if (rqptr->ParseOds.NamTypeLength)
      cptr = rqptr->ParseOds.NamTypePtr;
   else
   if (prptr && prptr->MultipartUploadFileName[0])
   {
      /* try to conjure up a fit-for-purpose file type */
      for (cptr = prptr->MultipartUploadFileName; *cptr; cptr++);
      while (*cptr != '.' && cptr > prptr->MultipartUploadFileName) cptr--;
      if (*cptr != '.')
      {
         for (cptr = prptr->MultipartFileName; *cptr; cptr++);
         while (*cptr != '.' && cptr > prptr->MultipartFileName) cptr--;
         if (*cptr != '.') cptr = "";
      }
   }
   else
      cptr = "";

   if (rqptr->rqHeader.ContentTypePtr)
   {
      tkptr->FileContentTypePtr = rqptr->rqHeader.ContentTypePtr;
      ConfigContentType (&ContType, cptr);
   }
   else
   {
      /*
         If no MIME content-type is supplied with the request, as
         commonly occurs with Microsoft's WevDAV redirector, then
         attempt to come up with a plausable content-type from the
         file type (extension).  This is useful information when a
         little later setting record characteristics on the file.
      */
      tkptr->FileContentTypePtr = ConfigContentType (&ContType, cptr);
   }

   zptr = (sptr = tkptr->FileName) + sizeof(tkptr->FileName);

   /* use directory from path and form field 'uploadfilename' */
   for (cptr = rqptr->ParseOds.NamDevicePtr;
        cptr < rqptr->ParseOds.NamNamePtr && sptr < zptr;
        *sptr++ = *cptr++);

   if (prptr && (prptr->MultipartFileName[0] ||
                 prptr->MultipartUploadFileName[0]))
   {
      if (!*(cptr = prptr->MultipartUploadFileName))
      {
         cptr = prptr->MultipartFileName;
         while (*cptr) cptr++;
         while (cptr > prptr->MultipartFileName &&
                *cptr != '/' && *cptr != '\\' && *cptr != ']') cptr--;
         if (*cptr == '/' || *cptr == '\\' || *cptr == ']') cptr++;
      }
      MapOdsUrlToVms (cptr, FileName, sizeof(FileName), 0,
                      rqptr->rqPathSet.MapEllipsis, rqptr->PathOds);
      for (cptr = FileName; *cptr && sptr < zptr; *sptr++ = *cptr++);
      /* update the content-type with whatever is specified in the part */
      tkptr->FileContentTypePtr = prptr->MultipartContentTypePtr;
   }
   else
   if (prptr && (tkptr->PreviewOnly = prptr->PreviewOnly))
   {
      /* use request total count to add a *unique* value to the file name */
      InstanceMutexLock (INSTANCE_MUTEX_HTTPD);
      TotalCount = AccountingPtr->ProcessingTotalCount[HTTP12];
      InstanceMutexUnLock (INSTANCE_MUTEX_HTTPD);
      /* file name to indicate it's to be deleted on request rundown */
      FaoToBuffer (FileName, sizeof(FileName), NULL,
                   "-!4ZL!2ZL!2ZL!2ZL!2ZL!2ZL!2ZL!4ZL-",
                   rqptr->rqTime.BeginTime7[0], rqptr->rqTime.BeginTime7[1],
                   rqptr->rqTime.BeginTime7[2], rqptr->rqTime.BeginTime7[3],
                   rqptr->rqTime.BeginTime7[4], rqptr->rqTime.BeginTime7[5],
                   rqptr->rqTime.BeginTime7[6], TotalCount % 1000);
      for (cptr = FileName; *cptr && sptr < zptr; *sptr++ = *cptr++);
      /* append the request file type to it */
      for (cptr = rqptr->ParseOds.NamTypePtr;
           cptr < rqptr->ParseOds.NamVersionPtr && sptr < zptr;
           *sptr++ = *cptr++);
   }
   else
   {
      /* use the file name and type derived from the request path */
      for (cptr = rqptr->ParseOds.NamNamePtr;
           cptr < rqptr->ParseOds.NamVersionPtr && sptr < zptr;
           *sptr++ = *cptr++);
   }

   if (sptr >= zptr)
   {
      ErrorGeneralOverflow (rqptr, FI_LI);
      PutEnd (rqptr);
      return (SS$_RESULTOVF);
   }
   *sptr = '\0';
   tkptr->FileNameLength = sptr - tkptr->FileName;

   if (WATCHING (rqptr, WATCH_RESPONSE))
      WatchThis (WATCHITM(rqptr), WATCH_RESPONSE, "!AZ", tkptr->FileName);

   tkptr->FileOpen = false;

   tkptr->FileOds.Fab = cc$rms_fab;
   tkptr->FileOds.Fab.fab$b_fac = FAB$M_PUT | FAB$M_BIO | FAB$M_TRN;
   tkptr->FileOds.Fab.fab$l_fop = FAB$M_SQO | FAB$M_TEF;

   /* pre-allocate space if the final size is fixed */
   if (tkptr->ContentLengthFixed)
   {
      tkptr->FileOds.Fab.fab$l_alq = (rqptr->rqBody.ContentLength64 >> 9) + 1;

      if (rqptr->WebDavTaskPtr &&
          rqptr->WebDavTaskPtr->MicrosoftAgent)
      {
         /*
             Microsoft's file creation sequence appears to be to PUT 
             zero bytes, then to PROPPATCH some win32 attributes, and
             then finally PUT lotsa bytes; as three independent requests.
             Propagate the allocation quantity so that the originally
             created zero-length file gets extended to the full size 
             required upon the first write.
         */
         tkptr->FileOds.Fab.fab$w_deq = (short)tkptr->FileOds.Fab.fab$l_alq;
      }
   }

   /* if it's a WebDAV PUT then confine to a single version */
   if (rqptr->WebDavTaskPtr)
      tkptr->FileOds.Fab.fab$l_fop |= FAB$M_CIF;

   tkptr->FileOds.Fab.fab$l_nam = &tkptr->FileOds.Nam;
   tkptr->FileOds.Fab.fab$b_rat = 0;

   if (prptr && prptr->UrlEncodedFabRfm &&
                prptr->UrlEncodedFabRat)
   {
      tkptr->FileOds.Fab.fab$b_rfm = prptr->UrlEncodedFabRfm;
      tkptr->FileOds.Fab.fab$b_rat = prptr->UrlEncodedFabRat;
      if (WATCHING (rqptr, WATCH_RESPONSE))
         WatchThis (WATCHITM(rqptr), WATCH_RESPONSE, "RFM:0x!2XL RAT:0x!2XL",
                    tkptr->FileOds.Fab.fab$b_rfm, tkptr->FileOds.Fab.fab$b_rat);
   }
   else
   /* the [AddType] RFM will always override identification as text */
   if (!ContType.PutRFM &&
       ConfigSameContentType (tkptr->FileContentTypePtr, "text/", 5))
   {
      /* "textual" content-type, guess about the _best_ RMS record format! */
      int  CrLfCount, LfCount;
      CrLfCount = LfCount = 0;
      if (rqptr->rqBody.DataCount > 1024)
         zptr = rqptr->rqBody.DataPtr + 1024;
      else
         zptr = rqptr->rqBody.DataPtr + rqptr->rqBody.DataCount;
      for (cptr = rqptr->rqBody.DataPtr; cptr < zptr; cptr++)
      {
         if (SAME2(cptr,'\r\n'))
         {
            CrLfCount++;
            cptr++;
         }
         else
         if (*cptr == '\n')
            LfCount++;
      }
      if (CrLfCount >= LfCount)
      {
         /* more CR+LFs than just LFs (DOS-style), STREAM */
         tkptr->FileOds.Fab.fab$b_rfm = FAB$C_STM;
         tkptr->FileOds.Fab.fab$b_rat = FAB$M_CR;
         if (WATCHING (rqptr, WATCH_RESPONSE))
            WatchThis (WATCHITM(rqptr), WATCH_RESPONSE, "RFM: stream (DOS)");
      }
      else
      {
         /* STREAM-LF (Unix-style) */
         tkptr->FileOds.Fab.fab$b_rfm = FAB$C_STMLF;
         tkptr->FileOds.Fab.fab$b_rat = FAB$M_CR;
         if (WATCHING (rqptr, WATCH_RESPONSE))
            WatchThis (WATCHITM(rqptr), WATCH_RESPONSE, "RFM: stream-LF");
      }
      tkptr->FileOds.Fab.fab$b_rat = FAB$M_CR;
   }
   else
   /* the [AddType] RFM will always override identification as text */
   if (!ContType.PutRFM &&
       ConfigSameContentType (tkptr->FileContentTypePtr,
                              "application/x-www-form-urlencoded", -1))
   {
      /* STREAM-LF (Unix-style) */
      tkptr->FileOds.Fab.fab$b_rfm = FAB$C_STMLF;
      tkptr->FileOds.Fab.fab$b_rat = FAB$M_CR;
      if (WATCHING (rqptr, WATCH_RESPONSE))
         WatchThis (WATCHITM(rqptr), WATCH_RESPONSE, "RFM: stream-LF");
   }
   else
   {
      /* binary - precedence: SET path then [AddType] then global config */       
      if (!(PutRFM = rqptr->rqPathSet.PutRFM))
         if (!(PutRFM = ContType.PutRFM))
            PutRFM = Config.cfMisc.PutBinaryRFM;

      if (PutRFM == PUT_RFM_FIX512)
      {
         tkptr->FileOds.Fab.fab$b_rfm = FAB$C_FIX;
         tkptr->FileOds.Fab.fab$w_mrs = 512;
         tkptr->FileOds.Fab.fab$b_rat = 0;
         cptr = "fixed-512";
      }
      else
      if (PutRFM == PUT_RFM_STM)
      {
         tkptr->FileOds.Fab.fab$b_rfm = FAB$C_STM;
         tkptr->FileOds.Fab.fab$b_rat = FAB$M_CR;
         cptr = "stream";
      }
      else
      if (PutRFM == PUT_RFM_STMCR)
      {
         tkptr->FileOds.Fab.fab$b_rfm = FAB$C_STMCR;
         tkptr->FileOds.Fab.fab$b_rat = FAB$M_CR;
         cptr = "stream-CR";
      }
      else
      if (PutRFM == PUT_RFM_STMLF)
      {
         tkptr->FileOds.Fab.fab$b_rfm = FAB$C_STMLF;
         tkptr->FileOds.Fab.fab$b_rat = FAB$M_CR;
         cptr = "stream-LF";
      }
      else
      {
         tkptr->FileOds.Fab.fab$b_rfm = FAB$C_UDF;
         tkptr->FileOds.Fab.fab$b_rat = 0;
         cptr = "undefined";
      }
      if (WATCHING (rqptr, WATCH_RESPONSE))
         WatchThis (WATCHITM(rqptr), WATCH_RESPONSE, "RFM: !AZ (non-text)", cptr);
   }

   tkptr->FileOds.Fab.fab$b_shr = FAB$M_NIL;

#ifdef ODS_EXTENDED
   if (OdsExtended)
   {
      tkptr->FileOds.Fab.fab$l_fna = -1;
      tkptr->FileOds.Fab.fab$b_fns = 0;
      tkptr->FileOds.Fab.fab$l_nam = &tkptr->FileOds.Naml;

      tkptr->FileOds.NamlInUse = true;
      ENAMEL_RMS_NAML(tkptr->FileOds.Naml)
      tkptr->FileOds.Naml.naml$l_long_filename = tkptr->FileName;
      tkptr->FileOds.Naml.naml$l_long_filename_size = tkptr->FileNameLength;
      tkptr->FileOds.Naml.naml$l_filesys_name = tkptr->FileOds.SysFileName;
      tkptr->FileOds.Naml.naml$l_filesys_name_alloc =
         sizeof(tkptr->FileOds.SysFileName)-1;
      tkptr->FileOds.Naml.naml$l_long_expand = tkptr->FileOds.ExpFileName;
      tkptr->FileOds.Naml.naml$l_long_expand_alloc =
         sizeof(tkptr->FileOds.ExpFileName)-1;
      tkptr->FileOds.Naml.naml$l_long_result = tkptr->FileOds.ResFileName;
      tkptr->FileOds.Naml.naml$l_long_result_alloc =
         sizeof(tkptr->FileOds.ResFileName)-1;
   }
   else
#endif /* ODS_EXTENDED */
   {
      tkptr->FileOds.Fab.fab$l_fna = tkptr->FileName;
      tkptr->FileOds.Fab.fab$b_fns = tkptr->FileNameLength;
      tkptr->FileOds.Fab.fab$l_nam = &tkptr->FileOds.Nam;

      tkptr->FileOds.NamlInUse = false;
      tkptr->FileOds.Nam = cc$rms_nam;
      tkptr->FileOds.Nam.nam$l_esa = tkptr->FileOds.ExpFileName;
      tkptr->FileOds.Nam.nam$b_ess = ODS2_MAX_FILE_NAME_LENGTH;
      tkptr->FileOds.Nam.nam$l_rsa = tkptr->FileOds.ResFileName;
      tkptr->FileOds.Nam.nam$b_rss = ODS2_MAX_FILE_NAME_LENGTH;
   }

   if (rqptr->WebDavTaskPtr)
   {
      /* propagate the profile security */
      AuthAccessEnable (rqptr, tkptr->FileName, AUTH_ACCESS_WRITE);
   }
   else
   {
      if (prptr && isxdigit(prptr->ProtectionHexString[0]))
         tkptr->ProtectionMask = strtol (prptr->ProtectionHexString, NULL, 16);

      if (WATCHMOD (rqptr, WATCH_MOD_PUT))
         WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "0x!4XL",
                    tkptr->ProtectionMask);

      /* initialize the protection extended attribute block */
      tkptr->FileOds.Fab.fab$l_xab = &tkptr->FileOds.XabPro;
      tkptr->FileOds.XabPro = cc$rms_xabpro;
      tkptr->FileOds.XabPro.xab$w_pro = tkptr->ProtectionMask;

      /* use SYSPRV to ensure appropriate access/ownership */
      sys$setprv (1, &SysPrvMask, 0, 0);
   }

   tkptr->FileOds.Fab.fab$l_ctx = &tkptr->FileOds;
   if (VMSok (status = sys$parse (&tkptr->FileOds.Fab, 0, 0)))
      OdsNamBlockAst (&tkptr->FileOds.Fab);

   /* override any (block I/O) incompatible existing file characteristics */
   if (VMSok (status = OdsFileAcpInfo (&tkptr->FileOds, NULL, rqptr)))
   {
      if ((tkptr->FileOds.FileQio.RecAttr.fat$b_rtype !=
           tkptr->FileOds.Fab.fab$b_rfm) ||
          (tkptr->FileOds.FileQio.RecAttr.fat$b_rattrib !=
           tkptr->FileOds.Fab.fab$b_rat))
      {
         if (WATCHING (rqptr, WATCH_RESPONSE))
            WatchThis (WATCHITM(rqptr), WATCH_RESPONSE,
                       "INCOMPATIBLE existing!AZ!AZ modified",
                       (tkptr->FileOds.FileQio.RecAttr.fat$b_rtype !=
                        tkptr->FileOds.Fab.fab$b_rfm) ? " RFM" : "",
                       (tkptr->FileOds.FileQio.RecAttr.fat$b_rattrib,
                        tkptr->FileOds.Fab.fab$b_rat) ? " RAT" : "");

         for (EraseCount = 0;
              VMSok (status = sys$erase (&tkptr->FileOds.Fab, 0, 0));
              EraseCount++);
         if (status == RMS$_FNF && EraseCount) status = SS$_NORMAL;

         if (WATCHMOD (rqptr, WATCH_MOD_PUT))
            WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT,
                       "sys$erase() !&S", status);
      }
   }

   if (VMSok (status) || status == SS$_NOSUCHFILE)
      status = sys$create (&tkptr->FileOds.Fab, 0, 0);

   if (rqptr->WebDavTaskPtr)
      AuthAccessEnable (rqptr, 0, 0);
   else
      sys$setprv (0, &SysPrvMask, 0, 0);

   if (VMSnok (status) && tkptr->FileOds.Fab.fab$l_stv)
      status = tkptr->FileOds.Fab.fab$l_stv;

   if (VMSnok (status))
   {
      /* sys$create() error */
      if (rqptr->WebDavTaskPtr)
         if (WATCHING (rqptr, WATCH_WEBDAV))
            WatchThis (WATCHITM(rqptr), WATCH_WEBDAV, "PUT create !&S !AZ",
                       status, tkptr->FileOds.ExpFileName);

      rqptr->rqResponse.ErrorTextPtr = rqptr->rqHeader.PathInfoPtr;
      rqptr->rqResponse.ErrorOtherTextPtr = tkptr->FileName;
      ErrorVmsStatus (rqptr, status, FI_LI);

      PutWriteFileClose (rqptr);

      PutEnd (rqptr);

      return (status);
   }

   tkptr->FileOpen = true;
   tkptr->SysCreateStatus = status;

   /* set up the generic information from the NAM(L) block */
   tkptr->FileOds.Fab.fab$l_ctx = &tkptr->FileOds;
   OdsNamBlockAst (&tkptr->FileOds.Fab);

   tkptr->FileOds.Rab = cc$rms_rab;
   tkptr->FileOds.Rab.rab$l_fab = &tkptr->FileOds.Fab;
   tkptr->FileOds.Rab.rab$l_ctx = rqptr;
   tkptr->FileOds.Rab.rab$l_rop = RAB$M_BIO | RAB$M_ASY | RAB$M_TPT;

   if (VMSnok (status = sys$connect (&tkptr->FileOds.Rab, 0, 0)))
   {
      rqptr->rqResponse.ErrorTextPtr = rqptr->rqHeader.PathInfoPtr;
      rqptr->rqResponse.ErrorOtherTextPtr = tkptr->FileName;
      ErrorVmsStatus (rqptr, status, FI_LI);
      PutWriteFileClose (rqptr);
      PutEnd (rqptr);
   }

   return (status);
}

/*****************************************************************************/
/*
Called when the file has been completely written or an error has been detected.
The presence of an error message results in the file being deleted. If OK the
file attributes are changed to limit versions and to stream-LF is "textual".

There is a small window between closing the file and changing the attributes
where it could conceivably be opened for write by another process and interfere
with that change.  Not a problem within this one server because all this
processing is occuring at user-AST-delivery level.  Don't know what to do about
it for the moment so I'll just say it's a low-risk scenario and live with it
for now!
*/ 

PutWriteFileClose (REQUEST_STRUCT *rqptr)

{
   static int  SingleVersion = 1;

   int  status;
   char  *cptr, *sptr, *tptr;
   char  ProtectionString [32];
   PUT_TASK  *tkptr;

   /*********/
   /* begin */
   /*********/

   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "PutWriteFileClose()");

   tkptr = rqptr->PutTaskPtr;

   if (!tkptr->FileOpen) return;
   tkptr->FileOpen = false;

   if (tkptr->FileOds.Fab.fab$w_ifi) sys$close (&tkptr->FileOds.Fab, 0, 0);

   if (ERROR_REPORTED (rqptr))
   {
      /*********************************************/
      /* an error has occured, delete created file */
      /*********************************************/

      /* use SYSPRV to ensure erasure of file */
      sys$setprv (1, &SysPrvMask, 0, 0);

      tkptr->FileOds.Fab.fab$l_fop = FAB$M_NAM;
      status = sys$erase (&tkptr->FileOds.Fab, 0, 0);
      if (WATCHMOD (rqptr, WATCH_MOD_PUT))
         WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "sys$erase() !&S", status);

      sys$setprv (0, &SysPrvMask, 0, 0);

      return;
   }
   else
   {
      if (rqptr->WebDavTaskPtr)
      {
         /**************/
         /* WebDAV PUT */
         /**************/

         if (tkptr->SysCreateStatus == RMS$_CREATED)
         {
            DavWebHref (rqptr, rqptr->ParseOds.ExpFileName, 0);
            DavWebResponse201 (rqptr);
         }
         else
            DavWebResponse (rqptr, 204, 0, "existing, overwrite", FI_LI);

         return;
      }

      /**************/
      /* non-WebDAV */
      /**************/

      if (Config.cfMisc.PutVersionLimit)
      {
         /* use SYSPRV to ensure modification of file */
         sys$setprv (1, &SysPrvMask, 0, 0);

         status = OdsFileAcpModify (&tkptr->FileOds, NULL,
                                    &Config.cfMisc.PutVersionLimit,
                                    NULL, rqptr);

         sys$setprv (0, &SysPrvMask, 0, 0);

         if (VMSnok (status))
         {
            /* as an error has occured terminate the PUT processing */
            rqptr->rqResponse.ErrorTextPtr = rqptr->rqHeader.PathInfoPtr;
            rqptr->rqResponse.ErrorOtherTextPtr = tkptr->FileName;
            ErrorVmsStatus (rqptr, status, FI_LI);
            return;
         }
      }

      if (tkptr->PreviewOnly)
      {
         /****************/
         /* preview only */
         /****************/

         /* redirect to temporary file */
         int  size;
         cptr = MapVmsPath (tkptr->FileName, rqptr);
         size = strlen(cptr) + 256;
         sptr = ResponseLocation (rqptr, NULL, size);
         /* the leading space indicates that it's a change of HTTP method */
         FaoToBuffer (sptr, size, NULL, " GET !&%AZ", cptr);
         return;
      }

      if (rqptr->rqResponse.LocationPtr)
      {
         /********************/
         /* success redirect */
         /********************/

         return;
      }

      if (ConfigSameContentType (tkptr->FileContentTypePtr, "text/", 5) ||
          ConfigSameContentType (tkptr->FileContentTypePtr,
                                 "application/x-www-form-urlencoded", -1))
         tptr = MsgFor(rqptr,MSG_GENERAL_DOCUMENT);
      else
         tptr = MsgFor(rqptr,MSG_GENERAL_FILE);

      FormatProtection (tkptr->ProtectionMask, ProtectionString);

      cptr = MapVmsPath (tkptr->FileName, rqptr);

      /* ReportSuccess() will convert this 201 back to a 200 */
      rqptr->rqResponse.HttpStatus = 201;
      rqptr->rqResponse.PreExpired = PRE_EXPIRE_PUT;
      ReportSuccess (rqptr,
"!AZ &nbsp;<a href=\"!&%AZ\">!&;AZ</a>&nbsp; !AZ (!UL bytes)&nbsp; (!AZ)",
                     tptr, cptr, cptr,
                     tkptr->FileOds.Nam_fnb & NAM$M_LOWVER ?
                        MsgFor(rqptr,MSG_PUT_SUPERCEDED) :
                        MsgFor(rqptr,MSG_PUT_CREATED),
                     tkptr->FileSizeBytes, ProtectionString);

      return;
   }
}

/*****************************************************************************/
/*
Create a directory!
*/ 
 
int PutCreateDirectory (REQUEST_STRUCT *rqptr)

{
   static unsigned short  ProtectionEnable = 0xffff, /* alter all S,O,G,W */ 
                          EnsureSystemAccessMask = 0xfff0; /* S:RWED */
   static $DESCRIPTOR (DirectoryDsc, "");

   int  status;
   unsigned short  Length;
   char  *dptr;
   PUT_TASK  *tkptr;

   /*********/
   /* begin */
   /*********/

   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "PutCreateDirectory()");

   tkptr = rqptr->PutTaskPtr;

   dptr = rqptr->ParseOds.NamDevicePtr;

   DirectoryDsc.dsc$a_pointer = dptr;
   DirectoryDsc.dsc$w_length = strlen(dptr);
   tkptr->ProtectionMask &= EnsureSystemAccessMask;

   AuthAccessEnable (rqptr, DirectoryDsc.dsc$a_pointer, AUTH_ACCESS_WRITE);

   status = lib$create_dir (&DirectoryDsc, 0, &ProtectionEnable,
                            &tkptr->ProtectionMask, 0, 0);
   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "lib$create_dir() !&S", status);

   AuthAccessEnable (rqptr, 0, 0);

   if (status == SS$_CREATED)
   {
      /* ReportSuccess() will convert this 201 back to a 200 */
      rqptr->rqResponse.HttpStatus = 201;
      rqptr->rqResponse.PreExpired = PRE_EXPIRE_PUT;
      ReportSuccess (rqptr,
"!AZ&nbsp; <a href=\"!&%AZ\">!&;&_AZ</a>&nbsp; !AZ",
                     MsgFor(rqptr,MSG_GENERAL_DIRECTORY),
                     rqptr->rqHeader.PathInfoPtr,
                     rqptr->rqHeader.PathInfoPtr,
                     MsgFor(rqptr,MSG_PUT_CREATED));
   }
   else
   if (status == SS$_NORMAL)
   {
      rqptr->rqResponse.HttpStatus = 409;
      ErrorGeneral (rqptr, MsgFor(rqptr,MSG_PUT_DIR_EXISTS), FI_LI);
   }
   else
   {
      rqptr->rqResponse.ErrorTextPtr = rqptr->rqHeader.PathInfoPtr;
      rqptr->rqResponse.ErrorOtherTextPtr = dptr;
      ErrorVmsStatus (rqptr, status, FI_LI);
   }
   return (status);
}

/*****************************************************************************/
/*
Deletes both files and directories.
*/ 
 
PutDelete (REQUEST_STRUCT *rqptr)

{
   BOOL  DeletingDirectory;
   int  status,
        EraseCount,
        FileCount,
        DirFileNameLength;
   char  *cptr, *sptr, *zptr;
   char  DirFileName [ODS_MAX_FILE_NAME_LENGTH+1],
         SearchFileName [ODS_MAX_FILE_NAME_LENGTH+1];
   PUT_TASK  *tkptr;
   ODS_STRUCT  DeleteOds,
               SearchOds;

   /*********/
   /* begin */
   /*********/

   if (WATCHMOD (rqptr, WATCH_MOD_PUT))
      WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "PutDelete()");

   tkptr = rqptr->PutTaskPtr;

   /* bit of a shonky, remove trailing ';*' from path */
   cptr = rqptr->rqHeader.PathInfoPtr + rqptr->rqHeader.PathInfoLength;
   if (*(cptr-1) == '*') cptr--;
   if (*(cptr-1) == ';') cptr--;
   *cptr = '\0';
   rqptr->rqHeader.PathInfoLength = cptr - rqptr->rqHeader.PathInfoPtr;

   DeletingDirectory = !rqptr->ParseOds.NamNameLength &&
                       !rqptr->ParseOds.NamTypeLength;

   if (DeletingDirectory)
   {
      OdsNameOfDirectoryFile (rqptr->ParseOds.ExpFileName,
                              rqptr->ParseOds.ExpFileNameLength,
                              DirFileName, &DirFileNameLength);

      AuthAccessEnable (rqptr, DirFileName, AUTH_ACCESS_READ);

      OdsStructInit (&SearchOds, true);
      OdsParse (&SearchOds, DirFileName,
                            DirFileNameLength,
                            NULL, 0, 0, NULL, rqptr);
      AuthAccessEnable (rqptr, 0, 0);
   }
   else
   {
      AuthAccessEnable (rqptr, rqptr->ParseOds.ExpFileName, AUTH_ACCESS_READ);

      OdsParse (&SearchOds, rqptr->ParseOds.ExpFileName,
                            rqptr->ParseOds.ExpFileNameLength,
                            ";*", 2, 0, NULL, rqptr);

      AuthAccessEnable (rqptr, 0, 0);
   }

   if (VMSnok (status = SearchOds.Fab.fab$l_sts))
   {
      rqptr->rqResponse.ErrorTextPtr = rqptr->rqHeader.PathInfoPtr;
      rqptr->rqResponse.ErrorOtherTextPtr = rqptr->ParseOds.ExpFileName;
      ErrorVmsStatus (rqptr, status, FI_LI);
      return (status);
   }

   /*******************/
   /* delete the file */
   /*******************/

   FileCount = 0;
   for (;;)
   {
      /* SYSPRV to ensure search (provided protection is S:RWED) */
      sys$setprv (1, &SysPrvMask, 0, 0);
      status = OdsSearch (&SearchOds, NULL, rqptr);
      sys$setprv (0, &SysPrvMask, 0, 0);

      if (VMSnok (status)) break;

      FileCount++;

      OdsStructInit (&DeleteOds, true);
      OdsParse (&DeleteOds,
                SearchOds.ResFileName, SearchOds.ResFileNameLength,
                NULL, 0,
                NAM$M_SYNCHK, NULL, rqptr);

      if (VMSnok (status = SearchOds.Fab.fab$l_sts))
      {
         rqptr->rqResponse.ErrorTextPtr = rqptr->rqHeader.PathInfoPtr;
         rqptr->rqResponse.ErrorOtherTextPtr = SearchOds.ExpFileName;
         ErrorVmsStatus (rqptr, status, FI_LI);
         return (status);
      }

      DeleteOds.NamVersionPtr[0] = '\0'; 
      if (WATCHMOD (rqptr, WATCH_MOD_PUT))
         WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT, "!&Z", DeleteOds.ExpFileName);

      if (rqptr->WebDavTaskPtr)
         AuthAccessEnable (rqptr, DeleteOds.ExpFileName, AUTH_ACCESS_WRITE);
      else
         /* use SYSPRV to ensure appropriate access/ownership */
         sys$setprv (1, &SysPrvMask, 0, 0);

      EraseCount = 0;
      while (VMSok (status = sys$erase (&DeleteOds.Fab, 0, 0)))
      {
         EraseCount++;
         if (WATCHMOD (rqptr, WATCH_MOD_PUT))
            WatchThis (WATCHITM(rqptr), WATCH_MOD_PUT,
                       "!UL sys$erase() !&S", EraseCount, status);
      }
      if (status == RMS$_FNF && EraseCount) status = SS$_NORMAL;

      if (rqptr->WebDavTaskPtr)
         AuthAccessEnable (rqptr, 0, 0);
      else
         sys$setprv (0, &SysPrvMask, 0, 0);

      if (VMSnok (status) && DeleteOds.Fab.fab$l_stv)
         status = DeleteOds.Fab.fab$l_stv;

      OdsParseRelease (&DeleteOds);

      if (VMSnok (status))
      {
         rqptr->rqResponse.ErrorTextPtr = rqptr->rqHeader.PathInfoPtr;
         rqptr->rqResponse.ErrorOtherTextPtr = SearchOds.ExpFileName;
         ErrorVmsStatus (rqptr, status, FI_LI);
         return (status);
      }
   }

   if (status == RMS$_NMF) status = SS$_NORMAL;

   OdsParseRelease (&SearchOds);

   if (VMSok (status))
   {
      rqptr->rqResponse.PreExpired = PRE_EXPIRE_PUT;
      if (DeletingDirectory)
         sptr = MsgFor(rqptr,MSG_GENERAL_DIRECTORY);
      else
         sptr = MsgFor(rqptr,MSG_GENERAL_FILE);
      ReportSuccess (rqptr, "!AZ&nbsp; !&;&_AZ&nbsp; !AZ",
                     sptr, rqptr->rqHeader.PathInfoPtr,
                     MsgFor(rqptr,MSG_PUT_DELETED)); 
      return (status);
   }
   else
   {
      if (DeletingDirectory && status == RMS$_FNF) status = RMS$_DNF;
      rqptr->rqResponse.ErrorTextPtr = rqptr->rqHeader.PathInfoPtr;
      rqptr->rqResponse.ErrorOtherTextPtr = SearchOds.ExpFileName;
      ErrorVmsStatus (rqptr, status, FI_LI);
      return (status);
   }
}

/*****************************************************************************/