1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski #include "xml_cdff.hxx"
25*b1cdbd2cSJim Jagielski 
26*b1cdbd2cSJim Jagielski #include <string.h>
27*b1cdbd2cSJim Jagielski #include <tools/stream.hxx>
28*b1cdbd2cSJim Jagielski #include "xml_cdim.hxx"
29*b1cdbd2cSJim Jagielski #include <ctype.h>
30*b1cdbd2cSJim Jagielski 
31*b1cdbd2cSJim Jagielski 
32*b1cdbd2cSJim Jagielski typedef ComponentDescriptionImpl::ValueList CdiValueList;
33*b1cdbd2cSJim Jagielski 
34*b1cdbd2cSJim Jagielski class dyn_buffer
35*b1cdbd2cSJim Jagielski {
36*b1cdbd2cSJim Jagielski   public:
dyn_buffer()37*b1cdbd2cSJim Jagielski 						dyn_buffer() 					: s(0) {}
~dyn_buffer()38*b1cdbd2cSJim Jagielski 						~dyn_buffer()                   { if (s) delete [] s; }
39*b1cdbd2cSJim Jagielski 						operator const char *() const 	{ return s; }
operator ->()40*b1cdbd2cSJim Jagielski 	char *              operator->()                    { return s; }
operator [](INT32 ix)41*b1cdbd2cSJim Jagielski 	char &              operator[](
42*b1cdbd2cSJim Jagielski 							INT32 ix )                  { return s[ix]; }
SetSize(INT32 i_size)43*b1cdbd2cSJim Jagielski 	void               	SetSize(
44*b1cdbd2cSJim Jagielski 							INT32 i_size )              { if (s) delete [] s; s = new char [i_size]; }
45*b1cdbd2cSJim Jagielski   private:
46*b1cdbd2cSJim Jagielski 	char * s;
47*b1cdbd2cSJim Jagielski };
48*b1cdbd2cSJim Jagielski 
49*b1cdbd2cSJim Jagielski 
50*b1cdbd2cSJim Jagielski inline BOOL
LoadXmlFile(dyn_buffer & o_rBuffer,const UniString & i_sXmlFilePath)51*b1cdbd2cSJim Jagielski LoadXmlFile( dyn_buffer & 		o_rBuffer,
52*b1cdbd2cSJim Jagielski 			 const UniString &	i_sXmlFilePath )
53*b1cdbd2cSJim Jagielski {
54*b1cdbd2cSJim Jagielski 	BOOL ret = TRUE;
55*b1cdbd2cSJim Jagielski 	SvFileStream aXmlFile;
56*b1cdbd2cSJim Jagielski 
57*b1cdbd2cSJim Jagielski 	aXmlFile.Open(i_sXmlFilePath, STREAM_READ);
58*b1cdbd2cSJim Jagielski 	if (aXmlFile.GetErrorCode() != FSYS_ERR_OK)
59*b1cdbd2cSJim Jagielski 		ret = FALSE;
60*b1cdbd2cSJim Jagielski 	if (ret)
61*b1cdbd2cSJim Jagielski 	{
62*b1cdbd2cSJim Jagielski 		aXmlFile.Seek(STREAM_SEEK_TO_END);
63*b1cdbd2cSJim Jagielski 		INT32 nBufferSize = aXmlFile.Tell();
64*b1cdbd2cSJim Jagielski 		o_rBuffer.SetSize(nBufferSize + 1);
65*b1cdbd2cSJim Jagielski 		o_rBuffer[nBufferSize] = '\0';
66*b1cdbd2cSJim Jagielski 		aXmlFile.Seek(0);
67*b1cdbd2cSJim Jagielski 		if (aXmlFile.Read(o_rBuffer.operator->(), nBufferSize) == 0)
68*b1cdbd2cSJim Jagielski 			ret = FALSE;
69*b1cdbd2cSJim Jagielski 	}
70*b1cdbd2cSJim Jagielski 
71*b1cdbd2cSJim Jagielski 	aXmlFile.Close();
72*b1cdbd2cSJim Jagielski 	return ret;
73*b1cdbd2cSJim Jagielski }
74*b1cdbd2cSJim Jagielski 
75*b1cdbd2cSJim Jagielski 
76*b1cdbd2cSJim Jagielski 
CompDescrsFromAnXmlFile()77*b1cdbd2cSJim Jagielski CompDescrsFromAnXmlFile::CompDescrsFromAnXmlFile()
78*b1cdbd2cSJim Jagielski 	:	dpDescriptions(new std::vector< ComponentDescriptionImpl* >),
79*b1cdbd2cSJim Jagielski 		eStatus(not_yet_parsed)
80*b1cdbd2cSJim Jagielski {
81*b1cdbd2cSJim Jagielski 	dpDescriptions->reserve(3);
82*b1cdbd2cSJim Jagielski }
83*b1cdbd2cSJim Jagielski 
~CompDescrsFromAnXmlFile()84*b1cdbd2cSJim Jagielski CompDescrsFromAnXmlFile::~CompDescrsFromAnXmlFile()
85*b1cdbd2cSJim Jagielski {
86*b1cdbd2cSJim Jagielski 	Empty();
87*b1cdbd2cSJim Jagielski 	delete dpDescriptions;
88*b1cdbd2cSJim Jagielski }
89*b1cdbd2cSJim Jagielski 
90*b1cdbd2cSJim Jagielski 
91*b1cdbd2cSJim Jagielski BOOL
Parse(const UniString & i_sXmlFilePath)92*b1cdbd2cSJim Jagielski CompDescrsFromAnXmlFile::Parse( const UniString & i_sXmlFilePath )
93*b1cdbd2cSJim Jagielski {
94*b1cdbd2cSJim Jagielski 	dyn_buffer  	dpBuffer;
95*b1cdbd2cSJim Jagielski 
96*b1cdbd2cSJim Jagielski 	if (! LoadXmlFile(dpBuffer,i_sXmlFilePath) )
97*b1cdbd2cSJim Jagielski 	{
98*b1cdbd2cSJim Jagielski 		eStatus = cant_read_file;
99*b1cdbd2cSJim Jagielski 		return FALSE;
100*b1cdbd2cSJim Jagielski 	}
101*b1cdbd2cSJim Jagielski 
102*b1cdbd2cSJim Jagielski 	const char * 	pTokenStart = 0;
103*b1cdbd2cSJim Jagielski 	const char *    pBufferPosition = dpBuffer;
104*b1cdbd2cSJim Jagielski 	INT32           nTokenLength = 0;
105*b1cdbd2cSJim Jagielski 	BOOL			bWithinElement = FALSE;
106*b1cdbd2cSJim Jagielski 
107*b1cdbd2cSJim Jagielski 	CdiValueList *	pCurTagData = 0;
108*b1cdbd2cSJim Jagielski 	ByteString      sStatusValue;	// Used only if a <Status ...> tag is found.
109*b1cdbd2cSJim Jagielski 
110*b1cdbd2cSJim Jagielski 
111*b1cdbd2cSJim Jagielski 	for ( ComponentDescriptionImpl::ParseUntilStartOfDescription(pBufferPosition);
112*b1cdbd2cSJim Jagielski 		  pBufferPosition != 0;
113*b1cdbd2cSJim Jagielski 		  ComponentDescriptionImpl::ParseUntilStartOfDescription(pBufferPosition) )
114*b1cdbd2cSJim Jagielski 	{
115*b1cdbd2cSJim Jagielski 		ComponentDescriptionImpl * pCurCD = 0;
116*b1cdbd2cSJim Jagielski 		pCurCD = new ComponentDescriptionImpl;
117*b1cdbd2cSJim Jagielski 		dpDescriptions->push_back(pCurCD);
118*b1cdbd2cSJim Jagielski 
119*b1cdbd2cSJim Jagielski 		for ( ; *pBufferPosition != '\0' && pCurCD != 0; )
120*b1cdbd2cSJim Jagielski 		{
121*b1cdbd2cSJim Jagielski 			switch (*pBufferPosition)
122*b1cdbd2cSJim Jagielski 			{
123*b1cdbd2cSJim Jagielski 				case '<' :
124*b1cdbd2cSJim Jagielski 						if (! bWithinElement)
125*b1cdbd2cSJim Jagielski 						{
126*b1cdbd2cSJim Jagielski 							pCurTagData = pCurCD->GetBeginTag(sStatusValue, pBufferPosition);
127*b1cdbd2cSJim Jagielski 							if (pCurTagData != 0)
128*b1cdbd2cSJim Jagielski 							{
129*b1cdbd2cSJim Jagielski 								if (sStatusValue.Len () == 0)
130*b1cdbd2cSJim Jagielski 								{
131*b1cdbd2cSJim Jagielski 									// Start new token:
132*b1cdbd2cSJim Jagielski 									pTokenStart = pBufferPosition;
133*b1cdbd2cSJim Jagielski 									nTokenLength = 0;
134*b1cdbd2cSJim Jagielski 									bWithinElement = TRUE;;
135*b1cdbd2cSJim Jagielski 								}
136*b1cdbd2cSJim Jagielski 								else
137*b1cdbd2cSJim Jagielski 								{
138*b1cdbd2cSJim Jagielski 									// Status tag is already parsed:
139*b1cdbd2cSJim Jagielski 									pCurTagData->push_back(sStatusValue);
140*b1cdbd2cSJim Jagielski 								}	// endif (sStatusValue.Length () == 0)
141*b1cdbd2cSJim Jagielski 							}
142*b1cdbd2cSJim Jagielski 							else if ( ComponentDescriptionImpl::CheckEndOfDescription(pBufferPosition) )
143*b1cdbd2cSJim Jagielski 							{
144*b1cdbd2cSJim Jagielski 								pBufferPosition += ComponentDescriptionImpl::DescriptionEndTagSize();
145*b1cdbd2cSJim Jagielski 								pCurCD = 0;
146*b1cdbd2cSJim Jagielski 							}
147*b1cdbd2cSJim Jagielski 							else
148*b1cdbd2cSJim Jagielski 							{
149*b1cdbd2cSJim Jagielski 								eStatus = inconsistent_file;
150*b1cdbd2cSJim Jagielski 								return FALSE;
151*b1cdbd2cSJim Jagielski 							}	// endif (pCurTagData != 0) elseif() else
152*b1cdbd2cSJim Jagielski 						}
153*b1cdbd2cSJim Jagielski 						else if ( pCurTagData->MatchesEndTag(pBufferPosition) )
154*b1cdbd2cSJim Jagielski 						{
155*b1cdbd2cSJim Jagielski 							// Finish token:
156*b1cdbd2cSJim Jagielski 							pBufferPosition += pCurTagData->EndTagLength();
157*b1cdbd2cSJim Jagielski 							bWithinElement = FALSE;
158*b1cdbd2cSJim Jagielski 
159*b1cdbd2cSJim Jagielski 								// Remove leading and trailing spaces:
160*b1cdbd2cSJim Jagielski 							while ( isspace(*pTokenStart) )
161*b1cdbd2cSJim Jagielski 							{
162*b1cdbd2cSJim Jagielski 							   pTokenStart++;
163*b1cdbd2cSJim Jagielski 							   nTokenLength--;
164*b1cdbd2cSJim Jagielski 							}
165*b1cdbd2cSJim Jagielski 							while ( nTokenLength > 0
166*b1cdbd2cSJim Jagielski 									&& isspace(pTokenStart[nTokenLength-1]) )
167*b1cdbd2cSJim Jagielski 							{
168*b1cdbd2cSJim Jagielski 								nTokenLength--;
169*b1cdbd2cSJim Jagielski 							}
170*b1cdbd2cSJim Jagielski 								// Add token to tag values list.
171*b1cdbd2cSJim Jagielski 							pCurTagData->push_back(ByteString(pTokenStart,nTokenLength));
172*b1cdbd2cSJim Jagielski 						}
173*b1cdbd2cSJim Jagielski 						else
174*b1cdbd2cSJim Jagielski 						{
175*b1cdbd2cSJim Jagielski 							nTokenLength++;
176*b1cdbd2cSJim Jagielski 							++pBufferPosition;
177*b1cdbd2cSJim Jagielski 						}	// endif (!bWithinElement) else if () else
178*b1cdbd2cSJim Jagielski 					   break;
179*b1cdbd2cSJim Jagielski 				default:
180*b1cdbd2cSJim Jagielski 						if (bWithinElement)
181*b1cdbd2cSJim Jagielski 						{
182*b1cdbd2cSJim Jagielski 							++nTokenLength;
183*b1cdbd2cSJim Jagielski 						}
184*b1cdbd2cSJim Jagielski 						++pBufferPosition;
185*b1cdbd2cSJim Jagielski 			}	// end switch
186*b1cdbd2cSJim Jagielski 		} 	// end for
187*b1cdbd2cSJim Jagielski 
188*b1cdbd2cSJim Jagielski 		if (bWithinElement)
189*b1cdbd2cSJim Jagielski 		{
190*b1cdbd2cSJim Jagielski 			eStatus = inconsistent_file;
191*b1cdbd2cSJim Jagielski 			return FALSE;
192*b1cdbd2cSJim Jagielski 		}
193*b1cdbd2cSJim Jagielski 	} // end for
194*b1cdbd2cSJim Jagielski 
195*b1cdbd2cSJim Jagielski 	return TRUE;
196*b1cdbd2cSJim Jagielski }
197*b1cdbd2cSJim Jagielski 
198*b1cdbd2cSJim Jagielski INT32
NrOfDescriptions() const199*b1cdbd2cSJim Jagielski CompDescrsFromAnXmlFile::NrOfDescriptions() const
200*b1cdbd2cSJim Jagielski {
201*b1cdbd2cSJim Jagielski 	return dpDescriptions->size();
202*b1cdbd2cSJim Jagielski }
203*b1cdbd2cSJim Jagielski 
204*b1cdbd2cSJim Jagielski const ComponentDescription &
operator [](INT32 i_nIndex) const205*b1cdbd2cSJim Jagielski CompDescrsFromAnXmlFile::operator[](INT32 i_nIndex) const
206*b1cdbd2cSJim Jagielski {
207*b1cdbd2cSJim Jagielski 	static const ComponentDescriptionImpl aNullDescr_;
208*b1cdbd2cSJim Jagielski 	return 0 <= i_nIndex && i_nIndex < dpDescriptions->size()
209*b1cdbd2cSJim Jagielski 				?	*(*dpDescriptions)[i_nIndex]
210*b1cdbd2cSJim Jagielski 				:   aNullDescr_;
211*b1cdbd2cSJim Jagielski }
212*b1cdbd2cSJim Jagielski 
213*b1cdbd2cSJim Jagielski void
Empty()214*b1cdbd2cSJim Jagielski CompDescrsFromAnXmlFile::Empty()
215*b1cdbd2cSJim Jagielski {
216*b1cdbd2cSJim Jagielski 	for ( std::vector< ComponentDescriptionImpl* >::iterator aIter = dpDescriptions->begin();
217*b1cdbd2cSJim Jagielski 		  aIter != dpDescriptions->end();
218*b1cdbd2cSJim Jagielski 		  ++aIter )
219*b1cdbd2cSJim Jagielski 	{
220*b1cdbd2cSJim Jagielski 		delete *aIter;
221*b1cdbd2cSJim Jagielski 	}
222*b1cdbd2cSJim Jagielski 	dpDescriptions->erase( dpDescriptions->begin(),
223*b1cdbd2cSJim Jagielski 						   dpDescriptions->end() );
224*b1cdbd2cSJim Jagielski }
225*b1cdbd2cSJim Jagielski 
226*b1cdbd2cSJim Jagielski 
227*b1cdbd2cSJim Jagielski 
228