1
1
Fork 0
mirror of https://github.com/NixOS/nix.git synced 2025-12-15 13:31:05 +01:00

Merge pull request #14677 from NixOS/restartable-source-no-path

libutil: Get rid of restartableSourceFromFactory, add createAnonymousTempFile
This commit is contained in:
John Ericson 2025-12-01 03:46:48 +00:00 committed by GitHub
commit 890a4e980a
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
6 changed files with 86 additions and 73 deletions

View file

@ -713,11 +713,31 @@ std::filesystem::path createTempDir(const std::filesystem::path & tmpRoot, const
}
}
AutoCloseFD createAnonymousTempFile()
{
AutoCloseFD fd;
#ifdef O_TMPFILE
fd = ::open(defaultTempDir().c_str(), O_TMPFILE | O_CLOEXEC | O_RDWR, S_IWUSR | S_IRUSR);
if (!fd)
throw SysError("creating anonymous temporary file");
#else
auto [fd2, path] = createTempFile("nix-anonymous");
if (!fd2)
throw SysError("creating temporary file '%s'", path);
fd = std::move(fd2);
# ifndef _WIN32
unlink(requireCString(path)); /* We only care about the file descriptor. */
# endif
#endif
return fd;
}
std::pair<AutoCloseFD, Path> createTempFile(const Path & prefix)
{
Path tmpl(defaultTempDir() + "/" + prefix + ".XXXXXX");
// Strictly speaking, this is UB, but who cares...
// FIXME: use O_TMPFILE.
// FIXME: Windows should use FILE_ATTRIBUTE_TEMPORARY | FILE_FLAG_DELETE_ON_CLOSE
AutoCloseFD fd = toDescriptor(mkstemp((char *) tmpl.c_str()));
if (!fd)
throw SysError("creating temporary file '%s'", tmpl);

View file

@ -341,6 +341,12 @@ typedef std::unique_ptr<DIR, DIRDeleter> AutoCloseDir;
std::filesystem::path
createTempDir(const std::filesystem::path & tmpRoot = "", const std::string & prefix = "nix", mode_t mode = 0755);
/**
* Create an anonymous readable/writable temporary file, returning a file handle.
* On UNIX there resulting file isn't linked to any path on the filesystem.
*/
AutoCloseFD createAnonymousTempFile();
/**
* Create a temporary file, returning a file handle and its path.
*/

View file

@ -105,7 +105,7 @@ struct Source
* A buffered abstract source. Warning: a BufferedSource should not be
* used from multiple threads concurrently.
*/
struct BufferedSource : Source
struct BufferedSource : virtual Source
{
size_t bufSize, bufPosIn, bufPosOut;
std::unique_ptr<char[]> buffer;
@ -132,6 +132,14 @@ protected:
virtual size_t readUnbuffered(char * data, size_t len) = 0;
};
/**
* Source type that can be restarted.
*/
struct RestartableSource : virtual Source
{
virtual void restart() = 0;
};
/**
* A sink that writes data to a file descriptor.
*/
@ -174,7 +182,7 @@ private:
/**
* A source that reads data from a file descriptor.
*/
struct FdSource : BufferedSource
struct FdSource : BufferedSource, RestartableSource
{
Descriptor fd;
size_t read = 0;
@ -196,6 +204,7 @@ struct FdSource : BufferedSource
FdSource & operator=(FdSource && s) = default;
bool good() override;
void restart() override;
/**
* Return true if the buffer is not empty after a non-blocking
@ -230,14 +239,6 @@ struct StringSink : Sink
void operator()(std::string_view data) override;
};
/**
* Source type that can be restarted.
*/
struct RestartableSource : Source
{
virtual void restart() = 0;
};
/**
* A source that reads data from a string.
*/
@ -316,15 +317,6 @@ public:
}
};
/**
* Create a restartable Source from a factory function.
*
* @param factory Factory function that returns a fresh instance of the Source. Gets
* called for each source restart.
* @pre factory must return an equivalent source for each invocation.
*/
std::unique_ptr<RestartableSource> restartableSourceFromFactory(std::function<std::unique_ptr<Source>()> factory);
/**
* A sink that writes all incoming data to two other sinks.
*/

View file

@ -201,6 +201,16 @@ bool FdSource::hasData()
}
}
void FdSource::restart()
{
if (!isSeekable)
throw Error("can't seek to the start of a file");
buffer.reset();
read = bufPosOut = bufPosOut = 0;
if (lseek(fd, 0, SEEK_SET) == -1)
throw SysError("seeking to the start of a file");
}
void FdSource::skip(size_t len)
{
/* Discard data in the buffer. */
@ -527,41 +537,4 @@ size_t ChainSource::read(char * data, size_t len)
}
}
std::unique_ptr<RestartableSource> restartableSourceFromFactory(std::function<std::unique_ptr<Source>()> factory)
{
struct RestartableSourceImpl : RestartableSource
{
RestartableSourceImpl(decltype(factory) factory_)
: factory_(std::move(factory_))
, impl(this->factory_())
{
}
decltype(factory) factory_;
std::unique_ptr<Source> impl = factory_();
size_t read(char * data, size_t len) override
{
return impl->read(data, len);
}
bool good() override
{
return impl->good();
}
void skip(size_t len) override
{
return impl->skip(len);
}
void restart() override
{
impl = factory_();
}
};
return std::make_unique<RestartableSourceImpl>(std::move(factory));
}
} // namespace nix