1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-11 05:18:09 +03:00

s3/utils: Add search client

Simple cli client for doing a basic windows search.

example:

  wspsearch -U$(USER)%$(PASSWD) //$(SERVER)/$(SHARE) --search='DSC' --kind=Picture

Signed-off-by: Noel Power <noel.power@suse.com>
Reviewed-by: Andrew Bartlett <abartlet@samba.org>
This commit is contained in:
Noel Power 2016-07-21 16:53:17 +01:00 committed by Andrew Bartlett
parent 387c9f3aa5
commit a3a7a94ff4
2 changed files with 837 additions and 0 deletions

View File

@ -337,6 +337,23 @@ bld.SAMBA3_BINARY('mdsearch',
mdssvc mdssvc
''') ''')
bld.SAMBA3_BINARY('wspsearch',
source='wspsearch.c',
deps='''
talloc
tevent
smbconf
CMDLINE_S3
cmdline_contexts
libsmb
msrpc3
LIBSAMBA_WSP
RPCCLI_WSP
WSP_UTIL
dcerpc
''',
enabled=bld.env.with_wsp)
pytalloc_util = bld.pyembed_libname('pytalloc-util') pytalloc_util = bld.pyembed_libname('pytalloc-util')
pyrpc_util = bld.pyembed_libname('pyrpc_util') pyrpc_util = bld.pyembed_libname('pyrpc_util')
bld.SAMBA3_PYTHON('python_net_s3', bld.SAMBA3_PYTHON('python_net_s3',

820
source3/utils/wspsearch.c Normal file
View File

@ -0,0 +1,820 @@
/*
* Unix SMB/CIFS implementation.
*
* Window Search Service
*
* Copyright (c) Noel Power
*
* 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, see <http://www.gnu.org/licenses/>.
*/
#include "includes.h"
#include "lib/util/debug.h"
#include "lib/cmdline/cmdline.h"
#include "lib/cmdline_contexts.h"
#include "param.h"
#include "client.h"
#include "libsmb/proto.h"
#include "librpc/rpc/rpc_common.h"
#include "librpc/wsp/wsp_util.h"
#include "rpc_client/cli_pipe.h"
#include "rpc_client/wsp_cli.h"
#include "libcli/wsp/wsp_aqs.h"
#include "librpc/gen_ndr/ndr_wsp.h"
#include "librpc/gen_ndr/ndr_wsp_data.h"
#include "dcerpc.h"
#define WIN_VERSION_64 0x10000
/* send connectin message */
static NTSTATUS wsp_connect(TALLOC_CTX *ctx,
struct wsp_client_ctx *wsp_ctx,
const char* clientmachine,
const char* clientuser,
const char* server,
bool *is_64bit)
{
struct wsp_request *request = NULL;
struct wsp_response *response = NULL;
uint32_t client_ver;
uint32_t server_ver;
DATA_BLOB unread = data_blob_null;
NTSTATUS status;
TALLOC_CTX *local_ctx = talloc_new(ctx);
if (local_ctx == NULL) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
response = talloc_zero(local_ctx, struct wsp_response);
if (!response) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
request = talloc_zero(local_ctx, struct wsp_request);
if (!request) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
if (!init_connectin_request(local_ctx, request,
clientmachine, clientuser, server)) {
DBG_ERR("Failed in initialise connection message\n");
status = NT_STATUS_INVALID_PARAMETER;
goto out;
}
status = wsp_request_response(local_ctx, wsp_ctx,
request, response, &unread);
if (NT_STATUS_IS_OK(status)) {
client_ver = request->message.cpmconnect.iclientversion;
server_ver = response->message.cpmconnect.server_version;
*is_64bit =
(server_ver & WIN_VERSION_64)
&& (client_ver & WIN_VERSION_64);
}
out:
data_blob_free(&unread);
TALLOC_FREE(local_ctx);
return status;
}
static NTSTATUS create_query(TALLOC_CTX *ctx,
struct wsp_client_ctx *wsp_ctx,
uint32_t limit,
t_select_stmt *select,
uint32_t *single_cursor)
{
struct wsp_request *request = NULL;
struct wsp_response *response = NULL;
NTSTATUS status;
DATA_BLOB unread = data_blob_null;
TALLOC_CTX *local_ctx = talloc_new(ctx);
if (local_ctx == NULL) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
request = talloc_zero(local_ctx, struct wsp_request);
if (!request) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
response = talloc_zero(local_ctx, struct wsp_response);
if (!response) {
status = NT_STATUS_NO_MEMORY;
goto out;;
}
if (!create_querysearch_request(ctx, request, select)) {
DBG_ERR("error setting up query request message\n");
status = NT_STATUS_INVALID_PARAMETER;
goto out;
}
request->message.cpmcreatequery.rowsetproperties.cmaxresults = limit;
status = wsp_request_response(local_ctx,
wsp_ctx,
request,
response,
&unread);
if (NT_STATUS_IS_OK(status)) {
if (unread.length == 4) {
*single_cursor = IVAL(unread.data, 0);
}
}
out:
data_blob_free(&unread);
TALLOC_FREE(local_ctx);
return status;
}
static NTSTATUS create_bindings(TALLOC_CTX *ctx,
struct wsp_client_ctx *wsp_ctx,
t_select_stmt *select,
uint32_t cursor,
struct wsp_cpmsetbindingsin *bindings_out,
bool is_64bit)
{
struct wsp_request *request = NULL;
struct wsp_response *response = NULL;
NTSTATUS status;
DATA_BLOB unread = data_blob_null;
request = talloc_zero(ctx, struct wsp_request);
if (!request) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
response = talloc_zero(ctx, struct wsp_response);
if (!response) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
if (!create_setbindings_request(ctx,
request,
select,
cursor,
is_64bit)) {
DBG_ERR("Failed to create setbindings message\n");
status = NT_STATUS_INVALID_PARAMETER;
goto out;
}
status = wsp_request_response(ctx,
wsp_ctx,
request,
response,
&unread);
if (NT_STATUS_IS_OK(status)) {
*bindings_out = request->message.cpmsetbindings;
}
out:
data_blob_free(&unread);
return status;
}
static NTSTATUS create_querystatusex(TALLOC_CTX *ctx,
struct wsp_client_ctx *wsp_ctx,
uint32_t cursor,
uint32_t *nrows)
{
struct wsp_request *request = NULL;
struct wsp_response *response = NULL;
struct wsp_cpmgetquerystatusexin *statusexin = NULL;
NTSTATUS status;
DATA_BLOB unread = data_blob_null;
TALLOC_CTX *local_ctx = talloc_new(ctx);
if (local_ctx == NULL) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
request = talloc_zero(local_ctx, struct wsp_request);
if (!request) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
response = talloc_zero(local_ctx, struct wsp_response);
if (!response) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
statusexin = &request->message.cpmgetquerystatusex;
request->header.msg = CPMGETQUERYSTATUSEX;
statusexin->hcursor = cursor;
statusexin->bmk = 0xfffffffc;
status = wsp_request_response(local_ctx,
wsp_ctx,
request,
response,
&unread);
if (NT_STATUS_IS_OK(status)) {
*nrows = response->message.cpmgetquerystatusex.resultsfound;
}
out:
data_blob_free(&unread);
TALLOC_FREE(local_ctx);
return status;
}
static NTSTATUS print_rowsreturned(
TALLOC_CTX *ctx,
DATA_BLOB *buffer,
bool is_64bit,
bool disp_all_cols,
struct wsp_cpmsetbindingsin *bindings,
uint32_t cbreserved,
uint64_t address,
uint32_t rowsreturned,
uint32_t *rows_processed)
{
NTSTATUS status;
int row;
TALLOC_CTX *local_ctx = NULL;
struct wsp_cbasestoragevariant **rowsarray = NULL;
enum ndr_err_code err;
local_ctx = talloc_init("results");
if (local_ctx == NULL) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
rowsarray = talloc_zero_array(local_ctx,
struct wsp_cbasestoragevariant*,
rowsreturned);
if (rowsarray == NULL) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
err = extract_rowsarray(rowsarray,
buffer,
is_64bit,
bindings,
cbreserved,
address,
rowsreturned,
rowsarray);
if (err) {
DBG_ERR("failed to extract rows from getrows response\n");
status = NT_STATUS_UNSUCCESSFUL;
goto out;
}
for(row = 0; row < rowsreturned; row++) {
TALLOC_CTX *row_ctx = NULL;
const char *col_str = NULL;
row_ctx = talloc_init("row");
if (row_ctx == NULL) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
if (disp_all_cols) {
int i;
for (i = 0; i < bindings->ccolumns; i++){
col_str =
variant_as_string(
row_ctx,
&rowsarray[row][i],
true);
if (col_str) {
printf("%s%s",
i ? ", " : "", col_str);
} else {
printf("%sN/A",
i ? ", " : "");
}
}
} else {
col_str = variant_as_string(
row_ctx,
&rowsarray[row][0],
true);
printf("%s", col_str);
}
printf("\n");
TALLOC_FREE(row_ctx);
}
status = NT_STATUS_OK;
out:
TALLOC_FREE(local_ctx);
*rows_processed = row;
return status;
}
static NTSTATUS create_getrows(TALLOC_CTX *ctx,
struct wsp_client_ctx *wsp_ctx,
struct wsp_cpmsetbindingsin *bindings,
uint32_t cursor,
uint32_t nrows,
bool disp_all_cols,
bool is_64bit)
{
struct wsp_request *request = NULL;
struct wsp_response *response = NULL;
NTSTATUS status;
DATA_BLOB unread = data_blob_null;
uint32_t bmk = 0xfffffffc;
uint32_t skip = 0;
uint32_t total_rows = 0;
uint32_t INITIAL_ROWS = 32;
uint32_t requested_rows = INITIAL_ROWS;
uint32_t rows_printed;
uint32_t current_row = 0;
TALLOC_CTX *row_ctx;
bool loop_again;
do {
row_ctx = talloc_new(NULL);
if (!row_ctx) {
status = NT_STATUS_UNSUCCESSFUL;
goto out;
}
request = talloc_zero(row_ctx, struct wsp_request);
if (!request) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
response = talloc_zero(row_ctx, struct wsp_response);
if (!response) {
status = NT_STATUS_NO_MEMORY;
goto out;
}
create_seekat_getrows_request(request,
request,
cursor,
bmk,
skip,
requested_rows,
40,
0xDEAbd860,
bindings->brow,
0);
status = wsp_request_response(request,
wsp_ctx,
request,
response,
&unread);
if (!NT_STATUS_IS_OK(status)) {
goto out;
}
total_rows += response->message.cpmgetrows.rowsreturned;
if (response->message.cpmgetrows.rowsreturned
!= requested_rows) {
uint32_t rowsreturned =
response->message.cpmgetrows.rowsreturned;
if (response->message.cpmgetrows.etype == EROWSEEKAT) {
struct wsp_cpmgetrowsout *resp;
struct wsp_crowseekat *seekat;
resp = &response->message.cpmgetrows;
seekat =
&resp->seekdescription.crowseekat;
bmk = seekat->bmkoffset;
skip = seekat->cskip;
} else {
bmk = 0xfffffffc;
skip = total_rows;
}
requested_rows = requested_rows - rowsreturned;
} else {
requested_rows = INITIAL_ROWS;
bmk = 0xfffffffc;
skip = total_rows;
}
if (response->message.cpmgetrows.rowsreturned) {
status = print_rowsreturned(row_ctx, &unread,
is_64bit,
disp_all_cols,
bindings, 40,
0xDEAbd860,
response->message.cpmgetrows.rowsreturned,
&rows_printed);
if (!NT_STATUS_IS_OK(status)) {
goto out;
}
current_row += rows_printed;
data_blob_free(&unread);
}
/*
* response is a talloc child of row_ctz so we need to
* assign loop_again before we delete row_ctx
*/
loop_again = response->message.cpmgetrows.rowsreturned;
TALLOC_FREE(row_ctx);
if (nrows && total_rows > nrows) {
DBG_ERR("Something is wrong, results returned %d "
"exceed expected number of results %d\n",
total_rows, nrows);
status = NT_STATUS_UNSUCCESSFUL;
goto out;
}
} while (loop_again);
out:
data_blob_free(&unread);
TALLOC_FREE(row_ctx);
return status;
}
const char *default_column = "System.ItemUrl";
static bool is_valid_kind(const char *kind)
{
const char* kinds[] = {"calendar",
"communication",
"contact",
"document",
"email",
"feed",
"folder",
"game",
"instantMessage",
"journal",
"link",
"movie",
"music",
"note",
"picture",
"program",
"recordedtv",
"searchfolder",
"task",
"video",
"webhistory"};
char* search_kind = NULL;
int i;
bool found = false;
search_kind = strlower_talloc(NULL, kind);
if (search_kind == NULL) {
DBG_ERR("couldn't convert %s to lower case\n",
kind);
return NULL;
}
for (i=0; i<ARRAY_SIZE(kinds); i++) {
if (strequal(search_kind, kinds[i])) {
found = true;
break;
}
}
if (found == false) {
DBG_ERR("Invalid kind %s\n", kind);
}
TALLOC_FREE(search_kind);
return found;
}
static char * build_default_sql(TALLOC_CTX *ctx,
const char *kind,
const char *phrase,
const char *location)
{
char *sql = NULL;
/* match what windows clients do */
sql = talloc_asprintf(ctx,
"Scope:\"%s\" AND NOT System.Shell.SFGAOFlagsStrings:hidden"
" AND NOT System.Shell.OmitFromView:true", location);
if (kind) {
if (!is_valid_kind(kind)) {
return NULL;
}
sql = talloc_asprintf(ctx, "System.Kind:%s AND %s",
kind, sql);
}
if (phrase) {
sql = talloc_asprintf(ctx,
"All:$=\"%s\" OR All:$<\"%s\""
" AND %s", phrase, phrase, sql);
}
sql = talloc_asprintf(ctx, "SELECT %s"
" WHERE %s", default_column, sql);
return sql;
}
int main(int argc, char **argv)
{
int opt;
int result = 0;
NTSTATUS status = NT_STATUS_OK;
poptContext pc;
char* server = NULL;
char* share = NULL;
char* path = NULL;
char* location = NULL;
char* query = NULL;
bool custom_query = false;
const char* phrase = NULL;
const char* kind = NULL;
uint32_t limit = 500;
uint32_t nrows = 0;
struct wsp_cpmsetbindingsin bindings_used = {0};
bool is_64bit = false;
struct poptOption long_options[] = {
POPT_AUTOHELP
{ "limit",
0,
POPT_ARG_INT,
&limit,
0,
"limit results",
"default is 500, specifying 0 means unlimited" },
{ "search",
0,
POPT_ARG_STRING,
&phrase,
0,
"Search phrase",
"phrase" },
{ "kind", 0, POPT_ARG_STRING, &kind, 0,
"Kind of thing to search for [Calendar|Communication|"
"Contact|Document|Email|Feed|Folder|Game|"
"InstantMessage|Journal|Link|Movie|Music|Note|Picture|"
"Program|RecordedTV|SearchFolder|Task|Video"
"|WebHistory]",
"kind" },
{ "query",
0,
POPT_ARG_STRING,
&query,
0,
"specify a more complex query",
"query" },
POPT_COMMON_SAMBA
POPT_COMMON_CONNECTION
POPT_COMMON_CREDENTIALS
POPT_TABLEEND
};
TALLOC_CTX *frame = talloc_stackframe();
struct tevent_context *ev_ctx
= samba_tevent_context_init(talloc_tos());
uint32_t cursor = 0;
struct wsp_client_ctx *wsp_ctx = NULL;
t_select_stmt *select_stmt = NULL;
const char **const_argv = discard_const_p(const char *, argv);
struct dcerpc_binding_handle *h = NULL;
struct cli_state *c = NULL;
uint32_t flags = CLI_FULL_CONNECTION_IPC;
samba_cmdline_init(frame,
SAMBA_CMDLINE_CONFIG_CLIENT,
false /* require_smbconf */);
pc = samba_popt_get_context("wspsearch",
argc,
const_argv,
long_options,
0);
poptSetOtherOptionHelp(pc, "[OPTIONS] //server1/share1");
while ((opt = poptGetNextOpt(pc)) != -1) ;
if(!poptPeekArg(pc)) {
poptPrintUsage(pc, stderr, 0);
result = -1;
goto out;
}
path = talloc_strdup(talloc_tos(), poptGetArg(pc));
if (!path || limit < 0) {
DBG_ERR("Invalid argument\n");
result = -1;
goto out;
}
string_replace(path,'/','\\');
server = talloc_strdup(talloc_tos(), path+2);
if (!server) {
DBG_ERR("Invalid argument\n");
return -1;
}
if (server) {
/*
* if we specify --query then we don't need actually need the
* share part, if it is specified then we don't care as we
* expect the scope to be part of the query (and if it isn't
* then it will probably fail anyway)
*/
share = strchr_m(server,'\\');
if (!query && !share) {
DBG_ERR("Invalid argument\n");
return -1;
}
if (share) {
*share = 0;
share++;
}
}
DBG_INFO("server name is %s\n", server ? server : "N/A");
DBG_INFO("share name is %s\n", share ? share : "N/A");
DBG_INFO("search phrase is %s\n", phrase ? phrase : "N/A");
DBG_INFO("search kind is %s\n", kind ? kind : "N/A");
if (!query && (kind == NULL && phrase == NULL)) {
poptPrintUsage(pc, stderr, 0);
result = -1;
goto out;
}
if (!query) {
location = talloc_asprintf(talloc_tos(),
"FILE://%s/%s", server, share);
query = build_default_sql(talloc_tos(), kind, phrase, location);
if (!query) {
result = -1;
goto out;
}
} else {
custom_query = true;
}
printf("custom_query %d\n", custom_query);
select_stmt = get_wsp_sql_tree(query);
poptFreeContext(pc);
if (select_stmt == NULL) {
DBG_ERR("query failed\n");
result = -1;
goto out;
}
if (select_stmt->cols == NULL) {
select_stmt->cols = talloc_zero(select_stmt, t_col_list);
if (select_stmt->cols == NULL) {
DBG_ERR("out of memory\n");
result = -1;
goto out;
}
select_stmt->cols->num_cols = 1;
select_stmt->cols->cols =
talloc_zero_array(select_stmt->cols, char*, 1);
if (select_stmt->cols->cols == NULL) {
DBG_ERR("out of memory\n");
result = -1;
goto out;
}
select_stmt->cols->cols[0] =
talloc_strdup(select_stmt->cols, default_column);
}
status = cli_full_connection_creds(&c,
lp_netbios_name(),
server,
NULL,
0,
"IPC$",
"IPC",
samba_cmdline_get_creds(),
flags);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("failed to connect to IPC$: %s\n",
nt_errstr(status));
result = -1;
goto out;
}
status = wsp_server_connect(talloc_tos(),
server,
ev_ctx,
samba_cmdline_get_lp_ctx(),
samba_cmdline_get_creds(),
c,
&wsp_ctx);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("failed to connect to wsp: %s\n",
nt_errstr(status));
result = -1;
goto out;
}
h = get_wsp_pipe(wsp_ctx);
if (h == NULL) {
DBG_ERR("Failed to communicate with server, no pipe\n");
result = -1;
goto out;
}
dcerpc_binding_handle_set_timeout(h,
DCERPC_REQUEST_TIMEOUT * 1000);
/* connect */
DBG_INFO("sending connect\n");
status = wsp_connect(talloc_tos(),
wsp_ctx,
lpcfg_netbios_name(samba_cmdline_get_lp_ctx()),
cli_credentials_get_username(
samba_cmdline_get_creds()),
server,
&is_64bit);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("failed to connect to wsp: %s\n",
nt_errstr(status));
result = -1;
goto out;
}
DBG_INFO("sending query\n");
status = create_query(talloc_tos(),
wsp_ctx,
limit,
select_stmt,
&cursor);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("failed to send query: %s)\n",
nt_errstr(status));
result = -1;
goto out;
}
DBG_INFO("sending createbindings\n");
/* set bindings */
status = create_bindings(talloc_tos(),
wsp_ctx,
select_stmt,
cursor,
&bindings_used,
is_64bit);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("failed to setbindings: %s)\n",
nt_errstr(status));
result = -1;
goto out;
}
status = create_querystatusex(talloc_tos(),
wsp_ctx,
bindings_used.hcursor,
&nrows);
if (!nrows) {
result = 0;
DBG_ERR("no results found\n");
goto out;
}
printf("found %d results, returning %d \n",
nrows,
limit ? MIN(nrows, limit) : nrows);
status = create_getrows(talloc_tos(),
wsp_ctx,
&bindings_used,
bindings_used.hcursor,
limit ? MIN(nrows, limit) : nrows,
custom_query,
is_64bit);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to retrieve rows, error: %s\n",
nt_errstr(status));
result = -1;
goto out;
}
result = 0;
out:
TALLOC_FREE(frame);
return result;
}