mirror of
https://github.com/samba-team/samba.git
synced 2024-12-24 21:34:56 +03:00
94fcbec8af
The next commit will change the Samba Spotlight server to return absolute paths that start with the sharename as "/SHARENAME/..." followed by the share path relative appended. So given a share [spotlight] path = /foo/bar spotlight = yes and a file inside this share with a full path of /foo/bar/dir/file previously a search that matched this file would returns the absolute server-side pato of the file, ie /foo/bar/dir/file This will be change to /spotlight/dir/file As currently the mdscli library and hence the mdsearch tool print out these paths returned from the server, we have to change the output to accomodate these fake paths. The only way to do this sensibly is by makeing the paths relative to the containing share, so just dir/file in the example above. The client learns about the share root path prefix – real server-side of fake in the future – in an initial handshake in the "share_path" out argument of the mdssvc_open() RPC call, so the client can use this path to convert the absolute path to relative. There is however an additional twist: the macOS Spotlight server prefixes this absolute path with another prefix, typically "/System/Volumes/Data", so in the example above the full path for the same search would be /System/Volumes/Data/foo/bar/dir/file So macOS does return the full server-side path too, just prefixed with an additional path. This path prefixed can be queried by the client in the mdssvc_cmd() RPC call with an Spotlight command of "fetchPropertiesForContext:" and the path is returned in a dictionary with key "kMDSStorePathScopes". Samba just returns "/" for this. Currently the mdscli library doesn't issue this Spotlight RPC request (fetchPropertiesForContext), so this is added in this commit. In the end, all search result paths are stripped of the combined prefix kMDSStorePathScopes + share_path (from mdssvc_open). eg kMDSStorePathScopes = /System/Volumes/Data share_path = /foo/bar search result = /System/Volumes/Data/foo/bar/dir/file relative path returned by mdscli = dir/file Makes sense? :) BUG: https://bugzilla.samba.org/show_bug.cgi?id=15388 Signed-off-by: Ralph Boehme <slow@samba.org> Reviewed-by: Stefan Metzmacher <metze@samba.org>
552 lines
12 KiB
C
552 lines
12 KiB
C
/*
|
|
Unix SMB/CIFS implementation.
|
|
Main metadata server / Spotlight client functions
|
|
|
|
Copyright (C) Ralph Boehme 2019
|
|
|
|
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 "rpc_client.h"
|
|
#include "librpc/gen_ndr/mdssvc.h"
|
|
#include "cli_mdssvc.h"
|
|
#include "cli_mdssvc_private.h"
|
|
#include "cli_mdssvc_util.h"
|
|
#include "lib/util/tevent_ntstatus.h"
|
|
#include "rpc_server/mdssvc/dalloc.h"
|
|
#include "rpc_server/mdssvc/marshalling.h"
|
|
|
|
NTSTATUS mdscli_blob_fetch_props(TALLOC_CTX *mem_ctx,
|
|
struct mdscli_ctx *ctx,
|
|
struct mdssvc_blob *blob)
|
|
{
|
|
DALLOC_CTX *d = NULL;
|
|
uint64_t *uint64p = NULL;
|
|
sl_array_t *array = NULL;
|
|
sl_array_t *cmd_array = NULL;
|
|
NTSTATUS status;
|
|
int ret;
|
|
|
|
d = dalloc_new(mem_ctx);
|
|
if (d == NULL) {
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
array = dalloc_zero(d, sl_array_t);
|
|
if (array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(d, array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
cmd_array = dalloc_zero(d, sl_array_t);
|
|
if (cmd_array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(array, cmd_array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(cmd_array, "fetchPropertiesForContext:");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
uint64p = talloc_zero_array(cmd_array, uint64_t, 2);
|
|
if (uint64p == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
talloc_set_name(uint64p, "uint64_t *");
|
|
|
|
ret = dalloc_add(cmd_array, uint64p, uint64_t *);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
status = sl_pack_alloc(mem_ctx, d, blob, ctx->max_fragment_size);
|
|
TALLOC_FREE(d);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
return status;
|
|
}
|
|
return NT_STATUS_OK;
|
|
}
|
|
|
|
NTSTATUS mdscli_blob_search(TALLOC_CTX *mem_ctx,
|
|
struct mdscli_search_ctx *search,
|
|
struct mdssvc_blob *blob)
|
|
{
|
|
struct mdscli_ctx *ctx = search->mdscli_ctx;
|
|
DALLOC_CTX *d = NULL;
|
|
uint64_t *uint64p = NULL;
|
|
sl_array_t *array = NULL;
|
|
sl_array_t *cmd_array = NULL;
|
|
sl_dict_t *query_dict = NULL;
|
|
sl_array_t *attr_array = NULL;
|
|
sl_array_t *scope_array = NULL;
|
|
double dval;
|
|
uint64_t uint64val;
|
|
NTSTATUS status;
|
|
int ret;
|
|
|
|
d = dalloc_new(mem_ctx);
|
|
if (d == NULL) {
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
array = dalloc_zero(d, sl_array_t);
|
|
if (array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(d, array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
cmd_array = dalloc_zero(d, sl_array_t);
|
|
if (cmd_array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(array, cmd_array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(cmd_array, "openQueryWithParams:forContext:");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
uint64p = talloc_zero_array(cmd_array, uint64_t, 2);
|
|
if (uint64p == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
talloc_set_name(uint64p, "uint64_t *");
|
|
|
|
uint64p[0] = search->ctx_id.id;
|
|
uint64p[1] = search->ctx_id.connection;
|
|
|
|
ret = dalloc_add(cmd_array, uint64p, uint64_t *);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
query_dict = dalloc_zero(array, sl_dict_t);
|
|
if (query_dict == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(array, query_dict, sl_dict_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(query_dict, "kMDQueryBatchFirstDelay");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
dval = 1;
|
|
ret = dalloc_add_copy(query_dict, &dval, double);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(query_dict, "kMDQueryUniqueId");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
ret = dalloc_add_copy(query_dict, &search->unique_id, uint64_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(query_dict, "kMDAttributeArray");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
attr_array = dalloc_zero(query_dict, sl_array_t);
|
|
if (attr_array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
ret = dalloc_add(query_dict, attr_array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
ret = dalloc_stradd(attr_array, "kMDItemFSName");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(query_dict, "kMDQueryBatchFirstCount");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
uint64val = 10;
|
|
ret = dalloc_add_copy(query_dict, &uint64val, uint64_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(query_dict, "kMDQueryBatchUpdateCount");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
uint64val = 100;
|
|
ret = dalloc_add_copy(query_dict, &uint64val, uint64_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(query_dict, "kMDQueryString");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
ret = dalloc_stradd(query_dict, search->mds_query);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(query_dict, "kMDScopeArray");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
scope_array = dalloc_zero(query_dict, sl_array_t);
|
|
if (scope_array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
ret = dalloc_add(query_dict, scope_array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
ret = dalloc_stradd(scope_array, search->path_scope);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
status = sl_pack_alloc(mem_ctx, d, blob, ctx->max_fragment_size);
|
|
TALLOC_FREE(d);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
return status;
|
|
}
|
|
return NT_STATUS_OK;
|
|
}
|
|
|
|
NTSTATUS mdscli_blob_get_results(TALLOC_CTX *mem_ctx,
|
|
struct mdscli_search_ctx *search,
|
|
struct mdssvc_blob *blob)
|
|
{
|
|
struct mdscli_ctx *ctx = search->mdscli_ctx;
|
|
DALLOC_CTX *d = NULL;
|
|
uint64_t *uint64p = NULL;
|
|
sl_array_t *array = NULL;
|
|
sl_array_t *cmd_array = NULL;
|
|
NTSTATUS status;
|
|
int ret;
|
|
|
|
d = dalloc_new(mem_ctx);
|
|
if (d == NULL) {
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
array = dalloc_zero(d, sl_array_t);
|
|
if (array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(d, array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
cmd_array = dalloc_zero(d, sl_array_t);
|
|
if (cmd_array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(array, cmd_array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(cmd_array, "fetchQueryResultsForContext:");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
uint64p = talloc_zero_array(cmd_array, uint64_t, 2);
|
|
if (uint64p == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
talloc_set_name(uint64p, "uint64_t *");
|
|
|
|
uint64p[0] = search->ctx_id.id;
|
|
uint64p[1] = search->ctx_id.connection;
|
|
|
|
ret = dalloc_add(cmd_array, uint64p, uint64_t *);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
status = sl_pack_alloc(mem_ctx, d, blob, ctx->max_fragment_size);
|
|
TALLOC_FREE(d);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
return status;
|
|
}
|
|
return NT_STATUS_OK;
|
|
}
|
|
|
|
NTSTATUS mdscli_blob_get_path(TALLOC_CTX *mem_ctx,
|
|
struct mdscli_ctx *ctx,
|
|
uint64_t cnid,
|
|
struct mdssvc_blob *blob)
|
|
{
|
|
struct mdsctx_id ctx_id = mdscli_new_ctx_id(ctx);
|
|
DALLOC_CTX *d = NULL;
|
|
uint64_t *uint64var = NULL;
|
|
sl_array_t *array = NULL;
|
|
sl_array_t *cmd_array = NULL;
|
|
sl_array_t *attr_array = NULL;
|
|
sl_cnids_t *cnids = NULL;
|
|
NTSTATUS status;
|
|
int ret;
|
|
|
|
d = dalloc_new(mem_ctx);
|
|
if (d == NULL) {
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
array = dalloc_zero(d, sl_array_t);
|
|
if (array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(d, array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
cmd_array = dalloc_zero(d, sl_array_t);
|
|
if (cmd_array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(array, cmd_array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(cmd_array, "fetchAttributes:forOIDArray:context:");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
uint64var = talloc_zero_array(cmd_array, uint64_t, 2);
|
|
if (uint64var == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
talloc_set_name(uint64var, "uint64_t *");
|
|
|
|
uint64var[0] = ctx_id.id;
|
|
uint64var[1] = 0;
|
|
|
|
ret = dalloc_add(cmd_array, &uint64var[0], uint64_t *);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
attr_array = dalloc_zero(d, sl_array_t);
|
|
if (attr_array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(array, attr_array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(attr_array, "kMDItemPath");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
/* CNIDs */
|
|
cnids = talloc_zero(array, sl_cnids_t);
|
|
if (cnids == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
cnids->ca_cnids = dalloc_new(cnids);
|
|
if (cnids->ca_cnids == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
cnids->ca_unkn1 = 0xadd;
|
|
cnids->ca_context = 0x6b000020;
|
|
|
|
ret = dalloc_add_copy(cnids->ca_cnids, &cnid, uint64_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(array, cnids, sl_cnids_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
status = sl_pack_alloc(mem_ctx, d, blob, ctx->max_fragment_size);
|
|
TALLOC_FREE(d);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
return status;
|
|
}
|
|
return NT_STATUS_OK;
|
|
}
|
|
|
|
NTSTATUS mdscli_blob_close_search(TALLOC_CTX *mem_ctx,
|
|
struct mdscli_search_ctx *search,
|
|
struct mdssvc_blob *blob)
|
|
{
|
|
struct mdscli_ctx *ctx = search->mdscli_ctx;
|
|
DALLOC_CTX *d = NULL;
|
|
uint64_t *uint64p = NULL;
|
|
sl_array_t *array = NULL;
|
|
sl_array_t *cmd_array = NULL;
|
|
NTSTATUS status;
|
|
int ret;
|
|
|
|
d = dalloc_new(mem_ctx);
|
|
if (d == NULL) {
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
array = dalloc_zero(d, sl_array_t);
|
|
if (array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(d, array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
cmd_array = dalloc_zero(d, sl_array_t);
|
|
if (cmd_array == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_add(array, cmd_array, sl_array_t);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
ret = dalloc_stradd(cmd_array, "closeQueryForContext:");
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
uint64p = talloc_zero_array(cmd_array, uint64_t, 2);
|
|
if (uint64p == NULL) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
talloc_set_name(uint64p, "uint64_t *");
|
|
|
|
uint64p[0] = search->ctx_id.id;
|
|
uint64p[1] = search->ctx_id.connection;
|
|
|
|
ret = dalloc_add(cmd_array, uint64p, uint64_t *);
|
|
if (ret != 0) {
|
|
TALLOC_FREE(d);
|
|
return NT_STATUS_NO_MEMORY;
|
|
}
|
|
|
|
status = sl_pack_alloc(mem_ctx, d, blob, ctx->max_fragment_size);
|
|
TALLOC_FREE(d);
|
|
if (!NT_STATUS_IS_OK(status)) {
|
|
return status;
|
|
}
|
|
return NT_STATUS_OK;
|
|
}
|