diff options
author | Robin Hack <hack.robin@gmail.com> | 2015-08-21 13:54:03 +1200 |
---|---|---|
committer | Andrew Bartlett <abartlet@samba.org> | 2015-08-21 07:17:35 +0200 |
commit | 59e955bfd136cbe798d155a4930f6ce4bfe00020 (patch) | |
tree | 0ae3e67ea0ca0f4bee1f5b7937f5a1695c2cdff3 | |
parent | ba4c9bd08c09e77df5f653dc7c39be9e2908cfc0 (diff) | |
download | samba-59e955bfd136cbe798d155a4930f6ce4bfe00020.tar.gz |
vfs_scannedonly: Remove vfs_scannedonly from samba source tree.
BUG: https://bugzilla.samba.org/show_bug.cgi?id=11459
Signed-off-by: Robin Hack <hack.robin@gmail.com>
Reviewed-by: Andrew Bartlett <abartlet@samba.org>
Reviewed-by: Andreas Schneider <asn@samba.org>
Autobuild-User(master): Andrew Bartlett <abartlet@samba.org>
Autobuild-Date(master): Fri Aug 21 07:17:35 CEST 2015 on sn-devel-104
-rw-r--r-- | docs-xml/manpages/vfs_scannedonly.8.xml | 243 | ||||
-rw-r--r-- | docs-xml/wscript_build | 1 | ||||
-rw-r--r-- | packaging/RHEL-CTDB/samba.spec.tmpl | 1 | ||||
-rw-r--r-- | source3/modules/vfs_scannedonly.c | 1044 | ||||
-rw-r--r-- | source3/modules/wscript_build | 8 | ||||
-rw-r--r-- | source3/wscript | 2 |
6 files changed, 1 insertions, 1298 deletions
diff --git a/docs-xml/manpages/vfs_scannedonly.8.xml b/docs-xml/manpages/vfs_scannedonly.8.xml deleted file mode 100644 index 5f569b7d8c0..00000000000 --- a/docs-xml/manpages/vfs_scannedonly.8.xml +++ /dev/null @@ -1,243 +0,0 @@ -<?xml version="1.0" encoding="iso-8859-1"?> -<!DOCTYPE refentry PUBLIC "-//Samba-Team//DTD DocBook V4.2-Based Variant V1.0//EN" "http://www.samba.org/samba/DTD/samba-doc"> -<refentry id="vfs_scannedonly.8"> - -<refmeta> - <refentrytitle>vfs_scannedonly</refentrytitle> - <manvolnum>8</manvolnum> - <refmiscinfo class="source">Samba</refmiscinfo> - <refmiscinfo class="manual">System Administration tools</refmiscinfo> - <refmiscinfo class="version">4.3</refmiscinfo> -</refmeta> - - -<refnamediv> - <refname>vfs_scannedonly</refname> - <refpurpose>Ensures that only files that have been scanned for viruses are - visible and accessible to the end user.</refpurpose> -</refnamediv> - -<refsynopsisdiv> - <cmdsynopsis> - <command>vfs objects = scannedonly</command> - </cmdsynopsis> -</refsynopsisdiv> - -<refsect1> - <title>DESCRIPTION</title> - - <para>This VFS module is part of the - <citerefentry><refentrytitle>samba</refentrytitle> - <manvolnum>8</manvolnum></citerefentry> suite.</para> - - <para>The <command>vfs_scannedonly</command> VFS module ensures that - only files that have been scanned for viruses are visible and accessible - to the end user. If non-scanned files are found an anti-virus scanning - daemon is notified. The anti-virus scanning daemon is not part of the - Samba suite. - </para> - - <para>Scannedonly comes in two parts: a samba vfs module and (one or - more) daemons. The daemon scans files. If a certain file is clean, - a second file is created with prefix <filename>.scanned:</filename>. - The Samba module simply looks if such a <filename>.scanned:</filename> - file exists, and is newer than the pertinent file. If this is the case, - the file is shown to the user. If this is not the case, the file is not - returned in a directory listing (configurable), and cannot be opened - (configurable). The Samba vfs module will notify the daemon to scan - this file. - </para> - - <para>So what happens for the user in the default configuration. The - first time a directory is listed, it shows files as 'file is being - scanned for viruses, but after the first time all files are shown. - There is a utility scannedonly_prescan that can help you to prescan - all directories. When new files are written the daemon is notified - immediately after the file is complete. - </para> - - <para>If a virus is found by the daemon, a file with a warning message - is created in the directory of the user, a warning is sent to the logs, - and the file is renamed to have prefix <filename>.virus:</filename>. - Files with the <filename>.virus:</filename> prefix are never shown to - the user and all access is denied. - </para> - - <para>This module is stackable.</para> - -</refsect1> - -<refsect1> - <title>CONFIGURATION</title> - - <para><command>vfs_scannedonly</command> relies on a anti-virus scanning - daemon that listens on the scannedonly socket (unix domain socket or UDP - socket). - </para> -</refsect1> - -<refsect1> - <title>OPTIONS</title> - - <variablelist> - <varlistentry> - <term>scannedonly:domain_socket = True </term> - <listitem> - <para>Whether to use a unix domain socket or not (false reverts - to use udp) - </para> - </listitem> - </varlistentry> - - <varlistentry> - <term>scannedonly:socketname = /var/lib/scannedonly/scan</term> - <listitem> - <para>The location of the unix domain socket to connect to</para> - </listitem> - </varlistentry> - - <varlistentry> - <term>scannedonly:portnum = 2020</term> - <listitem> - <para>The udp port number to connect to - </para> - </listitem> - </varlistentry> - <varlistentry><term>scannedonly:scanhost = localhost</term> - <listitem> - <para> - When using UDP the host that runs the scanning daemon (this host - needs access to the files!) - </para> - </listitem> - </varlistentry> - <varlistentry><term>scannedonly:show_special_files = True</term> - <listitem> - <para> - Whether sockets, devices and fifo's (all not scanned for - viruses) should be visible to the user - </para> - </listitem> - </varlistentry> - <varlistentry><term>scannedonly:rm_hidden_files_on_rmdir = True</term> - <listitem> - <para> - Whether files that are not visible (<filename>.scanned:</filename> - files, <filename>.failed:</filename> files and <filename>.virus: - </filename> files) should be deleted if the user tries to remove - the directory. If false, the user will get the "directory is not - empty" error. - </para> - </listitem> - </varlistentry> - <varlistentry><term>scannedonly:hide_nonscanned_files = True</term> - <listitem> - <para> - If false, all non-scanned files are visible in directory listings. - If such files are found in a directory listing the scanning daemon - is notified that scanning is required. Access to non-scanned files - is still denied (see scannedonly:allow_nonscanned_files). - </para> - </listitem> - </varlistentry> - <varlistentry><term>scannedonly:scanning_message = is being scanned for - viruses</term> - <listitem> - <para> - If non-scanned files are hidden - (if scannedonly:hide_nonscanned_files = True), a fake 0 byte file - is shown. The filename is the original filename with the message - as suffix. - </para> - </listitem> - </varlistentry> - <varlistentry><term>scannedonly:recheck_time_open = 50</term> - <listitem> - <para> - If a non-scanned file is opened, the vfs module will wait - recheck_tries_open times for recheck_time_open milliseconds for - the scanning daemon to create a <filename>.scanned:</filename> - file. For small files that are scanned by the daemon within the - time (tries * time) the behavior will be just like on-access - scanning. - </para> - </listitem> - </varlistentry> - <varlistentry><term>scannedonly:recheck_tries_open = 100</term> - <listitem> - <para> - See recheck_time_open. - </para> - </listitem> - </varlistentry> - <varlistentry><term>scannedonly:recheck_time_readdir = 50</term> - <listitem> - <para> - If a non-scanned file is in a directory listing the vfs module - notifies the daemon (once for all files that need scanning in - that directory), and waits recheck_tries_readdir times for - recheck_time_readdir milliseconds. Only used when - hide_nonscanned_files is false. - </para> - </listitem> - </varlistentry> - <varlistentry><term>scannedonly:recheck_tries_readdir = 20</term> - <listitem> - <para> - See recheck_time_readdir. - </para> - </listitem> - </varlistentry> - <varlistentry><term>scannedonly:allow_nonscanned_files = False</term> - <listitem> - <para> - Allow access to non-scanned files. The daemon is notified, - however, and special files such as <filename>.scanned:</filename> - files. <filename>.virus:</filename> files and - <filename>.failed:</filename> files are not listed. - </para> - </listitem> - </varlistentry> - - </variablelist> -</refsect1> - -<refsect1> - <title>EXAMPLES</title> - - <para>Enable anti-virus scanning:</para> -<programlisting> - <smbconfsection name="[homes]"/> - <smbconfoption name="vfs objects">scannedonly</smbconfoption> - <smbconfoption name="scannedonly:hide_nonscanned_files">False</smbconfoption> -</programlisting> - -</refsect1> - -<refsect1> - <title>CAVEATS</title> - - <para>This is not true on-access scanning. However, it is very fast - for files that have been scanned already. - </para> -</refsect1> - -<refsect1> - <title>VERSION</title> - - <para>This man page is correct for version 4.0.0 of the Samba suite. - </para> -</refsect1> - -<refsect1> - <title>AUTHOR</title> - - <para>The original Samba software and related utilities - were created by Andrew Tridgell. Scannedonly was - developed for Samba by Olivier Sessink. Samba is now developed - by the Samba Team as an Open Source project similar - to the way the Linux kernel is developed.</para> - -</refsect1> - -</refentry> diff --git a/docs-xml/wscript_build b/docs-xml/wscript_build index 5c42a31f4af..515746075ac 100644 --- a/docs-xml/wscript_build +++ b/docs-xml/wscript_build @@ -73,7 +73,6 @@ manpages=''' manpages/vfs_readahead.8 manpages/vfs_readonly.8 manpages/vfs_recycle.8 - manpages/vfs_scannedonly.8 manpages/vfs_shadow_copy.8 manpages/vfs_shadow_copy2.8 manpages/vfs_shell_snap.8 diff --git a/packaging/RHEL-CTDB/samba.spec.tmpl b/packaging/RHEL-CTDB/samba.spec.tmpl index ad188269f4b..75228069c4d 100644 --- a/packaging/RHEL-CTDB/samba.spec.tmpl +++ b/packaging/RHEL-CTDB/samba.spec.tmpl @@ -419,7 +419,6 @@ exit 0 %{_libarchdir}/samba/vfs/readahead.so %{_libarchdir}/samba/vfs/readonly.so %{_libarchdir}/samba/vfs/recycle.so -%{_libarchdir}/samba/vfs/scannedonly.so %{_libarchdir}/samba/vfs/shadow_copy.so %{_libarchdir}/samba/vfs/shadow_copy2.so %{_libarchdir}/samba/vfs/smb_traffic_analyzer.so diff --git a/source3/modules/vfs_scannedonly.c b/source3/modules/vfs_scannedonly.c deleted file mode 100644 index 128374a3db5..00000000000 --- a/source3/modules/vfs_scannedonly.c +++ /dev/null @@ -1,1044 +0,0 @@ -/* - * scannedonly VFS module for Samba 3.5 and beyond - * - * Copyright 2007,2008,2009,2010,2011 (C) Olivier Sessink - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation; either version 3 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program; if not, write to the Free Software - * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. - * - * ABOUT SCANNEDONLY - * - * scannedonly implements a 'filter' like vfs module that talks over a - * unix domain socket or over UDP to a anti-virus engine. - * - * files that are clean have a corresponding .scanned:{filename} file - * in the same directory. So why the .scanned: files? They take up - * only an inode, because they are 0 bytes. To test if the file is - * scanned only a stat() call on the filesystem is needed which is - * very quick compared to a database lookup. All modern filesystems - * use database technology such as balanced trees for lookups anyway. - * The number of inodes in modern filesystems is also not limiting - * anymore. The .scanned: files are also easy scriptable. You can - * remove them with a simple find command or create them with a - * simple touch command. Extended filesystem attributes have similar - * properties, but are not supported on all filesystems, so that - * would limit the usage of the module (and attributes are not as - * easily scriptable) - * - * files that are not clean are sent to the AV-engine. Only the - * filename is sent over the socket. The protocol is very simple: - * a newline separated list of filenames inside each datagram. - * - * a file AV-scan may be requested multiple times, the AV-engine - * should also check if the file has been scanned already. Requests - * can also be dropped by the AV-engine (and we thus don't need the - * reliability of TCP). - * - */ - -#include "includes.h" -#include "smbd/smbd.h" -#include "system/filesys.h" - -#include "config.h" - -#define SENDBUFFERSIZE 1450 - -#ifndef SUN_LEN -#define SUN_LEN(sunp) ((size_t)((struct sockaddr_un *)0)->sun_path \ - + strlen((sunp)->sun_path)) -#endif - - -struct Tscannedonly { - int socket; - int domain_socket; - int portnum; - int scanning_message_len; - int recheck_time_open; - int recheck_tries_open; - int recheck_size_open; - int recheck_time_readdir; - int recheck_tries_readdir; - bool show_special_files; - bool rm_hidden_files_on_rmdir; - bool hide_nonscanned_files; - bool allow_nonscanned_files; - const char *socketname; - const char *scanhost; - const char *scanning_message; - const char *p_scanned; /* prefix for scanned files */ - const char *p_virus; /* prefix for virus containing files */ - const char *p_failed; /* prefix for failed to scan files */ - char gsendbuffer[SENDBUFFERSIZE + 1]; -}; - -#define STRUCTSCANO(var) ((struct Tscannedonly *)var) - -struct scannedonly_DIR { - char *base; - int recheck_tries_done; /* if 0 the directory listing has not yet - been checked for files that need to be scanned. */ - DIR *DIR; -}; -#define SCANNEDONLY_DEBUG 9 -/*********************/ -/* utility functions */ -/*********************/ - -static char *real_path_from_notify_path(TALLOC_CTX *ctx, - struct Tscannedonly *so, - const char *path) -{ - char *name; - int len, pathlen; - - name = strrchr(path, '/'); - if (!name) { - return NULL; - } - pathlen = name - path; - name++; - len = strlen(name); - if (len <= so->scanning_message_len) { - return NULL; - } - - if (strcmp(name + (len - so->scanning_message_len), - so->scanning_message) != 0) { - return NULL; - } - - return talloc_strndup(ctx,path, - pathlen + len - so->scanning_message_len); -} - -static char *cachefile_name(TALLOC_CTX *ctx, - const char *shortname, - const char *base, - const char *p_scanned) -{ - return talloc_asprintf(ctx, "%s%s%s", base, p_scanned, shortname); -} - -static char *name_w_ending_slash(TALLOC_CTX *ctx, const char *name) -{ - int len = strlen(name); - if (name[len - 1] == '/') { - return talloc_strdup(ctx,name); - } else { - return talloc_asprintf(ctx, "%s/", name); - } -} - -static char *cachefile_name_f_fullpath(TALLOC_CTX *ctx, - const char *fullpath, - const char *p_scanned) -{ - const char *base; - char *tmp, *cachefile; - const char *shortname; - tmp = strrchr(fullpath, '/'); - if (tmp) { - base = talloc_strndup(ctx, fullpath, (tmp - fullpath) + 1); - shortname = tmp + 1; - } else { - base = ""; - shortname = (const char *)fullpath; - } - cachefile = cachefile_name(ctx, shortname, base, p_scanned); - DEBUG(SCANNEDONLY_DEBUG, - ("cachefile_name_f_fullpath cachefile=%s\n", cachefile)); - return cachefile; -} - -static char *construct_full_path(TALLOC_CTX *ctx, vfs_handle_struct * handle, - const char *somepath, bool ending_slash) -{ - const char *tmp; - - if (!somepath) { - return NULL; - } - if (somepath[0] == '/') { - if (ending_slash) { - return name_w_ending_slash(ctx,somepath); - } - return talloc_strdup(ctx,somepath); - } - tmp = somepath; - if (tmp[0]=='.'&&tmp[1]=='/') { - tmp+=2; - } - /* vfs_GetWd() seems to return a path with a slash */ - if (ending_slash) { - return talloc_asprintf(ctx, "%s/%s/", - vfs_GetWd(ctx, handle->conn),tmp); - } - return talloc_asprintf(ctx, "%s/%s", - vfs_GetWd(ctx, handle->conn),tmp); -} - -static int connect_to_scanner(vfs_handle_struct * handle) -{ - struct Tscannedonly *so = (struct Tscannedonly *)handle->data; - - if (so->domain_socket) { - struct sockaddr_un saun; - DEBUG(SCANNEDONLY_DEBUG, ("socket=%s\n", so->socketname)); - if ((so->socket = socket(AF_UNIX, SOCK_DGRAM, 0)) < 0) { - DEBUG(2, ("failed to create socket %s\n", - so->socketname)); - return -1; - } - saun.sun_family = AF_UNIX; - strncpy(saun.sun_path, so->socketname, - sizeof(saun.sun_path) - 1); - if (connect(so->socket, (struct sockaddr *)(void *)&saun, - SUN_LEN(&saun)) < 0) { - DEBUG(2, ("failed to connect to socket %s\n", - so->socketname)); - return -1; - } - DEBUG(SCANNEDONLY_DEBUG,("bound %s to socket %d\n", - saun.sun_path, so->socket)); - - } else { - so->socket = open_udp_socket(so->scanhost, so->portnum); - if (so->socket < 0) { - DEBUG(2,("failed to open UDP socket to %s:%d\n", - so->scanhost,so->portnum)); - return -1; - } - } - - {/* increasing the socket buffer is done because we have large bursts - of UDP packets or DGRAM's on a domain socket whenever we hit a - large directory with lots of unscanned files. */ - int sndsize; - socklen_t size = sizeof(int); - if (getsockopt(so->socket, SOL_SOCKET, SO_RCVBUF, - (char *)&sndsize, &size) == 0) { - DEBUG(SCANNEDONLY_DEBUG, - ("current socket buffer size=%d\n", - sndsize)); - } - sndsize = 262144; - if (setsockopt(so->socket, SOL_SOCKET, SO_RCVBUF, - (char *)&sndsize, - (int)sizeof(sndsize)) != 0) { - DEBUG(SCANNEDONLY_DEBUG, - ("error setting socket buffer %s (%d)\n", - strerror(errno), errno)); - } - } - set_blocking(so->socket, false); - return 0; -} - -static void flush_sendbuffer(vfs_handle_struct * handle) -{ - struct Tscannedonly *so = (struct Tscannedonly *)handle->data; - int ret, len, loop = 10; - if (so->gsendbuffer[0] == '\0') { - return; - } - - do { - loop--; - len = strlen(so->gsendbuffer); - ret = send(so->socket, so->gsendbuffer, len, 0); - if (ret == len) { - so->gsendbuffer[0] = '\0'; - break; - } - if (ret == -1) { - DEBUG(3,("scannedonly flush_sendbuffer: " - "error sending on socket %d to scanner:" - " %s (%d)\n", - so->socket, strerror(errno), errno)); - if (errno == ECONNREFUSED || errno == ENOTCONN - || errno == ECONNRESET) { - if (connect_to_scanner(handle) == -1) - break; /* connecting fails, abort */ - /* try again */ - } else if (errno != EINTR) { - /* on EINTR we just try again, all remaining - other errors we log the error - and try again ONCE */ - loop = 1; - DEBUG(3,("scannedonly flush_sendbuffer: " - "error sending data to scanner: %s " - "(%d)\n", strerror(errno), errno)); - } - } else { - /* --> partial write: Resend all filenames that were - not or not completely written. a partial filename - written means the filename will not arrive correctly, - so resend it completely */ - int pos = 0; - while (pos < len) { - char *tmp = strchr(so->gsendbuffer+pos, '\n'); - if (tmp && tmp - so->gsendbuffer < ret) - pos = tmp - so->gsendbuffer + 1; - else - break; - } - memmove(so->gsendbuffer, so->gsendbuffer + pos, - SENDBUFFERSIZE - ret); - /* now try again */ - } - } while (loop > 0); - - if (so->gsendbuffer[0] != '\0') { - DEBUG(2, - ("scannedonly flush_sendbuffer: " - "failed to send files to AV scanner, " - "discarding files.")); - so->gsendbuffer[0] = '\0'; - } -} - -static void notify_scanner(vfs_handle_struct * handle, const char *scanfile) -{ - const char *tmp; - int tmplen, gsendlen; - struct Tscannedonly *so = (struct Tscannedonly *)handle->data; - TALLOC_CTX *ctx=talloc_tos(); - if (scanfile[0] != '/') { - tmp = construct_full_path(ctx,handle, scanfile, false); - } else { - tmp = (const char *)scanfile; - } - tmplen = strlen(tmp); - gsendlen = strlen(so->gsendbuffer); - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly notify_scanner: tmp=%s, tmplen=%d, gsendlen=%d\n", - tmp, tmplen, gsendlen)); - if (gsendlen + tmplen >= SENDBUFFERSIZE) { - flush_sendbuffer(handle); - } - /* FIXME ! Truncate checks... JRA. */ - (void)strlcat(so->gsendbuffer, tmp, SENDBUFFERSIZE + 1); - (void)strlcat(so->gsendbuffer, "\n", SENDBUFFERSIZE + 1); -} - -static bool is_scannedonly_file(struct Tscannedonly *so, const char *shortname) -{ - if (shortname[0]!='.') { - return false; - } - if (strncmp(shortname, so->p_scanned, strlen(so->p_scanned)) == 0) { - return true; - } - if (strncmp(shortname, so->p_virus, strlen(so->p_virus)) == 0) { - return true; - } - if (strncmp(shortname, so->p_failed, strlen(so->p_failed)) == 0) { - return true; - } - return false; -} - -static bool timespec_is_newer(struct timespec *base, struct timespec *test) -{ - return timespec_compare(base,test) < 0; -} - -/* -vfs_handle_struct *handle the scannedonly handle -scannedonly_DIR * sDIR the scannedonly struct if called from _readdir() -or NULL -fullpath is a full path starting from / or a relative path to the -current working directory -shortname is the filename without directory components -basename, is the directory without file name component -allow_nonexistent return TRUE if stat() on the requested file fails -recheck_time, the time in milliseconds to wait for the daemon to -create a .scanned file -recheck_tries, the number of tries to wait -recheck_size, size in Kb of files that should not be waited for -loop : boolean if we should try to loop over all files in the directory -and send a notify to the scanner for all files that need scanning -*/ -static bool scannedonly_allow_access(vfs_handle_struct * handle, - struct scannedonly_DIR *sDIR, - struct smb_filename *smb_fname, - const char *shortname, - const char *base_name, - int allow_nonexistent, - int recheck_time, int recheck_tries, - int recheck_size, int loop) -{ - struct smb_filename *cache_smb_fname; - TALLOC_CTX *ctx=talloc_tos(); - char *cachefile; - int retval = -1; - DEBUG(SCANNEDONLY_DEBUG, - ("smb_fname->base_name=%s, shortname=%s, base_name=%s\n" - ,smb_fname->base_name,shortname,base_name)); - - if (ISDOT(shortname) || ISDOTDOT(shortname)) { - return true; - } - if (is_scannedonly_file(STRUCTSCANO(handle->data), shortname)) { - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_allow_access, %s is a scannedonly file, " - "return 0\n", shortname)); - return false; - } - - if (!VALID_STAT(smb_fname->st)) { - DEBUG(SCANNEDONLY_DEBUG,("stat %s\n",smb_fname->base_name)); - retval = SMB_VFS_NEXT_STAT(handle, smb_fname); - if (retval != 0) { - /* failed to stat this file?!? --> hide it */ - DEBUG(SCANNEDONLY_DEBUG,("no valid stat, return" - " allow_nonexistent=%d\n", - allow_nonexistent)); - return allow_nonexistent; - } - } - if (!S_ISREG(smb_fname->st.st_ex_mode)) { - DEBUG(SCANNEDONLY_DEBUG, - ("%s is not a regular file, ISDIR=%d\n", - smb_fname->base_name, - S_ISDIR(smb_fname->st.st_ex_mode))); - return (STRUCTSCANO(handle->data)-> - show_special_files || - S_ISDIR(smb_fname->st.st_ex_mode)); - } - if (smb_fname->st.st_ex_size == 0) { - DEBUG(SCANNEDONLY_DEBUG,("empty file, return 1\n")); - return true; /* empty files cannot contain viruses ! */ - } - cachefile = cachefile_name(ctx, - shortname, - base_name, - STRUCTSCANO(handle->data)->p_scanned); - cache_smb_fname = synthetic_smb_fname(ctx, cachefile,NULL,NULL); - if (!VALID_STAT(cache_smb_fname->st)) { - retval = SMB_VFS_NEXT_STAT(handle, cache_smb_fname); - } - if (retval == 0 && VALID_STAT(cache_smb_fname->st)) { - if (timespec_is_newer(&smb_fname->st.st_ex_ctime, - &cache_smb_fname->st.st_ex_ctime)) { - talloc_free(cache_smb_fname); - return true; - } - /* no cachefile or too old */ - SMB_VFS_NEXT_UNLINK(handle, cache_smb_fname); - retval = -1; - } - - notify_scanner(handle, smb_fname->base_name); - - if (loop && sDIR && sDIR->recheck_tries_done == 0) { - /* check the rest of the directory and notify the - scanner if some file needs scanning */ - long offset; - struct dirent *dire; - - offset = SMB_VFS_NEXT_TELLDIR(handle, sDIR->DIR); - dire = SMB_VFS_NEXT_READDIR(handle, sDIR->DIR, NULL); - while (dire) { - char *fpath2; - struct smb_filename *smb_fname2; - fpath2 = talloc_asprintf(ctx, "%s%s", base_name,dire->d_name); - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_allow_access in loop, " - "found %s\n", fpath2)); - smb_fname2 = synthetic_smb_fname( - ctx, fpath2,NULL,NULL); - scannedonly_allow_access(handle, NULL, - smb_fname2, - dire->d_name, - base_name, 0, 0, 0, 0, 0); - talloc_free(fpath2); - talloc_free(smb_fname2); - dire = SMB_VFS_NEXT_READDIR(handle, sDIR->DIR,NULL); - } - sDIR->recheck_tries_done = 1; - SMB_VFS_NEXT_SEEKDIR(handle, sDIR->DIR, offset); - } - if (recheck_time > 0 - && ((recheck_size > 0 - && smb_fname->st.st_ex_size < (1024 * recheck_size)) - || (sDIR && sDIR->recheck_tries_done < recheck_tries) - )) { - int numloops = sDIR ? sDIR->recheck_tries_done : 0; - flush_sendbuffer(handle); - while (retval != 0 /*&& errno == ENOENT */ - && numloops < recheck_tries) { - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_allow_access, wait (try=%d " - "(max %d), %d ms) for %s\n", - numloops, recheck_tries, - recheck_time, cache_smb_fname->base_name)); - smb_msleep(recheck_time); - retval = SMB_VFS_NEXT_STAT(handle, cache_smb_fname); - numloops++; - } - if (sDIR) - sDIR->recheck_tries_done = numloops; - } - /* still no cachefile, or still too old, return 0 */ - if (retval != 0 - || !timespec_is_newer(&smb_fname->st.st_ex_ctime, - &cache_smb_fname->st.st_ex_ctime)) { - DEBUG(SCANNEDONLY_DEBUG, - ("retval=%d, return 0\n",retval)); - return false; - } - return true; -} - -/*********************/ -/* VFS functions */ -/*********************/ - -static DIR *scannedonly_opendir(vfs_handle_struct * handle, - const char *fname, - const char *mask, uint32_t attr) -{ - DIR *DIRp; - struct scannedonly_DIR *sDIR; - - DIRp = SMB_VFS_NEXT_OPENDIR(handle, fname, mask, attr); - if (!DIRp) { - return NULL; - } - - sDIR = talloc(NULL, struct scannedonly_DIR); - if (fname[0] != '/') { - sDIR->base = construct_full_path(sDIR,handle, fname, true); - } else { - sDIR->base = name_w_ending_slash(sDIR, fname); - } - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_opendir, fname=%s, base=%s\n",fname,sDIR->base)); - sDIR->DIR = DIRp; - sDIR->recheck_tries_done = 0; - return (DIR *) sDIR; -} - -static DIR *scannedonly_fdopendir(vfs_handle_struct * handle, - files_struct *fsp, - const char *mask, uint32_t attr) -{ - DIR *DIRp; - struct scannedonly_DIR *sDIR; - const char *fname; - - DIRp = SMB_VFS_NEXT_FDOPENDIR(handle, fsp, mask, attr); - if (!DIRp) { - return NULL; - } - - fname = (const char *)fsp->fsp_name->base_name; - - sDIR = talloc(NULL, struct scannedonly_DIR); - if (fname[0] != '/') { - sDIR->base = construct_full_path(sDIR,handle, fname, true); - } else { - sDIR->base = name_w_ending_slash(sDIR, fname); - } - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_fdopendir, fname=%s, base=%s\n",fname,sDIR->base)); - sDIR->DIR = DIRp; - sDIR->recheck_tries_done = 0; - return (DIR *) sDIR; -} - - -static struct dirent *scannedonly_readdir(vfs_handle_struct *handle, - DIR * dirp, - SMB_STRUCT_STAT *sbuf) -{ - struct dirent *result; - int allowed = 0; - char *tmp; - struct smb_filename *smb_fname; - char *notify_name; - int namelen; - struct dirent *newdirent; - TALLOC_CTX *ctx=talloc_tos(); - - struct scannedonly_DIR *sDIR = (struct scannedonly_DIR *)dirp; - if (!dirp) { - return NULL; - } - - result = SMB_VFS_NEXT_READDIR(handle, sDIR->DIR, sbuf); - - if (!result) - return NULL; - - if (is_scannedonly_file(STRUCTSCANO(handle->data), result->d_name)) { - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_readdir, %s is a scannedonly file, " - "skip to next entry\n", result->d_name)); - return scannedonly_readdir(handle, dirp, NULL); - } - tmp = talloc_asprintf(ctx, "%s%s", sDIR->base, result->d_name); - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_readdir, check access to %s (sbuf=%p)\n", - tmp,sbuf)); - - /* even if we don't hide nonscanned files or we allow non scanned - files we call allow_access because it will notify the daemon to - scan these files */ - smb_fname = synthetic_smb_fname(ctx, tmp,NULL, - sbuf?VALID_STAT(*sbuf)?sbuf:NULL:NULL); - allowed = scannedonly_allow_access( - handle, sDIR, smb_fname, - result->d_name, - sDIR->base, 0, - STRUCTSCANO(handle->data)->hide_nonscanned_files - ? STRUCTSCANO(handle->data)->recheck_time_readdir - : 0, - STRUCTSCANO(handle->data)->recheck_tries_readdir, - -1, - 1); - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_readdir access to %s (%s) = %d\n", tmp, - result->d_name, allowed)); - if (allowed) { - return result; - } - DEBUG(SCANNEDONLY_DEBUG, - ("hide_nonscanned_files=%d, allow_nonscanned_files=%d\n", - STRUCTSCANO(handle->data)->hide_nonscanned_files, - STRUCTSCANO(handle->data)->allow_nonscanned_files - )); - - if (!STRUCTSCANO(handle->data)->hide_nonscanned_files - || STRUCTSCANO(handle->data)->allow_nonscanned_files) { - return result; - } - - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_readdir, readdir listing for %s not " - "allowed, notify user\n", result->d_name)); - notify_name = talloc_asprintf( - ctx,"%s %s",result->d_name, - STRUCTSCANO(handle->data)->scanning_message); - namelen = strlen(notify_name); - newdirent = (struct dirent *)talloc_array( - ctx, char, sizeof(struct dirent) + namelen + 1); - if (!newdirent) { - return NULL; - } - memcpy(newdirent, result, sizeof(struct dirent)); - memcpy(&newdirent->d_name, notify_name, namelen + 1); - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_readdir, return newdirent at %p with " - "notification %s\n", newdirent, newdirent->d_name)); - return newdirent; -} - -static void scannedonly_seekdir(struct vfs_handle_struct *handle, - DIR * dirp, long offset) -{ - struct scannedonly_DIR *sDIR = (struct scannedonly_DIR *)dirp; - SMB_VFS_NEXT_SEEKDIR(handle, sDIR->DIR, offset); -} - -static long scannedonly_telldir(struct vfs_handle_struct *handle, - DIR * dirp) -{ - struct scannedonly_DIR *sDIR = (struct scannedonly_DIR *)dirp; - return SMB_VFS_NEXT_TELLDIR(handle, sDIR->DIR); -} - -static void scannedonly_rewinddir(struct vfs_handle_struct *handle, - DIR * dirp) -{ - struct scannedonly_DIR *sDIR = (struct scannedonly_DIR *)dirp; - SMB_VFS_NEXT_REWINDDIR(handle, sDIR->DIR); -} - -static int scannedonly_closedir(vfs_handle_struct * handle, - DIR * dirp) -{ - int retval; - struct scannedonly_DIR *sDIR = (struct scannedonly_DIR *)dirp; - flush_sendbuffer(handle); - retval = SMB_VFS_NEXT_CLOSEDIR(handle, sDIR->DIR); - TALLOC_FREE(sDIR); - return retval; -} - -static int scannedonly_stat(vfs_handle_struct * handle, - struct smb_filename *smb_fname) -{ - int ret; - ret = SMB_VFS_NEXT_STAT(handle, smb_fname); - DEBUG(SCANNEDONLY_DEBUG, ("scannedonly_stat: %s returned %d\n", - smb_fname->base_name, ret)); - if (ret != 0 && errno == ENOENT) { - TALLOC_CTX *ctx=talloc_tos(); - char *test_base_name, *tmp_base_name = smb_fname->base_name; - /* possibly this was a fake name (file is being scanned for - viruses.txt): check for that and create the real name and - stat the real name */ - test_base_name = real_path_from_notify_path( - ctx, - STRUCTSCANO(handle->data), - smb_fname->base_name); - if (test_base_name) { - smb_fname->base_name = test_base_name; - ret = SMB_VFS_NEXT_STAT(handle, smb_fname); - DEBUG(5, ("_stat: %s returned %d\n", - test_base_name, ret)); - smb_fname->base_name = tmp_base_name; - } - } - return ret; -} - -static int scannedonly_lstat(vfs_handle_struct * handle, - struct smb_filename *smb_fname) -{ - int ret; - ret = SMB_VFS_NEXT_LSTAT(handle, smb_fname); - DEBUG(SCANNEDONLY_DEBUG, ("scannedonly_lstat: %s returned %d\n", - smb_fname->base_name, ret)); - if (ret != 0 && errno == ENOENT) { - TALLOC_CTX *ctx=talloc_tos(); - char *test_base_name, *tmp_base_name = smb_fname->base_name; - /* possibly this was a fake name (file is being scanned for - viruses.txt): check for that and create the real name and - stat the real name */ - test_base_name = real_path_from_notify_path( - ctx, STRUCTSCANO(handle->data), smb_fname->base_name); - if (test_base_name) { - smb_fname->base_name = test_base_name; - ret = SMB_VFS_NEXT_LSTAT(handle, smb_fname); - DEBUG(5, ("_lstat: %s returned %d\n", - test_base_name, ret)); - smb_fname->base_name = tmp_base_name; - } - } - return ret; -} - -static int scannedonly_open(vfs_handle_struct * handle, - struct smb_filename *smb_fname, - files_struct * fsp, int flags, mode_t mode) -{ - const char *base; - char *tmp, *shortname; - int allowed, write_access = 0; - TALLOC_CTX *ctx=talloc_tos(); - /* if open for writing ignore it */ - if ((flags & O_ACCMODE) == O_WRONLY) { - return SMB_VFS_NEXT_OPEN(handle, smb_fname, fsp, flags, mode); - } - if ((flags & O_ACCMODE) == O_RDWR) { - write_access = 1; - } - /* check if this file is scanned already */ - tmp = strrchr(smb_fname->base_name, '/'); - if (tmp) { - base = talloc_strndup(ctx,smb_fname->base_name, - (tmp - smb_fname->base_name) + 1); - shortname = tmp + 1; - } else { - base = ""; - shortname = (char *)smb_fname->base_name; - } - allowed = scannedonly_allow_access( - handle, NULL, smb_fname, shortname, - base, - write_access, - STRUCTSCANO(handle->data)->recheck_time_open, - STRUCTSCANO(handle->data)->recheck_tries_open, - STRUCTSCANO(handle->data)->recheck_size_open, - 0); - flush_sendbuffer(handle); - DEBUG(SCANNEDONLY_DEBUG, ("scannedonly_open: allow=%d for %s\n", - allowed, smb_fname->base_name)); - if (allowed - || STRUCTSCANO(handle->data)->allow_nonscanned_files) { - return SMB_VFS_NEXT_OPEN(handle, smb_fname, fsp, flags, mode); - } - errno = EACCES; - return -1; -} - -static int scannedonly_close(vfs_handle_struct * handle, files_struct * fsp) -{ - /* we only have to notify the scanner - for files that were open readwrite or writable. */ - if (fsp->can_write) { - TALLOC_CTX *ctx = talloc_tos(); - notify_scanner(handle, construct_full_path( - ctx,handle, - fsp->fsp_name->base_name,false)); - flush_sendbuffer(handle); - } - return SMB_VFS_NEXT_CLOSE(handle, fsp); -} - -static int scannedonly_rename(vfs_handle_struct * handle, - const struct smb_filename *smb_fname_src, - const struct smb_filename *smb_fname_dst) -{ - /* rename the cache file before we pass the actual rename on */ - struct smb_filename *smb_fname_src_tmp = NULL; - struct smb_filename *smb_fname_dst_tmp = NULL; - char *cachefile_src, *cachefile_dst; - bool needscandst=false; - int ret; - TALLOC_CTX *ctx = talloc_tos(); - - /* Setup temporary smb_filename structs. */ - cachefile_src = cachefile_name_f_fullpath( - ctx, - smb_fname_src->base_name, - STRUCTSCANO(handle->data)->p_scanned); - cachefile_dst = cachefile_name_f_fullpath( - ctx, - smb_fname_dst->base_name, - STRUCTSCANO(handle->data)->p_scanned); - smb_fname_src_tmp = synthetic_smb_fname(ctx, cachefile_src,NULL,NULL); - smb_fname_dst_tmp = synthetic_smb_fname(ctx, cachefile_dst,NULL,NULL); - - ret = SMB_VFS_NEXT_RENAME(handle, smb_fname_src_tmp, smb_fname_dst_tmp); - if (ret == ENOENT) { - needscandst=true; - } else if (ret != 0) { - DEBUG(SCANNEDONLY_DEBUG, - ("failed to rename %s into %s error %d: %s\n", cachefile_src, - cachefile_dst, ret, strerror(ret))); - needscandst=true; - } - ret = SMB_VFS_NEXT_RENAME(handle, smb_fname_src, smb_fname_dst); - if (ret == 0 && needscandst) { - notify_scanner(handle, smb_fname_dst->base_name); - flush_sendbuffer(handle); - } - return ret; -} - -static int scannedonly_unlink(vfs_handle_struct * handle, - const struct smb_filename *smb_fname) -{ - /* unlink the 'scanned' file too */ - struct smb_filename *smb_fname_cache = NULL; - char * cachefile; - TALLOC_CTX *ctx = talloc_tos(); - - cachefile = cachefile_name_f_fullpath( - ctx, - smb_fname->base_name, - STRUCTSCANO(handle->data)->p_scanned); - smb_fname_cache = synthetic_smb_fname(ctx, cachefile,NULL,NULL); - if (SMB_VFS_NEXT_UNLINK(handle, smb_fname_cache) != 0) { - DEBUG(SCANNEDONLY_DEBUG, ("_unlink: failed to unlink %s\n", - smb_fname_cache->base_name)); - } - return SMB_VFS_NEXT_UNLINK(handle, smb_fname); -} - -static int scannedonly_rmdir(vfs_handle_struct * handle, const char *path) -{ - /* if there are only .scanned: .virus: or .failed: files, we delete - those, because the client cannot see them */ - DIR *dirp; - struct dirent *dire; - TALLOC_CTX *ctx = talloc_tos(); - bool only_deletable_files = true, have_files = false; - char *path_w_slash; - - if (!STRUCTSCANO(handle->data)->rm_hidden_files_on_rmdir) - return SMB_VFS_NEXT_RMDIR(handle, path); - - path_w_slash = name_w_ending_slash(ctx,path); - dirp = SMB_VFS_NEXT_OPENDIR(handle, path, NULL, 0); - if (!dirp) { - return -1; - } - while ((dire = SMB_VFS_NEXT_READDIR(handle, dirp, NULL)) != NULL) { - if (ISDOT(dire->d_name) || ISDOTDOT(dire->d_name)) { - continue; - } - have_files = true; - if (!is_scannedonly_file(STRUCTSCANO(handle->data), - dire->d_name)) { - struct smb_filename *smb_fname = NULL; - char *fullpath; - int retval; - - if (STRUCTSCANO(handle->data)->show_special_files) { - only_deletable_files = false; - break; - } - /* stat the file and see if it is a - special file */ - fullpath = talloc_asprintf(ctx, "%s%s", path_w_slash, - dire->d_name); - smb_fname = synthetic_smb_fname(ctx, fullpath, - NULL,NULL); - retval = SMB_VFS_NEXT_STAT(handle, smb_fname); - if (retval == 0 - && S_ISREG(smb_fname->st.st_ex_mode)) { - only_deletable_files = false; - } - TALLOC_FREE(fullpath); - TALLOC_FREE(smb_fname); - break; - } - } - DEBUG(SCANNEDONLY_DEBUG, - ("path=%s, have_files=%d, only_deletable_files=%d\n", - path, have_files, only_deletable_files)); - if (have_files && only_deletable_files) { - DEBUG(SCANNEDONLY_DEBUG, - ("scannedonly_rmdir, remove leftover scannedonly " - "files from %s\n", path_w_slash)); - SMB_VFS_NEXT_REWINDDIR(handle, dirp); - while ((dire = SMB_VFS_NEXT_READDIR(handle, dirp, NULL)) - != NULL) { - char *fullpath; - struct smb_filename *smb_fname = NULL; - if (ISDOT(dire->d_name) || ISDOTDOT(dire->d_name)) { - continue; - } - fullpath = talloc_asprintf(ctx, "%s%s", path_w_slash, - dire->d_name); - smb_fname = synthetic_smb_fname(ctx, fullpath, - NULL,NULL); - DEBUG(SCANNEDONLY_DEBUG, ("unlink %s\n", fullpath)); - SMB_VFS_NEXT_UNLINK(handle, smb_fname); - TALLOC_FREE(fullpath); - TALLOC_FREE(smb_fname); - } - } - SMB_VFS_NEXT_CLOSEDIR(handle, dirp); - return SMB_VFS_NEXT_RMDIR(handle, path); -} - -static void free_scannedonly_data(void **data) -{ - SAFE_FREE(*data); -} - -static int scannedonly_connect(struct vfs_handle_struct *handle, - const char *service, const char *user) -{ - - struct Tscannedonly *so; - - so = SMB_MALLOC_P(struct Tscannedonly); - if (so == NULL) { - errno = ENOMEM; - return -1; - } - handle->data = (void *)so; - handle->free_data = free_scannedonly_data; - so->gsendbuffer[0]='\0'; - so->domain_socket = - lp_parm_bool(SNUM(handle->conn), "scannedonly", - "domain_socket", True); - so->socketname = lp_parm_const_string(SNUM(handle->conn), - "scannedonly", "socketname", - "/var/lib/scannedonly/scan"); - - so->portnum = - lp_parm_int(SNUM(handle->conn), "scannedonly", "portnum", - 2020); - so->scanhost = lp_parm_const_string(SNUM(handle->conn), - "scannedonly", "scanhost", - "localhost"); - - so->show_special_files = - lp_parm_bool(SNUM(handle->conn), "scannedonly", - "show_special_files", True); - so->rm_hidden_files_on_rmdir = - lp_parm_bool(SNUM(handle->conn), "scannedonly", - "rm_hidden_files_on_rmdir", True); - so->hide_nonscanned_files = - lp_parm_bool(SNUM(handle->conn), "scannedonly", - "hide_nonscanned_files", False); - so->allow_nonscanned_files = - lp_parm_bool(SNUM(handle->conn), "scannedonly", - "allow_nonscanned_files", False); - so->scanning_message = lp_parm_const_string(SNUM(handle->conn), - "scannedonly", - "scanning_message", - "is being scanned for viruses"); - so->scanning_message_len = strlen(so->scanning_message); - so->recheck_time_open = - lp_parm_int(SNUM(handle->conn), "scannedonly", - "recheck_time_open", 50); - so->recheck_tries_open = - lp_parm_int(SNUM(handle->conn), "scannedonly", - "recheck_tries_open", 100); - so->recheck_size_open = - lp_parm_int(SNUM(handle->conn), "scannedonly", - "recheck_size_open", 100); - so->recheck_time_readdir = - lp_parm_int(SNUM(handle->conn), "scannedonly", - "recheck_time_readdir", 50); - so->recheck_tries_readdir = - lp_parm_int(SNUM(handle->conn), "scannedonly", - "recheck_tries_readdir", 20); - - so->p_scanned = - lp_parm_const_string(SNUM(handle->conn), - "scannedonly", - "pref_scanned", - ".scanned:"); - so->p_virus = - lp_parm_const_string(SNUM(handle->conn), - "scannedonly", - "pref_virus", - ".virus:"); - so->p_failed = - lp_parm_const_string(SNUM(handle->conn), - "scannedonly", - "pref_failed", - ".failed:"); - connect_to_scanner(handle); - - return SMB_VFS_NEXT_CONNECT(handle, service, user); -} - -/* VFS operations structure */ -static struct vfs_fn_pointers vfs_scannedonly_fns = { - .opendir_fn = scannedonly_opendir, - .fdopendir_fn = scannedonly_fdopendir, - .readdir_fn = scannedonly_readdir, - .seekdir_fn = scannedonly_seekdir, - .telldir_fn = scannedonly_telldir, - .rewind_dir_fn = scannedonly_rewinddir, - .closedir_fn = scannedonly_closedir, - .rmdir_fn = scannedonly_rmdir, - .stat_fn = scannedonly_stat, - .lstat_fn = scannedonly_lstat, - .open_fn = scannedonly_open, - .close_fn = scannedonly_close, - .rename_fn = scannedonly_rename, - .unlink_fn = scannedonly_unlink, - .connect_fn = scannedonly_connect -}; - -static_decl_vfs; -NTSTATUS vfs_scannedonly_init(void) -{ - return smb_register_vfs(SMB_VFS_INTERFACE_VERSION, "scannedonly", - &vfs_scannedonly_fns); -} diff --git a/source3/modules/wscript_build b/source3/modules/wscript_build index e6576ac2462..fef412a5f31 100644 --- a/source3/modules/wscript_build +++ b/source3/modules/wscript_build @@ -385,14 +385,6 @@ bld.SAMBA3_MODULE('vfs_dirsort', internal_module=bld.SAMBA3_IS_STATIC_MODULE('vfs_dirsort'), enabled=bld.SAMBA3_IS_ENABLED_MODULE('vfs_dirsort')) -bld.SAMBA3_MODULE('vfs_scannedonly', - subsystem='vfs', - source='vfs_scannedonly.c', - deps='samba-util', - init_function='', - internal_module=bld.SAMBA3_IS_STATIC_MODULE('vfs_scannedonly'), - enabled=bld.SAMBA3_IS_ENABLED_MODULE('vfs_scannedonly')) - bld.SAMBA3_MODULE('vfs_crossrename', subsystem='vfs', source='vfs_crossrename.c', diff --git a/source3/wscript b/source3/wscript index 4918951476d..566adfbbb6e 100644 --- a/source3/wscript +++ b/source3/wscript @@ -1607,7 +1607,7 @@ main() { vfs_expand_msdfs vfs_shadow_copy vfs_shadow_copy2 vfs_readahead vfs_xattr_tdb vfs_posix_eadb vfs_streams_xattr vfs_streams_depot vfs_acl_xattr vfs_acl_tdb - vfs_smb_traffic_analyzer vfs_preopen vfs_catia vfs_scannedonly + vfs_smb_traffic_analyzer vfs_preopen vfs_catia vfs_media_harmony vfs_unityed_media vfs_fruit vfs_shell_snap vfs_commit vfs_worm vfs_crossrename vfs_linux_xfs_sgid vfs_time_audit |