diff options
Diffstat (limited to 'src/ca/scamimemap.cpp')
| -rw-r--r-- | src/ca/scamimemap.cpp | 200 |
1 files changed, 200 insertions, 0 deletions
diff --git a/src/ca/scamimemap.cpp b/src/ca/scamimemap.cpp new file mode 100644 index 00000000..8afe99f9 --- /dev/null +++ b/src/ca/scamimemap.cpp | |||
| @@ -0,0 +1,200 @@ | |||
| 1 | // Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. | ||
| 2 | |||
| 3 | #include "precomp.h" | ||
| 4 | |||
| 5 | enum eMimeMapQuery { mmqMimeMap = 1, mmqParentType, mmqParentValue, | ||
| 6 | mmqMimeType, mmqExtension}; | ||
| 7 | |||
| 8 | // prototypes | ||
| 9 | static HRESULT AddMimeMapToList(SCA_MIMEMAP** ppsmmList); | ||
| 10 | |||
| 11 | |||
| 12 | void ScaMimeMapFreeList(SCA_MIMEMAP* psmmList) | ||
| 13 | { | ||
| 14 | SCA_MIMEMAP* psmmDelete = psmmList; | ||
| 15 | while (psmmList) | ||
| 16 | { | ||
| 17 | psmmDelete = psmmList; | ||
| 18 | psmmList = psmmList->psmmNext; | ||
| 19 | |||
| 20 | MemFree(psmmDelete); | ||
| 21 | } | ||
| 22 | } | ||
| 23 | |||
| 24 | |||
| 25 | HRESULT __stdcall ScaMimeMapRead( | ||
| 26 | SCA_MIMEMAP** ppsmmList, | ||
| 27 | __inout LPWSTR *ppwzCustomActionData | ||
| 28 | ) | ||
| 29 | { | ||
| 30 | HRESULT hr = S_OK; | ||
| 31 | MSIHANDLE hRec; | ||
| 32 | LPWSTR pwzData = NULL; | ||
| 33 | SCA_MIMEMAP* psmm; | ||
| 34 | WCA_WRAPQUERY_HANDLE hWrapQuery = NULL; | ||
| 35 | |||
| 36 | hr = WcaBeginUnwrapQuery(&hWrapQuery, ppwzCustomActionData); | ||
| 37 | ExitOnFailure(hr, "Failed to unwrap query for ScaMimeMapRead"); | ||
| 38 | |||
| 39 | if (0 == WcaGetQueryRecords(hWrapQuery)) | ||
| 40 | { | ||
| 41 | WcaLog(LOGMSG_VERBOSE, "Skipping ScaMimeMapRead() - required table not present"); | ||
| 42 | ExitFunction1(hr = S_FALSE); | ||
| 43 | } | ||
| 44 | |||
| 45 | // loop through all the mimemappings | ||
| 46 | while (S_OK == (hr = WcaFetchWrappedRecord(hWrapQuery, &hRec))) | ||
| 47 | { | ||
| 48 | hr = AddMimeMapToList(ppsmmList); | ||
| 49 | ExitOnFailure(hr, "failed to add mime map to list"); | ||
| 50 | |||
| 51 | psmm = *ppsmmList; | ||
| 52 | |||
| 53 | hr = WcaGetRecordString(hRec, mmqMimeMap, &pwzData); | ||
| 54 | ExitOnFailure(hr, "Failed to get MimeMap.MimeMap"); | ||
| 55 | hr = ::StringCchCopyW(psmm->wzMimeMap, countof(psmm->wzMimeMap), pwzData); | ||
| 56 | ExitOnFailure(hr, "Failed to copy mimemap string to mimemap object"); | ||
| 57 | |||
| 58 | hr = WcaGetRecordInteger(hRec, mmqParentType, &psmm->iParentType); | ||
| 59 | ExitOnFailure(hr, "Failed to get MimeMap.iParentType"); | ||
| 60 | |||
| 61 | hr = WcaGetRecordString(hRec, mmqParentValue, &pwzData); | ||
| 62 | ExitOnFailure(hr, "Failed to get MimeMap.ParentValue"); | ||
| 63 | hr = ::StringCchCopyW(psmm->wzParentValue, countof(psmm->wzParentValue), pwzData); | ||
| 64 | ExitOnFailure(hr, "Failed to copy parent value string to mimemap object"); | ||
| 65 | |||
| 66 | hr = WcaGetRecordString(hRec, mmqExtension, &pwzData); | ||
| 67 | ExitOnFailure(hr, "Failed to get MimeMap.Extension"); | ||
| 68 | hr = ::StringCchCopyW(psmm->wzExtension, countof(psmm->wzExtension), pwzData); | ||
| 69 | ExitOnFailure(hr, "Failed to copy extension string to mimemap object"); | ||
| 70 | |||
| 71 | hr = WcaGetRecordString(hRec, mmqMimeType, &pwzData); | ||
| 72 | ExitOnFailure(hr, "Failed to get MimeMap.MimeType"); | ||
| 73 | hr = ::StringCchCopyW(psmm->wzMimeType, countof(psmm->wzMimeType), pwzData); | ||
| 74 | ExitOnFailure(hr, "Failed to copy mimetype string to mimemap object"); | ||
| 75 | } | ||
| 76 | |||
| 77 | if (E_NOMOREITEMS == hr) | ||
| 78 | hr = S_OK; | ||
| 79 | ExitOnFailure(hr, "Failure while processing mimemappings"); | ||
| 80 | |||
| 81 | LExit: | ||
| 82 | WcaFinishUnwrapQuery(hWrapQuery); | ||
| 83 | |||
| 84 | ReleaseStr(pwzData); | ||
| 85 | |||
| 86 | return hr; | ||
| 87 | } | ||
| 88 | |||
| 89 | |||
| 90 | HRESULT ScaGetMimeMap(int iParentType, LPCWSTR wzParentValue, SCA_MIMEMAP **ppsmmList, SCA_MIMEMAP **ppsmmOut) | ||
| 91 | { | ||
| 92 | HRESULT hr = S_OK; | ||
| 93 | SCA_MIMEMAP* psmmAdd = NULL; | ||
| 94 | SCA_MIMEMAP* psmmLast = NULL; | ||
| 95 | |||
| 96 | *ppsmmOut = NULL; | ||
| 97 | |||
| 98 | if (!*ppsmmList) | ||
| 99 | return hr; | ||
| 100 | |||
| 101 | SCA_MIMEMAP* psmm = *ppsmmList; | ||
| 102 | while (psmm) | ||
| 103 | { | ||
| 104 | if (iParentType == psmm->iParentType && 0 == lstrcmpW(wzParentValue, psmm->wzParentValue)) | ||
| 105 | { | ||
| 106 | // Found a match, take this one out of the list and add it to the matched out list | ||
| 107 | psmmAdd = psmm; | ||
| 108 | |||
| 109 | if (psmmLast) | ||
| 110 | { | ||
| 111 | // If we're not at the beginning of the list tell the last node about it's new next (since we're taking away it's current next) | ||
| 112 | psmmLast->psmmNext = psmmAdd->psmmNext; | ||
| 113 | } | ||
| 114 | else | ||
| 115 | { | ||
| 116 | // If we are at the beginning (no psmmLast) update the beginning (since we're taking it) | ||
| 117 | *ppsmmList = psmm->psmmNext; | ||
| 118 | } | ||
| 119 | psmm = psmm->psmmNext; // move on | ||
| 120 | |||
| 121 | // Add the one we've removed to the beginning of the out list | ||
| 122 | psmmAdd->psmmNext = *ppsmmOut; | ||
| 123 | *ppsmmOut = psmmAdd; | ||
| 124 | } | ||
| 125 | else | ||
| 126 | { | ||
| 127 | psmmLast = psmm; // remember the last we that didn't match | ||
| 128 | psmm = psmm->psmmNext; // move on | ||
| 129 | } | ||
| 130 | } | ||
| 131 | |||
| 132 | return hr; | ||
| 133 | } | ||
| 134 | |||
| 135 | HRESULT ScaMimeMapCheckList(SCA_MIMEMAP* psmmList) | ||
| 136 | { | ||
| 137 | if (!psmmList) | ||
| 138 | return S_OK; | ||
| 139 | |||
| 140 | while (psmmList) | ||
| 141 | { | ||
| 142 | WcaLog(LOGMSG_STANDARD, "MimeMapping of %ls with ParentType=%d and ParentValue=%ls not used!", psmmList->wzMimeMap, psmmList->iParentType, psmmList->wzParentValue); | ||
| 143 | psmmList = psmmList->psmmNext; | ||
| 144 | } | ||
| 145 | |||
| 146 | return E_FAIL; | ||
| 147 | } | ||
| 148 | |||
| 149 | |||
| 150 | HRESULT ScaWriteMimeMap(IMSAdminBase* piMetabase, LPCWSTR wzRootOfWeb, | ||
| 151 | SCA_MIMEMAP* psmmList) | ||
| 152 | { | ||
| 153 | HRESULT hr = S_OK; | ||
| 154 | |||
| 155 | WCHAR wzMimeMap[8192]; | ||
| 156 | WCHAR *pwzNext = wzMimeMap; | ||
| 157 | const WCHAR *pwzMac = wzMimeMap + countof(wzMimeMap); // used to properly create the MULTI_SZ | ||
| 158 | |||
| 159 | // fill the MULTI_SZ wzMimeMap buffer for the MimeMap attribute | ||
| 160 | ::ZeroMemory(wzMimeMap, sizeof(wzMimeMap)); | ||
| 161 | |||
| 162 | for (SCA_MIMEMAP* psmm = psmmList; psmm; psmm = psmm->psmmNext) | ||
| 163 | { | ||
| 164 | hr = ::StringCchPrintfW(pwzNext, max(0, pwzMac - pwzNext), L"%s,%s", psmm->wzExtension, psmm->wzMimeType); | ||
| 165 | ExitOnFailure(hr, "Failed to set MimeMap string"); | ||
| 166 | |||
| 167 | pwzNext += lstrlenW(pwzNext) + 1; // reserve space for null | ||
| 168 | Assert(pwzNext <= pwzMac); | ||
| 169 | } | ||
| 170 | |||
| 171 | if (pwzNext != wzMimeMap) | ||
| 172 | { | ||
| 173 | // now write the CustomErrors to the metabase | ||
| 174 | hr = ScaWriteMetabaseValue(piMetabase, wzRootOfWeb, NULL, MD_MIME_MAP, METADATA_INHERIT, IIS_MD_UT_FILE, MULTISZ_METADATA, wzMimeMap); | ||
| 175 | ExitOnFailure(hr, "Failed to write MimeMap"); | ||
| 176 | } | ||
| 177 | else | ||
| 178 | { | ||
| 179 | WcaLog(LOGMSG_VERBOSE, "Skipping ScaWriteMimeMap() - no mappings found."); | ||
| 180 | ExitFunction1(hr = S_FALSE); | ||
| 181 | } | ||
| 182 | |||
| 183 | LExit: | ||
| 184 | return hr; | ||
| 185 | } | ||
| 186 | |||
| 187 | |||
| 188 | static HRESULT AddMimeMapToList(SCA_MIMEMAP** ppsmmList) | ||
| 189 | { | ||
| 190 | HRESULT hr = S_OK; | ||
| 191 | |||
| 192 | SCA_MIMEMAP* psmm = static_cast<SCA_MIMEMAP*>(MemAlloc(sizeof(SCA_MIMEMAP), TRUE)); | ||
| 193 | ExitOnNull(psmm, hr, E_OUTOFMEMORY, "failed to allocate memory for new mime map list element"); | ||
| 194 | |||
| 195 | psmm->psmmNext = *ppsmmList; | ||
| 196 | *ppsmmList = psmm; | ||
| 197 | |||
| 198 | LExit: | ||
| 199 | return hr; | ||
| 200 | } | ||
