]> git.saurik.com Git - apt.git/blame - apt-pkg/deb/debsrcrecords.cc
create directory for extended_states if needed
[apt.git] / apt-pkg / deb / debsrcrecords.cc
CommitLineData
11e7af84
AL
1// -*- mode: cpp; mode: fold -*-
2// Description /*{{{*/
c33b707e 3// $Id: debsrcrecords.cc,v 1.6 2004/03/17 05:58:54 mdz Exp $
11e7af84
AL
4/* ######################################################################
5
6 Debian Source Package Records - Parser implementation for Debian style
7 source indexes
8
9 ##################################################################### */
10 /*}}}*/
11// Include Files /*{{{*/
ea542140
DK
12#include <config.h>
13
b2e465d6 14#include <apt-pkg/deblistparser.h>
11e7af84
AL
15#include <apt-pkg/debsrcrecords.h>
16#include <apt-pkg/error.h>
36f610f1 17#include <apt-pkg/strutl.h>
b0e1a43f 18#include <apt-pkg/aptconfiguration.h>
453b82a3
DK
19#include <apt-pkg/srcrecords.h>
20#include <apt-pkg/tagfile.h>
cb6a2b3e 21#include <apt-pkg/hashes.h>
feab34c5 22#include <apt-pkg/gpgv.h>
4ab24e53 23
453b82a3
DK
24#include <ctype.h>
25#include <stdlib.h>
26#include <string.h>
27#include <algorithm>
28#include <string>
29#include <vector>
11e7af84
AL
30 /*}}}*/
31
453b82a3 32using std::max;
8f3ba4e8
DK
33using std::string;
34
11e7af84
AL
35// SrcRecordParser::Binaries - Return the binaries field /*{{{*/
36// ---------------------------------------------------------------------
37/* This member parses the binaries field into a pair of class arrays and
38 returns a list of strings representing all of the components of the
39 binaries field. The returned array need not be freed and will be
b2e465d6
AL
40 reused by the next Binaries function call. This function is commonly
41 used during scanning to find the right package */
11e7af84
AL
42const char **debSrcRecordParser::Binaries()
43{
39fb1e24
DK
44 const char *Start, *End;
45 if (Sect.Find("Binary", Start, End) == false)
46 return NULL;
47 for (; isspace(*Start) != 0; ++Start);
48 if (Start >= End)
49 return NULL;
50
51 StaticBinList.clear();
52 free(Buffer);
53 Buffer = strndup(Start, End - Start);
c33b707e 54
39fb1e24
DK
55 char* bin = Buffer;
56 do {
57 char* binStartNext = strchrnul(bin, ',');
58 char* binEnd = binStartNext - 1;
59 for (; isspace(*binEnd) != 0; --binEnd)
80624be7 60 binEnd = 0;
39fb1e24
DK
61 StaticBinList.push_back(bin);
62 if (*binStartNext != ',')
63 break;
64 *binStartNext = '\0';
80624be7
MV
65 for (bin = binStartNext + 1; isspace(*bin) != 0; ++bin)
66 ;
39fb1e24
DK
67 } while (*bin != '\0');
68 StaticBinList.push_back(NULL);
c33b707e 69
e788a834 70 return &StaticBinList[0];
b2e465d6
AL
71}
72 /*}}}*/
73// SrcRecordParser::BuildDepends - Return the Build-Depends information /*{{{*/
74// ---------------------------------------------------------------------
75/* This member parses the build-depends information and returns a list of
76 package/version records representing the build dependency. The returned
77 array need not be freed and will be reused by the next call to this
78 function */
8f3ba4e8 79bool debSrcRecordParser::BuildDepends(std::vector<pkgSrcRecords::Parser::BuildDepRec> &BuildDeps,
41c81fd8 80 bool const &ArchOnly, bool const &StripMultiArch)
b2e465d6
AL
81{
82 unsigned int I;
83 const char *Start, *Stop;
84 BuildDepRec rec;
85 const char *fields[] = {"Build-Depends",
86 "Build-Depends-Indep",
87 "Build-Conflicts",
88 "Build-Conflicts-Indep"};
89
90 BuildDeps.clear();
11e7af84 91
b2e465d6 92 for (I = 0; I < 4; I++)
11e7af84 93 {
45430cbf
AL
94 if (ArchOnly && (I == 1 || I == 3))
95 continue;
96
b2e465d6
AL
97 if (Sect.Find(fields[I], Start, Stop) == false)
98 continue;
11e7af84 99
b2e465d6
AL
100 while (1)
101 {
102 Start = debListParser::ParseDepends(Start, Stop,
565ded7b 103 rec.Package,rec.Version,rec.Op,true,StripMultiArch,true);
b2e465d6
AL
104
105 if (Start == 0)
106 return _error->Error("Problem parsing dependency: %s", fields[I]);
107 rec.Type = I;
108
109 if (rec.Package != "")
110 BuildDeps.push_back(rec);
111
112 if (Start == Stop)
113 break;
114 }
11e7af84
AL
115 }
116
b2e465d6 117 return true;
11e7af84
AL
118}
119 /*}}}*/
36f610f1
AL
120// SrcRecordParser::Files - Return a list of files for this source /*{{{*/
121// ---------------------------------------------------------------------
122/* This parses the list of files and returns it, each file is required to have
123 a complete source package */
8f3ba4e8 124bool debSrcRecordParser::Files(std::vector<pkgSrcRecords::File> &List)
36f610f1
AL
125{
126 List.erase(List.begin(),List.end());
cb6a2b3e 127
1262d358
DK
128 // Stash the / terminated directory prefix
129 string Base = Sect.FindS("Directory");
130 if (Base.empty() == false && Base[Base.length()-1] != '/')
131 Base += '/';
cb6a2b3e 132
1262d358 133 std::vector<std::string> const compExts = APT::Configuration::getCompressorExtensions();
cb6a2b3e 134
1262d358
DK
135 for (char const * const * type = HashString::SupportedHashes(); *type != NULL; ++type)
136 {
137 // derive field from checksum type
138 std::string checksumField("Checksums-");
139 if (strcmp(*type, "MD5Sum") == 0)
140 checksumField = "Files"; // historic name for MD5 checksums
141 else
142 checksumField.append(*type);
143
144 string const Files = Sect.FindS(checksumField.c_str());
145 if (Files.empty() == true)
146 continue;
cb6a2b3e
MV
147
148 // Iterate over the entire list grabbing each triplet
149 const char *C = Files.c_str();
150 while (*C != 0)
1262d358
DK
151 {
152 string hash, size, path;
153
154 // Parse each of the elements
155 if (ParseQuoteWord(C, hash) == false ||
156 ParseQuoteWord(C, size) == false ||
157 ParseQuoteWord(C, path) == false)
158 return _error->Error("Error parsing file record in %s of source package %s", checksumField.c_str(), Package().c_str());
159
160 HashString const hashString(*type, hash);
161 if (Base.empty() == false)
162 path = Base + path;
163
164 // look if we have a record for this file already
165 std::vector<pkgSrcRecords::File>::iterator file = List.begin();
166 for (; file != List.end(); ++file)
167 if (file->Path == path)
168 break;
169
170 // we have it already, store the new hash and be done
171 if (file != List.end())
172 {
1262d358 173 if (checksumField == "Files")
586d8704 174 APT_IGNORE_DEPRECATED(file->MD5Hash = hash;)
1262d358
DK
175 // an error here indicates that we have two different hashes for the same file
176 if (file->Hashes.push_back(hashString) == false)
177 return _error->Error("Error parsing checksum in %s of source package %s", checksumField.c_str(), Package().c_str());
178 continue;
179 }
180
181 // we haven't seen this file yet
182 pkgSrcRecords::File F;
183 F.Path = path;
184 F.Size = strtoull(size.c_str(), NULL, 10);
185 F.Hashes.push_back(hashString);
186
1262d358 187 if (checksumField == "Files")
586d8704 188 APT_IGNORE_DEPRECATED(F.MD5Hash = hash;)
1262d358
DK
189
190 // Try to guess what sort of file it is we are getting.
191 string::size_type Pos = F.Path.length()-1;
192 while (1)
193 {
194 string::size_type Tmp = F.Path.rfind('.',Pos);
195 if (Tmp == string::npos)
196 break;
197 if (F.Type == "tar") {
198 // source v3 has extension 'debian.tar.*' instead of 'diff.*'
199 if (string(F.Path, Tmp+1, Pos-Tmp) == "debian")
200 F.Type = "diff";
201 break;
202 }
203 F.Type = string(F.Path,Tmp+1,Pos-Tmp);
204
205 if (std::find(compExts.begin(), compExts.end(), std::string(".").append(F.Type)) != compExts.end() ||
206 F.Type == "tar")
207 {
208 Pos = Tmp-1;
209 continue;
210 }
211
212 break;
213 }
214 List.push_back(F);
cb6a2b3e 215 }
36f610f1 216 }
1262d358
DK
217
218 return true;
36f610f1
AL
219}
220 /*}}}*/
7a9f09bd
MV
221// SrcRecordParser::~SrcRecordParser - Destructor /*{{{*/
222// ---------------------------------------------------------------------
223/* */
224debSrcRecordParser::~debSrcRecordParser()
225{
7f48c4df
MV
226 // was allocated via strndup()
227 free(Buffer);
7a9f09bd
MV
228}
229 /*}}}*/
feab34c5
MV
230
231
a49e7948
MV
232debDscRecordParser::debDscRecordParser(std::string const &DscFile, pkgIndexFile const *Index)
233 : debSrcRecordParser(DscFile, Index)
feab34c5
MV
234{
235 // support clear signed files
236 if (OpenMaybeClearSignedFile(DscFile, Fd) == false)
237 {
238 _error->Error("Failed to open %s", DscFile.c_str());
239 return;
240 }
241
242 // re-init to ensure the updated Fd is used
243 Tags.Init(&Fd);
244 // read the first (and only) record
245 Step();
246
247}