স্থির বিষয়বস্তু পরিবেশন করার জন্য সার্লেট


145

আমি দুটি পৃথক পাত্রে (টমক্যাট এবং জেটি) একটি ওয়েব অ্যাপ্লিকেশন স্থাপন করি, তবে স্থির বিষয়বস্তু পরিবেশন করার জন্য তাদের ডিফল্ট সার্লেটগুলিতে যে ইউআরএল কাঠামোটি আমি ব্যবহার করতে চাই তা হ্যান্ডল করার ( ডিটেলস ) আলাদা পদ্ধতি রয়েছে ।

অতএব আমি নিজের স্ট্যাটিক সামগ্রী (চিত্র, সিএসএস, ইত্যাদি) পরিবেশন করতে ওয়েব অ্যাপে একটি ছোট সার্লেটকে অন্তর্ভুক্ত করতে চাইছি। সার্লেলেটের নিম্নলিখিত বৈশিষ্ট্য থাকতে হবে:

  • বাহ্যিক নির্ভরতা নেই
  • সহজ এবং নির্ভরযোগ্য
  • If-Modified-Sinceশিরোলেখের জন্য সমর্থন (যেমন কাস্টম getLastModifiedপদ্ধতি)
  • (Ptionচ্ছিক) জিজিপ এনকোডিং, ইটাগস, ...

কোথাও এই জাতীয় সার্লেট পাওয়া যায়? সর্বাধিক নিকটস্থ আমি সার্লেট বই থেকে 4-10 উদাহরণ খুঁজে পেতে পারি ।

আপডেট: আমি যে URL টির কাঠামোটি ব্যবহার করতে চাইছি - আপনি যদি ভাবছেন তবে এমনটি হ'ল:

    <servlet-mapping>
            <servlet-name>main</servlet-name>
            <url-pattern>/*</url-pattern>
    </servlet-mapping>
    <servlet-mapping>
            <servlet-name>default</servlet-name>
            <url-pattern>/static/*</url-pattern>
    </servlet-mapping>

সুতরাং সমস্ত অনুরোধগুলি মূল সার্লেলে পৌঁছে দেওয়া উচিত, যদি না তারা সেই staticপথে থাকে। সমস্যাটি হ'ল টমকেটের ডিফল্ট সার্লেটটি সার্লেটলেটকে অ্যাকাউন্টে নেয় না (সুতরাং এটি মূল ফোল্ডারে স্থির ফাইলগুলি সন্ধান করে), যখন জেটি (তাই এটি staticফোল্ডারে দেখায় )।


আপনি যে "ইউআরএল কাঠামো" ব্যবহার করতে চান তা কী বিশদভাবে বর্ণনা করতে পারেন? লিঙ্কযুক্ত উদাহরণ 4-10 এর উপর ভিত্তি করে আপনার নিজের ঘূর্ণায়মান একটি তুচ্ছ প্রচেষ্টা বলে মনে হচ্ছে। আমি নিজে অনেক সময় এটি করেছি ...
স্টু থম্পসন

আমি ইউআরএল কাঠামোটি বিস্তৃত করতে আমার প্রশ্ন সম্পাদনা করেছি। এবং হ্যাঁ, আমি আমার নিজের সার্লেটটি ঘুরিয়ে শেষ করেছি। আমার উত্তর নীচে দেখুন।
ব্রুনো ডি ফ্রেইন

1
আপনি স্থির সামগ্রীর জন্য ওয়েবসারভারটি কেন ব্যবহার করবেন না?
স্টিফেন 19

4
@ স্টেফেন: কারণ টমকেট / জেটির সামনে সবসময় অ্যাপাচি থাকে না। এবং একটি পৃথক কনফিগারেশন ঝামেলা এড়াতে। তবে আপনি ঠিক বলেছেন, আমি সেই বিকল্পটি বিবেচনা করতে পারি।
ব্রুনো ডি ফ্রেইন

আমি কেবল বুঝতে পারছি না, কেন আপনি এই <servlet-mapping> <servlet-name> ডিফল্ট </servlet-name> <url- Pattern> / </url-pattern> </ সার্লেট-ম্যাপিংয়ের মতো ম্যাপিং ব্যবহার করেন নি > স্থিতিশীল বিষয়বস্তু পরিবেশন করতে
ম্যাকিক ক্রেফ্ট

উত্তর:


53

আমি কিছুটা ভিন্ন সমাধান নিয়ে এসেছি। এটি কিছুটা হ্যাক-ইশ, তবে এখানে ম্যাপিংটি রয়েছে:

<servlet-mapping>   
    <servlet-name>default</servlet-name>
    <url-pattern>*.html</url-pattern>
</servlet-mapping>
<servlet-mapping>
    <servlet-name>default</servlet-name>
    <url-pattern>*.jpg</url-pattern>
</servlet-mapping>
<servlet-mapping>
 <servlet-name>default</servlet-name>
    <url-pattern>*.png</url-pattern>
</servlet-mapping>
<servlet-mapping>
    <servlet-name>default</servlet-name>
    <url-pattern>*.css</url-pattern>
</servlet-mapping>
<servlet-mapping>
    <servlet-name>default</servlet-name>
    <url-pattern>*.js</url-pattern>
</servlet-mapping>

<servlet-mapping>
    <servlet-name>myAppServlet</servlet-name>
    <url-pattern>/</url-pattern>
</servlet-mapping>

এটি মূলত কেবলমাত্র সমস্ত কন্টেন্ট ফাইলকে ডিফল্ট সার্লেটে এক্সটেনশান করে এবং "মাইএপসার্ভালেট" - এ অন্য সমস্ত কিছু ম্যাপ করে।

এটি জেটি এবং টমক্যাট উভয় ক্ষেত্রেই কাজ করে।


13
সার্ভলেট-ম্যাপিংয়ের ভিতরে আপনি একাধিক ইউআরএল-প্যাটার্ন ট্যাগ যুক্ত করতে পারেন;)
ফারিদ অ্যালামনোউটি

5
সার্লেলেট 2.5 এবং আরও নতুন সার্ভলেট-ম্যাপিংয়ের ভিতরে একাধিক url- প্যাটার্ন ট্যাগগুলি সমর্থন করে
vided_voidgroup

সূচী ফাইলগুলির সাথে কেবল সতর্কতা অবলম্বন করুন (index.html) যেহেতু তারা আপনার সার্লেটের চেয়ে অগ্রাধিকার নিতে পারে।
Andres

আমি মনে করি এটি খারাপ ধারণা ব্যবহার *.sth। যদি কেউ url পান তবে example.com/index.jsp?g=.sthতিনি jsp ফাইলের উত্স পাবেন। নাকি আমি ভুল? (আমি জাভা ইই তে নতুন) আমি সাধারণত ইউআরএল প্যাটার্ন /css/*ইত্যাদি ব্যবহার করি
SemperPeritus

46

এক্ষেত্রে ডিফল্ট সার্লেটটি সম্পূর্ণ কাস্টম প্রয়োগের প্রয়োজন নেই, আপনি ধারকটির প্রয়োগের জন্য অনুরোধটি মোড়ানোর জন্য এই সাধারণ সার্লেটটি ব্যবহার করতে পারেন:


package com.example;

import java.io.*;

import javax.servlet.*;
import javax.servlet.http.*;

public class DefaultWrapperServlet extends HttpServlet
{   
    public void doGet(HttpServletRequest req, HttpServletResponse resp)
        throws ServletException, IOException
    {
        RequestDispatcher rd = getServletContext().getNamedDispatcher("default");

        HttpServletRequest wrapped = new HttpServletRequestWrapper(req) {
            public String getServletPath() { return ""; }
        };

        rd.forward(wrapped, resp);
    }
}

এই প্রশ্নের একটি নিয়ামককে / ম্যাপিংয়ের একটি ঝরঝরে উপায় এবং একটি ফিল্টার ব্যবহার করে স্থিতিশীল সামগ্রীতে স্ট্যাটিক। গৃহীত উত্তরটির পরে উত্সাহিত উত্তরটি দেখুন: স্ট্যাকওভারফ্লো
ডেভিড কার্বোনি ২


30

ফাইলসার্ভলেটের সাথে আমার ভাল ফলাফল হয়েছে , কারণ এটি বেশিরভাগ এইচটিটিপি সমর্থন করে (ইটাগস, চঙ্কিং, ইত্যাদি)।


ধন্যবাদ! কয়েক ঘন্টা ব্যর্থ চেষ্টা এবং খারাপ উত্তর, এবং এটি আমার সমস্যার সমাধান করে
Yossi Shasho

4
যদিও অ্যাপ্লিকেশনটির বাইরে কোনও ফোল্ডার থেকে সামগ্রী পরিবেশন করার জন্য (আমি এটি ডিস্কের ফোল্ডারটি সার্ভার করতে ব্যবহার করি, সি: \ সংস্থানগুলি বলুন) আমি এই সারিটি সংশোধন করেছি: this.basePath = getServletContext ()। getRealPath (getInitParameter ("বেসপথ)" ")); এবং এর সাথে এটি প্রতিস্থাপন করা হয়েছে: this.basePath = getInitParameter ("বেসপথ");
ইয়োসি শাশো

1
আপডেট করা সংস্করণ এ উপলব্ধ showcase.omnifaces.org/servlets/FileServlet
koppor

26

স্থিতিশীল সংস্থান সার্লেটের জন্য বিমূর্ত টেম্পলেট

আংশিকভাবে ২০০ blog সালের এই ব্লগের উপর ভিত্তি করে , এখানে একটি আধুনিক এবং উচ্চ পুনরায় ব্যবহারযোগ্য বিমূর্ত টেমপ্লেট রয়েছে যা সার্চলেটের জন্য যথাযথভাবে ক্যাচিংয়ের সাথে সম্পর্কিত হয় ETag, If-None-Matchএবং If-Modified-Since(তবে কোনও জিজিপ এবং রেঞ্জ সমর্থন নয়; কেবল এটিকে সহজ রাখার জন্য; জিজিপ একটি ফিল্টার দিয়ে বা এর মাধ্যমে করা যেতে পারে) ধারক কনফিগারেশন)।

public abstract class StaticResourceServlet extends HttpServlet {

    private static final long serialVersionUID = 1L;
    private static final long ONE_SECOND_IN_MILLIS = TimeUnit.SECONDS.toMillis(1);
    private static final String ETAG_HEADER = "W/\"%s-%s\"";
    private static final String CONTENT_DISPOSITION_HEADER = "inline;filename=\"%1$s\"; filename*=UTF-8''%1$s";

    public static final long DEFAULT_EXPIRE_TIME_IN_MILLIS = TimeUnit.DAYS.toMillis(30);
    public static final int DEFAULT_STREAM_BUFFER_SIZE = 102400;

    @Override
    protected void doHead(HttpServletRequest request, HttpServletResponse response) throws ServletException ,IOException {
        doRequest(request, response, true);
    }

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doRequest(request, response, false);
    }

    private void doRequest(HttpServletRequest request, HttpServletResponse response, boolean head) throws IOException {
        response.reset();
        StaticResource resource;

        try {
            resource = getStaticResource(request);
        }
        catch (IllegalArgumentException e) {
            response.sendError(HttpServletResponse.SC_BAD_REQUEST);
            return;
        }

        if (resource == null) {
            response.sendError(HttpServletResponse.SC_NOT_FOUND);
            return;
        }

        String fileName = URLEncoder.encode(resource.getFileName(), StandardCharsets.UTF_8.name());
        boolean notModified = setCacheHeaders(request, response, fileName, resource.getLastModified());

        if (notModified) {
            response.sendError(HttpServletResponse.SC_NOT_MODIFIED);
            return;
        }

        setContentHeaders(response, fileName, resource.getContentLength());

        if (head) {
            return;
        }

        writeContent(response, resource);
    }

    /**
     * Returns the static resource associated with the given HTTP servlet request. This returns <code>null</code> when
     * the resource does actually not exist. The servlet will then return a HTTP 404 error.
     * @param request The involved HTTP servlet request.
     * @return The static resource associated with the given HTTP servlet request.
     * @throws IllegalArgumentException When the request is mangled in such way that it's not recognizable as a valid
     * static resource request. The servlet will then return a HTTP 400 error.
     */
    protected abstract StaticResource getStaticResource(HttpServletRequest request) throws IllegalArgumentException;

    private boolean setCacheHeaders(HttpServletRequest request, HttpServletResponse response, String fileName, long lastModified) {
        String eTag = String.format(ETAG_HEADER, fileName, lastModified);
        response.setHeader("ETag", eTag);
        response.setDateHeader("Last-Modified", lastModified);
        response.setDateHeader("Expires", System.currentTimeMillis() + DEFAULT_EXPIRE_TIME_IN_MILLIS);
        return notModified(request, eTag, lastModified);
    }

    private boolean notModified(HttpServletRequest request, String eTag, long lastModified) {
        String ifNoneMatch = request.getHeader("If-None-Match");

        if (ifNoneMatch != null) {
            String[] matches = ifNoneMatch.split("\\s*,\\s*");
            Arrays.sort(matches);
            return (Arrays.binarySearch(matches, eTag) > -1 || Arrays.binarySearch(matches, "*") > -1);
        }
        else {
            long ifModifiedSince = request.getDateHeader("If-Modified-Since");
            return (ifModifiedSince + ONE_SECOND_IN_MILLIS > lastModified); // That second is because the header is in seconds, not millis.
        }
    }

    private void setContentHeaders(HttpServletResponse response, String fileName, long contentLength) {
        response.setHeader("Content-Type", getServletContext().getMimeType(fileName));
        response.setHeader("Content-Disposition", String.format(CONTENT_DISPOSITION_HEADER, fileName));

        if (contentLength != -1) {
            response.setHeader("Content-Length", String.valueOf(contentLength));
        }
    }

    private void writeContent(HttpServletResponse response, StaticResource resource) throws IOException {
        try (
            ReadableByteChannel inputChannel = Channels.newChannel(resource.getInputStream());
            WritableByteChannel outputChannel = Channels.newChannel(response.getOutputStream());
        ) {
            ByteBuffer buffer = ByteBuffer.allocateDirect(DEFAULT_STREAM_BUFFER_SIZE);
            long size = 0;

            while (inputChannel.read(buffer) != -1) {
                buffer.flip();
                size += outputChannel.write(buffer);
                buffer.clear();
            }

            if (resource.getContentLength() == -1 && !response.isCommitted()) {
                response.setHeader("Content-Length", String.valueOf(size));
            }
        }
    }

}

স্থিতিশীল সংস্থান উপস্থাপন করে নীচের ইন্টারফেসের সাথে এটি একসাথে ব্যবহার করুন।

interface StaticResource {

    /**
     * Returns the file name of the resource. This must be unique across all static resources. If any, the file
     * extension will be used to determine the content type being set. If the container doesn't recognize the
     * extension, then you can always register it as <code>&lt;mime-type&gt;</code> in <code>web.xml</code>.
     * @return The file name of the resource.
     */
    public String getFileName();

    /**
     * Returns the last modified timestamp of the resource in milliseconds.
     * @return The last modified timestamp of the resource in milliseconds.
     */
    public long getLastModified();

    /**
     * Returns the content length of the resource. This returns <code>-1</code> if the content length is unknown.
     * In that case, the container will automatically switch to chunked encoding if the response is already
     * committed after streaming. The file download progress may be unknown.
     * @return The content length of the resource.
     */
    public long getContentLength();

    /**
     * Returns the input stream with the content of the resource. This method will be called only once by the
     * servlet, and only when the resource actually needs to be streamed, so lazy loading is not necessary.
     * @return The input stream with the content of the resource.
     * @throws IOException When something fails at I/O level.
     */
    public InputStream getInputStream() throws IOException;

}

আপনার যা দরকার তা কেবলমাত্র প্রদত্ত বিমূর্ত getStaticResource()সারলেট থেকে প্রসারিত করা এবং জাভাদোক অনুযায়ী পদ্ধতিটি প্রয়োগ করা ।

ফাইল সিস্টেম থেকে পরিবেষ্টিত কংক্রিট উদাহরণ:

এখানে একটি কংক্রিটের উদাহরণ যা /files/foo.extস্থানীয় ডিস্ক ফাইল সিস্টেমের মতো ইউআরএল এর মাধ্যমে এটি সরবরাহ করে :

@WebServlet("/files/*")
public class FileSystemResourceServlet extends StaticResourceServlet {

    private File folder;

    @Override
    public void init() throws ServletException {
        folder = new File("/path/to/the/folder");
    }

    @Override
    protected StaticResource getStaticResource(HttpServletRequest request) throws IllegalArgumentException {
        String pathInfo = request.getPathInfo();

        if (pathInfo == null || pathInfo.isEmpty() || "/".equals(pathInfo)) {
            throw new IllegalArgumentException();
        }

        String name = URLDecoder.decode(pathInfo.substring(1), StandardCharsets.UTF_8.name());
        final File file = new File(folder, Paths.get(name).getFileName().toString());

        return !file.exists() ? null : new StaticResource() {
            @Override
            public long getLastModified() {
                return file.lastModified();
            }
            @Override
            public InputStream getInputStream() throws IOException {
                return new FileInputStream(file);
            }
            @Override
            public String getFileName() {
                return file.getName();
            }
            @Override
            public long getContentLength() {
                return file.length();
            }
        };
    }

}

ডাটাবেস থেকে পরিবেষ্টিত কংক্রিট উদাহরণ:

এখানে একটি কংক্রিটের উদাহরণ যা এটির কোনও ইউআরএল এর /files/foo.extমাধ্যমে ডেটাবেস থেকে একটি ইজেবি পরিষেবা কলের মাধ্যমে পরিবেশন করে যা আপনার byte[] contentসম্পত্তি সত্তাকে ফিরিয়ে দেয় :

@WebServlet("/files/*")
public class YourEntityResourceServlet extends StaticResourceServlet {

    @EJB
    private YourEntityService yourEntityService;

    @Override
    protected StaticResource getStaticResource(HttpServletRequest request) throws IllegalArgumentException {
        String pathInfo = request.getPathInfo();

        if (pathInfo == null || pathInfo.isEmpty() || "/".equals(pathInfo)) {
            throw new IllegalArgumentException();
        }

        String name = URLDecoder.decode(pathInfo.substring(1), StandardCharsets.UTF_8.name());
        final YourEntity yourEntity = yourEntityService.getByName(name);

        return (yourEntity == null) ? null : new StaticResource() {
            @Override
            public long getLastModified() {
                return yourEntity.getLastModified();
            }
            @Override
            public InputStream getInputStream() throws IOException {
                return new ByteArrayInputStream(yourEntityService.getContentById(yourEntity.getId()));
            }
            @Override
            public String getFileName() {
                return yourEntity.getName();
            }
            @Override
            public long getContentLength() {
                return yourEntity.getContentLength();
            }
        };
    }

}

1
প্রিয় @BalusC আমি মনে করি আপনার পন্থা একটি হ্যাকার যারা নিম্নলিখিত অনুরোধ ফাইল সিস্টেম নালা নেভিগেট পারে পাঠানোর প্রবন হল: files/%2e%2e/mysecretfile.txt। এই অনুরোধ উত্পাদন করে files/../mysecretfile.txt। আমি এটি টমক্যাট 7.0.55 এ পরীক্ষা করেছি। তারা একে ডিরেক্টরিকে আরোহণের ডাক দেয়: owasp.org/index.php/Path_Traversal
ক্রিশ্চিয়ান

1
@ ক্রিশ্চিয়ান: হ্যাঁ, সম্ভব কীভাবে এটি প্রতিরোধ করা যায় তা দেখানোর জন্য উদাহরণটি আপডেট করেছি।
বালাসসি

এটি upvotes পাওয়া উচিত নয়। সার্ভলেটের সাথে ওয়েবপেজের জন্য স্ট্যাটিক ফাইলগুলি পরিবেশন করা বিপর্যয় সুরক্ষা অনুসারে একটি রেসিপি। এই ধরনের সমস্ত সমস্যা ইতিমধ্যে সমাধান করা হয়েছে এবং সম্ভবত আরও অনাবৃত নিরাপত্তা টাইম বোমাগুলি বন্ধ হয়ে যাওয়ার সাথে একটি নতুন কাস্টম পন্থা বাস্তবায়নের কোনও কারণ নেই। সঠিক পথটি হ'ল টমকেট / গ্লাস ফিশ / জেটি ইত্যাদি সামগ্রীটি পরিবেশন করতে, বা এনজিএনএক্সের মতো ডেডিকেটেড ফাইলসভার ব্যবহার করা আরও ভাল।
লিওনহার্ড প্রিন্টজ

@ লিওনহার্ডিনেজেটজ: আমি সুরক্ষা সম্পর্কিত বিষয়গুলি চিহ্নিত করার পরে আমি উত্তরটি মুছে ফেলব এবং টমক্যাটে আমার বন্ধুদের কাছে ফিরে রিপোর্ট করব। সমস্যা নেই.
বালুসসি

19

আমি আমার নিজের ঘূর্ণায়মান শেষ StaticServlet। এটি If-Modified-Sinceজিজেপ এনকোডিং সমর্থন করে এবং যুদ্ধ-ফাইল থেকে স্থির ফাইলগুলি সরবরাহ করতে সক্ষম হওয়া উচিত। এটি খুব কঠিন কোড নয়, তবে এটি সম্পূর্ণরূপে তুচ্ছও নয়।

কোডটি উপলব্ধ: স্ট্যাটিক সার্ভেল.জভা । মতামত নির্দ্বিধায়।

আপডেট: খুররাম ServletUtilsরেফারেন্সযুক্ত ক্লাস সম্পর্কে জিজ্ঞাসা করেছেন StaticServlet। এটি কেবলমাত্র আমার প্রকল্পের জন্য সহায়ক পদ্ধতির সাথে ক্লাস। আপনার কেবলমাত্র পদ্ধতিটি প্রয়োজন coalesce(যা এসকিউএল ফাংশনের সাথে অভিন্ন COALESCE)। এই কোড:

public static <T> T coalesce(T...ts) {
    for(T t: ts)
        if(t != null)
            return t;
    return null;
}

2
আপনার অভ্যন্তর শ্রেণীর ত্রুটির নাম রাখবেন না। এটি জাভা.এলং এর জন্য ভুল করতে পারে বলে বিভ্রান্তির সৃষ্টি হতে পারে r এররও এছাড়াও, আপনার ওয়েব.এক্সএমএল কি একই?
লিওনেল

ত্রুটি সতর্কতার জন্য ধন্যবাদ। ওয়েব.এক্সএমএল হ'ল স্ট্যাটিক সার্লেটের নাম দ্বারা "ডিফল্ট" প্রতিস্থাপন করা।
ব্রুনো ডি ফ্রেইন

1
কোলেসেস পদ্ধতিটি হিসাবে, এটি কমর্স-ল্যাং স্ট্রিংইটিলস.ডেফল্টস্ট্রিং (স্ট্রিং, স্ট্রিং) দ্বারা প্রতিস্থাপন করা যেতে পারে (সার্ভারলেট শ্রেণির অভ্যন্তরে)
মাইক মিনিকি

ট্রান্সফার স্ট্রিমগুলি () পদ্ধতিটি ফাইলসকপি (যেমন, ওএস) এর সাথেও প্রতিস্থাপন করা যেতে পারে;
জেরিট ব্রিংক

কেন এই পদ্ধতির এত জনপ্রিয়? লোকেরা স্ট্যাটিক ফাইল সার্ভারগুলিকে কেন এইভাবে পুনর্নির্মাণ করছে? কেবলমাত্র আবিষ্কারের জন্য অপেক্ষা করা অনেকগুলি সুরক্ষা গর্ত রয়েছে এবং বাস্তব স্ট্যাটিক ফাইল সার্ভারের এমন অনেকগুলি বৈশিষ্ট্য যা প্রয়োগ করা হয়নি।
লিওনহার্ড প্রিন্টজ

12

উপরের উদাহরণের তথ্য থেকে বিচার করে, আমার মনে হয় এই পুরো নিবন্ধটি টমক্যাট 6.0.29 এবং তার আগেরের একটি বাগড আচরণের উপর ভিত্তি করে। Https://issues.apache.org/bugzilla/show_bug.cgi?id=50026 দেখুন । টমক্যাট 6.0.30 এ আপগ্রেড করুন এবং (টমক্যাট | জেটি) এর মধ্যে আচরণটি মার্জ হওয়া উচিত।


1
এটিও আমার বোঝার svn diff -c1056763 http://svn.apache.org/repos/asf/tomcat/tc6.0.x/trunk/। শেষ অবধি, +3 বছর আগে এই ডাব্লুউন্টফিক্স চিহ্নিত করার পরে!
ব্রুনো ডি ফ্রেইন

12

এটা চেষ্টা কর

<servlet-mapping>
    <servlet-name>default</servlet-name>
    <url-pattern>*.js</url-pattern>
    <url-pattern>*.css</url-pattern>
    <url-pattern>*.ico</url-pattern>
    <url-pattern>*.png</url-pattern>
    <url-pattern>*.jpg</url-pattern>
    <url-pattern>*.htc</url-pattern>
    <url-pattern>*.gif</url-pattern>
</servlet-mapping>    

সম্পাদনা করুন: এটি কেবল সার্লেলেট 2.5 স্পেস এবং তার জন্য বৈধ।


দেখে মনে হচ্ছে এটি কোনও বৈধ কনফিগার নয়।
গেড্রাক্স

10

আমার একই সমস্যা ছিল এবং আমি টমক্যাট কোডবেস থেকে 'ডিফল্ট সার্লেট' কোড ব্যবহার করে এটি সমাধান করেছি।

https://github.com/apache/tomcat/blob/master/java/org/apache/catalina/servlets/DefaultServlet.java

DefaultServlet সার্ভলেট যে হুল বিড়াল স্ট্যাটিক সম্পদ (JPG, HTML, CSS, GIF ইত্যাদি) তোলে।

এই সার্লেটটি বেশ দক্ষ এবং আপনি উপরে বর্ণিত কিছু বৈশিষ্ট্য রয়েছে।

আমি মনে করি যে এই উত্স কোডটি আপনার প্রয়োজন নেই এমন কার্যকারিতা বা অবক্ষয় শুরু করার এবং অপসারণ করার একটি ভাল উপায়।

  • Org.apache.naming.res উত্স প্যাকেজ এর উল্লেখ java.io.File কোড দিয়ে মুছে ফেলা বা প্রতিস্থাপন করা যেতে পারে।
  • Org.apache.catalina.util প্যাকেজের উল্লেখগুলি কেবলমাত্র ইউটিলিটি পদ্ধতি / শ্রেণি যা আপনার উত্স কোডে নকল করা যায়।
  • Org.apache.catalina.Globals শ্রেণীর রেফারেন্স অন্তর্ভুক্ত বা সরানো যেতে পারে।

এটি থেকে প্রচুর স্টাফের উপর নির্ভর করে বলে মনে হচ্ছে org.apache.*। জেটির সাথে আপনি এটি কীভাবে ব্যবহার করতে পারেন?
ব্রুনো ডি ফ্রেইন

আপনি ঠিক, এই সংস্করণে হুল বিড়াল অনেকগুলি depedencies (caand এটি অনেক কিছু আপনি চান না পারে সমর্থন আমি আমার উত্তর সম্পাদনা
Panagiotis Korros


4

আমি হুল বিড়াল ব্যাপ্ত দ্বারা এই করেনি DefaultServlet ( src ) এবং getRelativePath () মেথড অগ্রাহ্য।

package com.example;

import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import org.apache.catalina.servlets.DefaultServlet;

public class StaticServlet extends DefaultServlet
{
   protected String pathPrefix = "/static";

   public void init(ServletConfig config) throws ServletException
   {
      super.init(config);

      if (config.getInitParameter("pathPrefix") != null)
      {
         pathPrefix = config.getInitParameter("pathPrefix");
      }
   }

   protected String getRelativePath(HttpServletRequest req)
   {
      return pathPrefix + super.getRelativePath(req);
   }
}

... এবং এখানে আমার সার্লেট মানচিত্রগুলি রয়েছে

<servlet>
    <servlet-name>StaticServlet</servlet-name>
    <servlet-class>com.example.StaticServlet</servlet-class>
    <init-param>
        <param-name>pathPrefix</param-name>
        <param-value>/static</param-value>
    </init-param>       
</servlet>

<servlet-mapping>
    <servlet-name>StaticServlet</servlet-name>
    <url-pattern>/static/*</url-pattern>
</servlet-mapping>  

1

একটি স্প্রিং অ্যাপ্লিকেশন পাশাপাশি /favicon.ico এবং / WEB-INF / jsp / * এর জেএসপি ফাইলগুলি থেকে সমস্ত অনুরোধগুলি পরিবেশন করতে যে স্প্রিংয়ের অ্যাবস্ট্রাক্টরালবেসড ভিউ আপনাকে অনুরোধ করবে যে আপনি কেবল জেএসপি সার্লেট এবং ডিফল্ট সার্লেটটি পুনরায় তৈরি করতে পারেন:

  <servlet>
    <servlet-name>springapp</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <load-on-startup>1</load-on-startup>
  </servlet>

  <servlet-mapping>
    <servlet-name>jsp</servlet-name>
    <url-pattern>/WEB-INF/jsp/*</url-pattern>
  </servlet-mapping>

  <servlet-mapping>
    <servlet-name>default</servlet-name>
    <url-pattern>/favicon.ico</url-pattern>
  </servlet-mapping>

  <servlet-mapping>
    <servlet-name>springapp</servlet-name>
    <url-pattern>/*</url-pattern>
  </servlet-mapping>

আমরা জেএসপি সার্লেটের মানক ম্যাপিংয়ের উপর * .jsp ইউআরএল-প্যাটার্নের উপর নির্ভর করতে পারি না কারণ কোনও এক্সটেনশন ম্যাপিং চেক করার আগে পাথ প্যাটার্ন '/ *' মেলানো হয়। আরও গভীর ফোল্ডারে জেএসপি সার্লেটকে ম্যাপিং করার অর্থ এটি প্রথমে মিলছে। '/Favicon.ico' ম্যাচটি পাথ প্যাটার্ন ম্যাচের আগে ঠিক ঘটে। গভীর পাথ ম্যাচগুলি, বা সঠিক মিলগুলি কাজ করবে, তবে কোনও এক্সটেনশান ম্যাচ এটি '/ *' পাথের ম্যাচটিকে অতিক্রম করতে পারে না। ডিফল্ট সার্লেটে '/' ম্যাপিং কাজ করে না appear আপনি ভাবেন যে ঠিক '/' স্প্রিংঅ্যাপে '/ *' প্যাটার্ন প্যাটার্নটিকে পরাজিত করবে।

উপরের ফিল্টার সমাধানটি আবেদন থেকে ফরওয়ার্ড / অন্তর্ভুক্ত জেএসপি অনুরোধগুলির জন্য কাজ করে না। এটি কাজ করার জন্য আমাকে সরাসরি স্প্রিংঅ্যাপে ফিল্টারটি প্রয়োগ করতে হয়েছিল, যেখানে ইউআরএল-প্যাটার্নের মিলটি অকেজো ছিল কারণ অ্যাপ্লিকেশনটিতে যাওয়া সমস্ত অনুরোধগুলিও এর ফিল্টারগুলিতে যায়। সুতরাং আমি ফিল্টারটির সাথে প্যাটার্নের মিলটি যুক্ত করেছি এবং তারপরে 'jsp' সার্ভলেট সম্পর্কে জানতে পেরেছি এবং দেখেছি এটি ডিফল্ট সার্লেটের মতো পাথের উপসর্গটিকে সরিয়ে দেয় না। এটি আমার সমস্যার সমাধান করে, যা একরকম নয় তবে যথেষ্ট সাধারণ ছিল।


1

টমক্যাট ৮.x এর জন্য চেক করা হয়েছে: রুট সার্লেট মানচিত্রটিতে "" স্থিতিশীল সংস্থানগুলি ঠিক আছে। Servlet 3.x এর জন্য এটি দ্বারা সম্পন্ন হতে পারে@WebServlet("")


0

Org.mortbay.jetty.handler.ContextHandler ব্যবহার করুন। আপনার স্ট্যাটিক সার্লেটের মতো অতিরিক্ত উপাদানগুলির প্রয়োজন নেই।

জেটি বাড়িতে,

d সিডি প্রসঙ্গে

p সিপি জাভাদোক.এক্সএমএল স্ট্যাটিক.এক্সএমএল

$ vi স্ট্যাটিক.এক্সএমএল

...

<Configure class="org.mortbay.jetty.handler.ContextHandler">
<Set name="contextPath">/static</Set>
<Set name="resourceBase"><SystemProperty name="jetty.home" default="."/>/static/</Set>
<Set name="handler">
  <New class="org.mortbay.jetty.handler.ResourceHandler">
    <Set name="cacheControl">max-age=3600,public</Set>
  </New>
 </Set>
</Configure>

আপনার ইউআরএল উপসর্গের সাথে প্রসঙ্গপথের মান সেট করুন এবং স্থির সামগ্রীটির ফাইল পথ হিসাবে রিসোর্সবেসের মান সেট করুন।

এটা আমার জন্য কাজ করেছে।


আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.