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 // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_svl.hxx"
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski #include <stdio.h>
28*b1cdbd2cSJim Jagielski 
29*b1cdbd2cSJim Jagielski #include <com/sun/star/ucb/XSimpleFileAccess.hpp>
30*b1cdbd2cSJim Jagielski #include <com/sun/star/ucb/XCommandEnvironment.hpp>
31*b1cdbd2cSJim Jagielski #include <com/sun/star/ucb/InsertCommandArgument.hpp>
32*b1cdbd2cSJim Jagielski #include <com/sun/star/ucb/NameClashException.hpp>
33*b1cdbd2cSJim Jagielski #include <com/sun/star/io/WrongFormatException.hpp>
34*b1cdbd2cSJim Jagielski 
35*b1cdbd2cSJim Jagielski #include <osl/time.h>
36*b1cdbd2cSJim Jagielski #include <osl/security.hxx>
37*b1cdbd2cSJim Jagielski #include <osl/socket.hxx>
38*b1cdbd2cSJim Jagielski #include <osl/file.hxx>
39*b1cdbd2cSJim Jagielski 
40*b1cdbd2cSJim Jagielski #include <rtl/string.hxx>
41*b1cdbd2cSJim Jagielski #include <rtl/ustring.hxx>
42*b1cdbd2cSJim Jagielski #include <rtl/strbuf.hxx>
43*b1cdbd2cSJim Jagielski #include <rtl/ustrbuf.hxx>
44*b1cdbd2cSJim Jagielski 
45*b1cdbd2cSJim Jagielski #include <comphelper/processfactory.hxx>
46*b1cdbd2cSJim Jagielski 
47*b1cdbd2cSJim Jagielski #include <tools/urlobj.hxx>
48*b1cdbd2cSJim Jagielski #include <unotools/bootstrap.hxx>
49*b1cdbd2cSJim Jagielski 
50*b1cdbd2cSJim Jagielski #include <ucbhelper/content.hxx>
51*b1cdbd2cSJim Jagielski 
52*b1cdbd2cSJim Jagielski #include <unotools/useroptions.hxx>
53*b1cdbd2cSJim Jagielski 
54*b1cdbd2cSJim Jagielski #include <svl/lockfilecommon.hxx>
55*b1cdbd2cSJim Jagielski 
56*b1cdbd2cSJim Jagielski using namespace ::com::sun::star;
57*b1cdbd2cSJim Jagielski 
58*b1cdbd2cSJim Jagielski namespace svt {
59*b1cdbd2cSJim Jagielski 
60*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------
LockFileCommon(const::rtl::OUString & aOrigURL,const uno::Reference<lang::XMultiServiceFactory> & xFactory,const::rtl::OUString & aPrefix)61*b1cdbd2cSJim Jagielski LockFileCommon::LockFileCommon( const ::rtl::OUString& aOrigURL, const uno::Reference< lang::XMultiServiceFactory >& xFactory, const ::rtl::OUString& aPrefix )
62*b1cdbd2cSJim Jagielski : m_xFactory( xFactory )
63*b1cdbd2cSJim Jagielski {
64*b1cdbd2cSJim Jagielski     if ( !m_xFactory.is() )
65*b1cdbd2cSJim Jagielski         m_xFactory = ::comphelper::getProcessServiceFactory();
66*b1cdbd2cSJim Jagielski 
67*b1cdbd2cSJim Jagielski     INetURLObject aDocURL = ResolveLinks( INetURLObject( aOrigURL ) );
68*b1cdbd2cSJim Jagielski 
69*b1cdbd2cSJim Jagielski     ::rtl::OUString aShareURLString = aDocURL.GetPartBeforeLastName();
70*b1cdbd2cSJim Jagielski     aShareURLString += aPrefix;
71*b1cdbd2cSJim Jagielski     aShareURLString += aDocURL.GetName();
72*b1cdbd2cSJim Jagielski     aShareURLString += ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "#" ) );
73*b1cdbd2cSJim Jagielski     m_aURL = INetURLObject( aShareURLString ).GetMainURL( INetURLObject::NO_DECODE );
74*b1cdbd2cSJim Jagielski }
75*b1cdbd2cSJim Jagielski 
76*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------
~LockFileCommon()77*b1cdbd2cSJim Jagielski LockFileCommon::~LockFileCommon()
78*b1cdbd2cSJim Jagielski {
79*b1cdbd2cSJim Jagielski }
80*b1cdbd2cSJim Jagielski 
81*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------
ResolveLinks(const INetURLObject & aDocURL)82*b1cdbd2cSJim Jagielski INetURLObject LockFileCommon::ResolveLinks( const INetURLObject& aDocURL )
83*b1cdbd2cSJim Jagielski {
84*b1cdbd2cSJim Jagielski     if ( aDocURL.HasError() )
85*b1cdbd2cSJim Jagielski         throw lang::IllegalArgumentException();
86*b1cdbd2cSJim Jagielski 
87*b1cdbd2cSJim Jagielski     ::rtl::OUString aURLToCheck = aDocURL.GetMainURL( INetURLObject::NO_DECODE );
88*b1cdbd2cSJim Jagielski 
89*b1cdbd2cSJim Jagielski     sal_Bool bNeedsChecking = sal_True;
90*b1cdbd2cSJim Jagielski     sal_Int32 nMaxLinkCount = 128;
91*b1cdbd2cSJim Jagielski     sal_Int32 nCount = 0;
92*b1cdbd2cSJim Jagielski 
93*b1cdbd2cSJim Jagielski     while( bNeedsChecking )
94*b1cdbd2cSJim Jagielski     {
95*b1cdbd2cSJim Jagielski         bNeedsChecking = sal_False;
96*b1cdbd2cSJim Jagielski 
97*b1cdbd2cSJim Jagielski         // do not allow too deep links
98*b1cdbd2cSJim Jagielski         if ( nCount++ >= nMaxLinkCount )
99*b1cdbd2cSJim Jagielski             throw io::IOException();
100*b1cdbd2cSJim Jagielski 
101*b1cdbd2cSJim Jagielski         // there is currently no UCB functionality to resolve the symbolic links;
102*b1cdbd2cSJim Jagielski         // since the lock files are used only for local file systems the osl functionality is used directly
103*b1cdbd2cSJim Jagielski 
104*b1cdbd2cSJim Jagielski         ::osl::FileStatus aStatus( FileStatusMask_Type | FileStatusMask_LinkTargetURL );
105*b1cdbd2cSJim Jagielski         ::osl::DirectoryItem aItem;
106*b1cdbd2cSJim Jagielski         if ( ::osl::FileBase::E_None == ::osl::DirectoryItem::get( aURLToCheck, aItem )
107*b1cdbd2cSJim Jagielski           && aItem.is() && ::osl::FileBase::E_None == aItem.getFileStatus( aStatus ) )
108*b1cdbd2cSJim Jagielski         {
109*b1cdbd2cSJim Jagielski             if ( aStatus.isValid( FileStatusMask_Type )
110*b1cdbd2cSJim Jagielski               && aStatus.isValid( FileStatusMask_LinkTargetURL )
111*b1cdbd2cSJim Jagielski               && aStatus.getFileType() == ::osl::FileStatus::Link )
112*b1cdbd2cSJim Jagielski             {
113*b1cdbd2cSJim Jagielski                 aURLToCheck = aStatus.getLinkTargetURL();
114*b1cdbd2cSJim Jagielski                 bNeedsChecking = sal_True;
115*b1cdbd2cSJim Jagielski             }
116*b1cdbd2cSJim Jagielski         }
117*b1cdbd2cSJim Jagielski     }
118*b1cdbd2cSJim Jagielski 
119*b1cdbd2cSJim Jagielski     return INetURLObject( aURLToCheck );
120*b1cdbd2cSJim Jagielski }
121*b1cdbd2cSJim Jagielski 
122*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------
ParseList(const uno::Sequence<sal_Int8> & aBuffer)123*b1cdbd2cSJim Jagielski uno::Sequence< uno::Sequence< ::rtl::OUString > > LockFileCommon::ParseList( const uno::Sequence< sal_Int8 >& aBuffer )
124*b1cdbd2cSJim Jagielski {
125*b1cdbd2cSJim Jagielski     sal_Int32 nCurPos = 0;
126*b1cdbd2cSJim Jagielski     sal_Int32 nCurEntry = 0;
127*b1cdbd2cSJim Jagielski     uno::Sequence< uno::Sequence< ::rtl::OUString > > aResult( 10 );
128*b1cdbd2cSJim Jagielski 
129*b1cdbd2cSJim Jagielski     while ( nCurPos < aBuffer.getLength() )
130*b1cdbd2cSJim Jagielski     {
131*b1cdbd2cSJim Jagielski         if ( nCurEntry >= aResult.getLength() )
132*b1cdbd2cSJim Jagielski             aResult.realloc( nCurEntry + 10 );
133*b1cdbd2cSJim Jagielski         aResult[nCurEntry] = ParseEntry( aBuffer, nCurPos );
134*b1cdbd2cSJim Jagielski         nCurEntry++;
135*b1cdbd2cSJim Jagielski     }
136*b1cdbd2cSJim Jagielski 
137*b1cdbd2cSJim Jagielski     aResult.realloc( nCurEntry );
138*b1cdbd2cSJim Jagielski     return aResult;
139*b1cdbd2cSJim Jagielski }
140*b1cdbd2cSJim Jagielski 
141*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------
ParseEntry(const uno::Sequence<sal_Int8> & aBuffer,sal_Int32 & io_nCurPos)142*b1cdbd2cSJim Jagielski uno::Sequence< ::rtl::OUString > LockFileCommon::ParseEntry( const uno::Sequence< sal_Int8 >& aBuffer, sal_Int32& io_nCurPos )
143*b1cdbd2cSJim Jagielski {
144*b1cdbd2cSJim Jagielski     uno::Sequence< ::rtl::OUString > aResult( LOCKFILE_ENTRYSIZE );
145*b1cdbd2cSJim Jagielski 
146*b1cdbd2cSJim Jagielski     for ( int nInd = 0; nInd < LOCKFILE_ENTRYSIZE; nInd++ )
147*b1cdbd2cSJim Jagielski     {
148*b1cdbd2cSJim Jagielski         aResult[nInd] = ParseName( aBuffer, io_nCurPos );
149*b1cdbd2cSJim Jagielski         if ( io_nCurPos >= aBuffer.getLength()
150*b1cdbd2cSJim Jagielski           || ( nInd < LOCKFILE_ENTRYSIZE - 1 && aBuffer[io_nCurPos++] != ',' )
151*b1cdbd2cSJim Jagielski           || ( nInd == LOCKFILE_ENTRYSIZE - 1 && aBuffer[io_nCurPos++] != ';' ) )
152*b1cdbd2cSJim Jagielski             throw io::WrongFormatException();
153*b1cdbd2cSJim Jagielski     }
154*b1cdbd2cSJim Jagielski 
155*b1cdbd2cSJim Jagielski     return aResult;
156*b1cdbd2cSJim Jagielski }
157*b1cdbd2cSJim Jagielski 
158*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------
ParseName(const uno::Sequence<sal_Int8> & aBuffer,sal_Int32 & io_nCurPos)159*b1cdbd2cSJim Jagielski ::rtl::OUString LockFileCommon::ParseName( const uno::Sequence< sal_Int8 >& aBuffer, sal_Int32& io_nCurPos )
160*b1cdbd2cSJim Jagielski {
161*b1cdbd2cSJim Jagielski     ::rtl::OStringBuffer aResult;
162*b1cdbd2cSJim Jagielski     sal_Bool bHaveName = sal_False;
163*b1cdbd2cSJim Jagielski     sal_Bool bEscape = sal_False;
164*b1cdbd2cSJim Jagielski 
165*b1cdbd2cSJim Jagielski     while( !bHaveName )
166*b1cdbd2cSJim Jagielski     {
167*b1cdbd2cSJim Jagielski         if ( io_nCurPos >= aBuffer.getLength() )
168*b1cdbd2cSJim Jagielski             throw io::WrongFormatException();
169*b1cdbd2cSJim Jagielski 
170*b1cdbd2cSJim Jagielski         if ( bEscape )
171*b1cdbd2cSJim Jagielski         {
172*b1cdbd2cSJim Jagielski             if ( aBuffer[io_nCurPos] == ',' || aBuffer[io_nCurPos] == ';' || aBuffer[io_nCurPos] == '\\' )
173*b1cdbd2cSJim Jagielski                 aResult.append( (sal_Char)aBuffer[io_nCurPos] );
174*b1cdbd2cSJim Jagielski             else
175*b1cdbd2cSJim Jagielski                 throw io::WrongFormatException();
176*b1cdbd2cSJim Jagielski 
177*b1cdbd2cSJim Jagielski             bEscape = sal_False;
178*b1cdbd2cSJim Jagielski             io_nCurPos++;
179*b1cdbd2cSJim Jagielski         }
180*b1cdbd2cSJim Jagielski         else if ( aBuffer[io_nCurPos] == ',' || aBuffer[io_nCurPos] == ';' )
181*b1cdbd2cSJim Jagielski             bHaveName = sal_True;
182*b1cdbd2cSJim Jagielski         else
183*b1cdbd2cSJim Jagielski         {
184*b1cdbd2cSJim Jagielski             if ( aBuffer[io_nCurPos] == '\\' )
185*b1cdbd2cSJim Jagielski                 bEscape = sal_True;
186*b1cdbd2cSJim Jagielski             else
187*b1cdbd2cSJim Jagielski                 aResult.append( (sal_Char)aBuffer[io_nCurPos] );
188*b1cdbd2cSJim Jagielski 
189*b1cdbd2cSJim Jagielski             io_nCurPos++;
190*b1cdbd2cSJim Jagielski         }
191*b1cdbd2cSJim Jagielski     }
192*b1cdbd2cSJim Jagielski 
193*b1cdbd2cSJim Jagielski     return ::rtl::OStringToOUString( aResult.makeStringAndClear(), RTL_TEXTENCODING_UTF8 );
194*b1cdbd2cSJim Jagielski }
195*b1cdbd2cSJim Jagielski 
196*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------
EscapeCharacters(const::rtl::OUString & aSource)197*b1cdbd2cSJim Jagielski ::rtl::OUString LockFileCommon::EscapeCharacters( const ::rtl::OUString& aSource )
198*b1cdbd2cSJim Jagielski {
199*b1cdbd2cSJim Jagielski     ::rtl::OUStringBuffer aBuffer;
200*b1cdbd2cSJim Jagielski     const sal_Unicode* pStr = aSource.getStr();
201*b1cdbd2cSJim Jagielski     for ( sal_Int32 nInd = 0; nInd < aSource.getLength() && pStr[nInd] != 0; nInd++ )
202*b1cdbd2cSJim Jagielski     {
203*b1cdbd2cSJim Jagielski         if ( pStr[nInd] == '\\' || pStr[nInd] == ';' || pStr[nInd] == ',' )
204*b1cdbd2cSJim Jagielski             aBuffer.append( (sal_Unicode)'\\' );
205*b1cdbd2cSJim Jagielski         aBuffer.append( pStr[nInd] );
206*b1cdbd2cSJim Jagielski     }
207*b1cdbd2cSJim Jagielski 
208*b1cdbd2cSJim Jagielski     return aBuffer.makeStringAndClear();
209*b1cdbd2cSJim Jagielski }
210*b1cdbd2cSJim Jagielski 
211*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------
GetOOOUserName()212*b1cdbd2cSJim Jagielski ::rtl::OUString LockFileCommon::GetOOOUserName()
213*b1cdbd2cSJim Jagielski {
214*b1cdbd2cSJim Jagielski     SvtUserOptions aUserOpt;
215*b1cdbd2cSJim Jagielski     ::rtl::OUString aName = aUserOpt.GetFirstName();
216*b1cdbd2cSJim Jagielski     if ( aName.getLength() )
217*b1cdbd2cSJim Jagielski         aName += ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( " " ) );
218*b1cdbd2cSJim Jagielski     aName += aUserOpt.GetLastName();
219*b1cdbd2cSJim Jagielski 
220*b1cdbd2cSJim Jagielski     return aName;
221*b1cdbd2cSJim Jagielski }
222*b1cdbd2cSJim Jagielski 
223*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------
GetCurrentLocalTime()224*b1cdbd2cSJim Jagielski ::rtl::OUString LockFileCommon::GetCurrentLocalTime()
225*b1cdbd2cSJim Jagielski {
226*b1cdbd2cSJim Jagielski     ::rtl::OUString aTime;
227*b1cdbd2cSJim Jagielski 
228*b1cdbd2cSJim Jagielski     TimeValue aSysTime;
229*b1cdbd2cSJim Jagielski     if ( osl_getSystemTime( &aSysTime ) )
230*b1cdbd2cSJim Jagielski     {
231*b1cdbd2cSJim Jagielski         TimeValue aLocTime;
232*b1cdbd2cSJim Jagielski         if ( osl_getLocalTimeFromSystemTime( &aSysTime, &aLocTime ) )
233*b1cdbd2cSJim Jagielski         {
234*b1cdbd2cSJim Jagielski             oslDateTime aDateTime;
235*b1cdbd2cSJim Jagielski             if ( osl_getDateTimeFromTimeValue( &aLocTime, &aDateTime ) )
236*b1cdbd2cSJim Jagielski             {
237*b1cdbd2cSJim Jagielski                 char pDateTime[20];
238*b1cdbd2cSJim Jagielski                 sprintf( pDateTime, "%02d.%02d.%4d %02d:%02d", aDateTime.Day, aDateTime.Month, aDateTime.Year, aDateTime.Hours, aDateTime.Minutes );
239*b1cdbd2cSJim Jagielski                 aTime = ::rtl::OUString::createFromAscii( pDateTime );
240*b1cdbd2cSJim Jagielski             }
241*b1cdbd2cSJim Jagielski         }
242*b1cdbd2cSJim Jagielski     }
243*b1cdbd2cSJim Jagielski 
244*b1cdbd2cSJim Jagielski     return aTime;
245*b1cdbd2cSJim Jagielski }
246*b1cdbd2cSJim Jagielski 
247*b1cdbd2cSJim Jagielski // ----------------------------------------------------------------------
GenerateOwnEntry()248*b1cdbd2cSJim Jagielski uno::Sequence< ::rtl::OUString > LockFileCommon::GenerateOwnEntry()
249*b1cdbd2cSJim Jagielski {
250*b1cdbd2cSJim Jagielski     uno::Sequence< ::rtl::OUString > aResult( LOCKFILE_ENTRYSIZE );
251*b1cdbd2cSJim Jagielski 
252*b1cdbd2cSJim Jagielski     aResult[LOCKFILE_OOOUSERNAME_ID] = GetOOOUserName();
253*b1cdbd2cSJim Jagielski 
254*b1cdbd2cSJim Jagielski     ::osl::Security aSecurity;
255*b1cdbd2cSJim Jagielski     aSecurity.getUserName( aResult[LOCKFILE_SYSUSERNAME_ID] );
256*b1cdbd2cSJim Jagielski 
257*b1cdbd2cSJim Jagielski     aResult[LOCKFILE_LOCALHOST_ID] = ::osl::SocketAddr::getLocalHostname();
258*b1cdbd2cSJim Jagielski 
259*b1cdbd2cSJim Jagielski     aResult[LOCKFILE_EDITTIME_ID] = GetCurrentLocalTime();
260*b1cdbd2cSJim Jagielski 
261*b1cdbd2cSJim Jagielski     ::utl::Bootstrap::locateUserInstallation( aResult[LOCKFILE_USERURL_ID] );
262*b1cdbd2cSJim Jagielski 
263*b1cdbd2cSJim Jagielski 
264*b1cdbd2cSJim Jagielski     return aResult;
265*b1cdbd2cSJim Jagielski }
266*b1cdbd2cSJim Jagielski 
267*b1cdbd2cSJim Jagielski } // namespace svt
268*b1cdbd2cSJim Jagielski 
269